2939 lines
106 KiB
C++
2939 lines
106 KiB
C++
/*
|
|
* Copyright 2011, Ben Langmead <langmea@cs.jhu.edu>
|
|
*
|
|
* This file is part of Bowtie 2.
|
|
*
|
|
* Bowtie 2 is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* Bowtie 2 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with Bowtie 2. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/*
|
|
* aligner_sw_driver.h
|
|
*
|
|
* REDUNDANT SEED HITS
|
|
*
|
|
* We say that two seed hits are redundant if they trigger identical
|
|
* seed-extend dynamic programming problems. Put another way, they both lie on
|
|
* the same diagonal of the overall read/reference dynamic programming matrix.
|
|
* Detecting redundant seed hits is simple when the seed hits are ungapped. We
|
|
* do this after offset resolution but before the offset is converted to genome
|
|
* coordinates (see uses of the seenDiags1_/seenDiags2_ fields for examples).
|
|
*
|
|
* REDUNDANT ALIGNMENTS
|
|
*
|
|
* In an unpaired context, we say that two alignments are redundant if they
|
|
* share any cells in the global DP table. Roughly speaking, this is like
|
|
* saying that two alignments are redundant if any read character aligns to the
|
|
* same reference character (same reference sequence, same strand, same offset)
|
|
* in both alignments.
|
|
*
|
|
* In a paired-end context, we say that two paired-end alignments are redundant
|
|
* if the mate #1s are redundant and the mate #2s are redundant.
|
|
*
|
|
* How do we enforce this? In the unpaired context, this is relatively simple:
|
|
* the cells from each alignment are checked against a set containing all cells
|
|
* from all previous alignments. Given a new alignment, for each cell in the
|
|
* new alignment we check whether it is in the set. If there is any overlap,
|
|
* the new alignment is rejected as redundant. Otherwise, the new alignment is
|
|
* accepted and its cells are added to the set.
|
|
*
|
|
* Enforcement in a paired context is a little trickier. Consider the
|
|
* following approaches:
|
|
*
|
|
* 1. Skip anchors that are redundant with any previous anchor or opposite
|
|
* alignment. This is sufficient to ensure no two concordant alignments
|
|
* found are redundant.
|
|
*
|
|
* 2. Same as scheme 1, but with a "transitive closure" scheme for finding all
|
|
* concordant pairs in the vicinity of an anchor. Consider the AB/AC
|
|
* scenario from the previous paragraph. If B is the anchor alignment, we
|
|
* will find AB but not AC. But under this scheme, once we find AB we then
|
|
* let B be a new anchor and immediately look for its opposites. Likewise,
|
|
* if we find any opposite, we make them anchors and continue searching. We
|
|
* don't stop searching until every opposite is used as an anchor.
|
|
*
|
|
* 3. Skip anchors that are redundant with any previous anchor alignment (but
|
|
* allow anchors that are redundant with previous opposite alignments).
|
|
* This isn't sufficient to avoid redundant concordant alignments. To avoid
|
|
* redundant concordants, we need an additional procedure that checks each
|
|
* new concordant alignment one-by-one against a list of previous concordant
|
|
* alignments to see if it is redundant.
|
|
*
|
|
* We take approach 1.
|
|
*/
|
|
|
|
#ifndef ALIGNER_SW_DRIVER_H_
|
|
#define ALIGNER_SW_DRIVER_H_
|
|
|
|
#include <iostream>
|
|
// -- BTL remove --
|
|
#include <stdlib.h>
|
|
#include <sys/time.h>
|
|
// -- --
|
|
#include <utility>
|
|
#include "ds.h"
|
|
#include "aligner_seed.h"
|
|
#include "aligner_sw.h"
|
|
#include "aligner_cache.h"
|
|
#include "reference.h"
|
|
#include "group_walk.h"
|
|
#include "gfm.h"
|
|
#include "mem_ids.h"
|
|
#include "aln_sink.h"
|
|
#include "pe.h"
|
|
#include "ival_list.h"
|
|
#include "simple_func.h"
|
|
#include "random_util.h"
|
|
#include "dp_framer.h"
|
|
|
|
using namespace std;
|
|
|
|
template <typename index_t>
|
|
struct SeedPos {
|
|
|
|
SeedPos() : fw(false), offidx(0), rdoff(0), seedlen(0) { }
|
|
|
|
SeedPos(
|
|
bool fw_,
|
|
index_t offidx_,
|
|
index_t rdoff_,
|
|
index_t seedlen_)
|
|
{
|
|
init(fw_, offidx_, rdoff_, seedlen_);
|
|
}
|
|
|
|
void init(
|
|
bool fw_,
|
|
index_t offidx_,
|
|
index_t rdoff_,
|
|
index_t seedlen_)
|
|
{
|
|
fw = fw_;
|
|
offidx = offidx_;
|
|
rdoff = rdoff_;
|
|
seedlen = seedlen_;
|
|
}
|
|
|
|
bool operator<(const SeedPos& o) const {
|
|
if(offidx < o.offidx) return true;
|
|
if(offidx > o.offidx) return false;
|
|
if(rdoff < o.rdoff) return true;
|
|
if(rdoff > o.rdoff) return false;
|
|
if(seedlen < o.seedlen) return true;
|
|
if(seedlen > o.seedlen) return false;
|
|
if(fw && !o.fw) return true;
|
|
if(!fw && o.fw) return false;
|
|
return false;
|
|
}
|
|
|
|
bool operator>(const SeedPos& o) const {
|
|
if(offidx < o.offidx) return false;
|
|
if(offidx > o.offidx) return true;
|
|
if(rdoff < o.rdoff) return false;
|
|
if(rdoff > o.rdoff) return true;
|
|
if(seedlen < o.seedlen) return false;
|
|
if(seedlen > o.seedlen) return true;
|
|
if(fw && !o.fw) return false;
|
|
if(!fw && o.fw) return true;
|
|
return false;
|
|
}
|
|
|
|
bool operator==(const SeedPos& o) const {
|
|
return fw == o.fw && offidx == o.offidx &&
|
|
rdoff == o.rdoff && seedlen == o.seedlen;
|
|
}
|
|
|
|
bool fw;
|
|
index_t offidx;
|
|
index_t rdoff;
|
|
index_t seedlen;
|
|
};
|
|
|
|
/**
|
|
* An SATuple along with the associated seed position.
|
|
*/
|
|
template <typename index_t>
|
|
struct SATupleAndPos {
|
|
|
|
SATuple<index_t> sat; // result for this seed hit
|
|
SeedPos<index_t> pos; // seed position that yielded the range this was taken from
|
|
index_t origSz; // size of range this was taken from
|
|
index_t nlex; // # position we can extend seed hit to left w/o edit
|
|
index_t nrex; // # position we can extend seed hit to right w/o edit
|
|
|
|
bool operator<(const SATupleAndPos& o) const {
|
|
if(sat < o.sat) return true;
|
|
if(sat > o.sat) return false;
|
|
return pos < o.pos;
|
|
}
|
|
|
|
bool operator==(const SATupleAndPos& o) const {
|
|
return sat == o.sat && pos == o.pos;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Encapsulates the weighted random sampling scheme we want to use to pick
|
|
* which seed hit range to sample a row from.
|
|
*/
|
|
template <typename index_t>
|
|
class RowSampler {
|
|
|
|
public:
|
|
|
|
RowSampler(int cat = 0) : elim_(cat), masses_(cat) {
|
|
mass_ = 0.0f;
|
|
}
|
|
|
|
/**
|
|
* Initialze sampler with respect to a range of elements in a list of
|
|
* SATupleAndPos's.
|
|
*/
|
|
void init(
|
|
const EList<SATupleAndPos<index_t>, 16>& salist,
|
|
index_t sai,
|
|
index_t saf,
|
|
bool lensq, // whether to square the numerator, which = extended length
|
|
bool szsq) // whether to square denominator, which =
|
|
{
|
|
assert_gt(saf, sai);
|
|
elim_.resize(saf - sai);
|
|
elim_.fill(false);
|
|
// Initialize mass
|
|
mass_ = 0.0f;
|
|
masses_.resize(saf - sai);
|
|
for(index_t i = sai; i < saf; i++) {
|
|
index_t len = salist[i].nlex + salist[i].nrex + 1; // + salist[i].sat.key.len;
|
|
double num = (double)len;
|
|
if(lensq) {
|
|
num *= num;
|
|
}
|
|
double denom = (double)salist[i].sat.size();
|
|
if(szsq) {
|
|
denom *= denom;
|
|
}
|
|
masses_[i - sai] = num / denom;
|
|
mass_ += masses_[i - sai];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Caller is indicating that the bin at index i is exhausted and we should
|
|
* exclude it from our sampling from now on.
|
|
*/
|
|
void finishedRange(index_t i) {
|
|
assert_lt(i, masses_.size());
|
|
elim_[i] = true;
|
|
mass_ -= masses_[i];
|
|
}
|
|
|
|
/**
|
|
* Sample randomly from the mass.
|
|
*/
|
|
size_t next(RandomSource& rnd) {
|
|
// Throw the dart
|
|
double rd = rnd.nextFloat() * mass_;
|
|
double mass_sofar = 0.0f;
|
|
size_t sz = masses_.size();
|
|
size_t last_unelim = std::numeric_limits<size_t>::max();
|
|
for(size_t i = 0; i < sz; i++) {
|
|
if(!elim_[i]) {
|
|
last_unelim = i;
|
|
mass_sofar += masses_[i];
|
|
if(rd < mass_sofar) {
|
|
// This is the one we hit
|
|
return i;
|
|
}
|
|
}
|
|
}
|
|
assert_neq(std::numeric_limits<size_t>::max(), last_unelim);
|
|
return last_unelim;
|
|
}
|
|
|
|
protected:
|
|
double mass_; // total probability mass to throw darts at
|
|
EList<bool> elim_; // whether the range is eliminated
|
|
EList<double> masses_; // mass of each range
|
|
};
|
|
|
|
/**
|
|
* Return values from extendSeeds and extendSeedsPaired.
|
|
*/
|
|
enum {
|
|
// All end-to-end and seed hits were examined
|
|
// The policy does not need us to look any further
|
|
EXTEND_EXHAUSTED_CANDIDATES = 1,
|
|
EXTEND_POLICY_FULFILLED,
|
|
// We stopped because we reached a point where the only remaining
|
|
// alignments of interest have perfect scores, but we already investigated
|
|
// perfect alignments
|
|
EXTEND_PERFECT_SCORE,
|
|
// We stopped because we ran up against a limit on how much work we should
|
|
// do for one set of seed ranges, e.g. the limit on number of consecutive
|
|
// unproductive DP extensions
|
|
EXTEND_EXCEEDED_SOFT_LIMIT,
|
|
// We stopped because we ran up against a limit on how much work we should
|
|
// do for overall before giving up on a mate
|
|
EXTEND_EXCEEDED_HARD_LIMIT
|
|
};
|
|
|
|
/**
|
|
* Data structure encapsulating a range that's been extended out in two
|
|
* directions.
|
|
*/
|
|
struct ExtendRange {
|
|
|
|
void init(size_t off_, size_t len_, size_t sz_) {
|
|
off = off_; len = len_; sz = sz_;
|
|
}
|
|
|
|
size_t off; // offset of extended region
|
|
size_t len; // length between extremes of extended region
|
|
size_t sz; // # of elements in SA range
|
|
};
|
|
|
|
template <typename index_t>
|
|
class SwDriver {
|
|
|
|
typedef PList<index_t, CACHE_PAGE_SZ> TSAList;
|
|
|
|
public:
|
|
|
|
SwDriver(size_t bytes) :
|
|
satups_(DP_CAT),
|
|
gws_(DP_CAT),
|
|
seenDiags1_(DP_CAT),
|
|
seenDiags2_(DP_CAT),
|
|
redAnchor_(DP_CAT),
|
|
redMate1_(DP_CAT),
|
|
redMate2_(DP_CAT),
|
|
pool_(bytes, CACHE_PAGE_SZ, DP_CAT),
|
|
salistEe_(DP_CAT),
|
|
gwstate_(GW_CAT) { }
|
|
|
|
/**
|
|
* Given a collection of SeedHits for a single read, extend seed alignments
|
|
* into full alignments. Where possible, try to avoid redundant offset
|
|
* lookups and dynamic programming problems. Optionally report alignments
|
|
* to a AlnSinkWrap object as they are discovered.
|
|
*
|
|
* If 'reportImmediately' is true, returns true iff a call to
|
|
* mhs->report() returned true (indicating that the reporting
|
|
* policy is satisfied and we can stop). Otherwise, returns false.
|
|
*/
|
|
int extendSeeds(
|
|
Read& rd, // read to align
|
|
bool mate1, // true iff rd is mate #1
|
|
SeedResults<index_t>& sh, // seed hits to extend into full alignments
|
|
const GFM<index_t>& gfmFw, // BWT
|
|
const GFM<index_t>* gfmBw, // BWT'
|
|
const BitPairReference& ref, // Reference strings
|
|
SwAligner& swa, // dynamic programming aligner
|
|
const Scoring& sc, // scoring scheme
|
|
int seedmms, // # mismatches allowed in seed
|
|
int seedlen, // length of seed
|
|
int seedival, // interval between seeds
|
|
TAlScore& minsc, // minimum score for anchor
|
|
int nceil, // maximum # Ns permitted in ref portion
|
|
size_t maxhalf, // maximum width on one side of DP table
|
|
bool doUngapped, // do ungapped alignment
|
|
size_t maxIters, // stop after this many seed-extend loop iters
|
|
size_t maxUg, // max # ungapped extends
|
|
size_t maxDp, // max # DPs
|
|
size_t maxUgStreak, // stop after streak of this many ungap fails
|
|
size_t maxDpStreak, // stop after streak of this many dp fails
|
|
bool doExtend, // do seed extension
|
|
bool enable8, // use 8-bit SSE where possible
|
|
size_t cminlen, // use checkpointer if read longer than this
|
|
size_t cpow2, // interval between diagonals to checkpoint
|
|
bool doTri, // triangular mini-fills
|
|
int tighten, // -M score tightening mode
|
|
AlignmentCacheIface<index_t>& ca, // alignment cache for seed hits
|
|
RandomSource& rnd, // pseudo-random source
|
|
WalkMetrics& wlm, // group walk left metrics
|
|
SwMetrics& swmSeed, // DP metrics for seed-extend
|
|
PerReadMetrics& prm, // per-read metrics
|
|
AlnSinkWrap<index_t>* mhs, // HitSink for multiseed-style aligner
|
|
bool reportImmediately, // whether to report hits immediately to mhs
|
|
bool& exhaustive);
|
|
|
|
/**
|
|
* Given a collection of SeedHits for a read pair, extend seed
|
|
* alignments into full alignments and then look for the opposite
|
|
* mate using dynamic programming. Where possible, try to avoid
|
|
* redundant offset lookups. Optionally report alignments to a
|
|
* AlnSinkWrap object as they are discovered.
|
|
*
|
|
* If 'reportImmediately' is true, returns true iff a call to
|
|
* mhs->report() returned true (indicating that the reporting
|
|
* policy is satisfied and we can stop). Otherwise, returns false.
|
|
*/
|
|
int extendSeedsPaired(
|
|
Read& rd, // mate to align as anchor
|
|
Read& ord, // mate to align as opposite
|
|
bool anchor1, // true iff anchor mate is mate1
|
|
bool oppFilt, // true iff opposite mate was filtered out
|
|
SeedResults<index_t>& sh, // seed hits for anchor
|
|
const GFM<index_t>& gfmFw, // BWT
|
|
const GFM<index_t>* gfmBw, // BWT'
|
|
const BitPairReference& ref, // Reference strings
|
|
SwAligner& swa, // dyn programming aligner for anchor
|
|
SwAligner& swao, // dyn programming aligner for opposite
|
|
const Scoring& sc, // scoring scheme
|
|
const PairedEndPolicy& pepol,// paired-end policy
|
|
int seedmms, // # mismatches allowed in seed
|
|
int seedlen, // length of seed
|
|
int seedival, // interval between seeds
|
|
TAlScore& minsc, // minimum score for anchor
|
|
TAlScore& ominsc, // minimum score for opposite
|
|
int nceil, // max # Ns permitted in ref for anchor
|
|
int onceil, // max # Ns permitted in ref for opposite
|
|
bool nofw, // don't align forward read
|
|
bool norc, // don't align revcomp read
|
|
size_t maxhalf, // maximum width on one side of DP table
|
|
bool doUngapped, // do ungapped alignment
|
|
size_t maxIters, // stop after this many seed-extend loop iters
|
|
size_t maxUg, // max # ungapped extends
|
|
size_t maxDp, // max # DPs
|
|
size_t maxEeStreak, // stop after streak of this many end-to-end fails
|
|
size_t maxUgStreak, // stop after streak of this many ungap fails
|
|
size_t maxDpStreak, // stop after streak of this many dp fails
|
|
size_t maxMateStreak, // stop seed range after N mate-find fails
|
|
bool doExtend, // do seed extension
|
|
bool enable8, // use 8-bit SSE where possible
|
|
size_t cminlen, // use checkpointer if read longer than this
|
|
size_t cpow2, // interval between diagonals to checkpoint
|
|
bool doTri, // triangular mini-fills
|
|
int tighten, // -M score tightening mode
|
|
AlignmentCacheIface<index_t>& cs, // alignment cache for seed hits
|
|
RandomSource& rnd, // pseudo-random source
|
|
WalkMetrics& wlm, // group walk left metrics
|
|
SwMetrics& swmSeed, // DP metrics for seed-extend
|
|
SwMetrics& swmMate, // DP metrics for mate finidng
|
|
PerReadMetrics& prm, // per-read metrics for anchor
|
|
AlnSinkWrap<index_t>* msink, // AlnSink wrapper for multiseed-style aligner
|
|
bool swMateImmediately, // whether to look for mate immediately
|
|
bool reportImmediately, // whether to report hits immediately to msink
|
|
bool discord, // look for discordant alignments?
|
|
bool mixed, // look for unpaired as well as paired alns?
|
|
bool& exhaustive);
|
|
|
|
/**
|
|
* Prepare for a new read.
|
|
*/
|
|
void nextRead(bool paired, size_t mate1len, size_t mate2len) {
|
|
redAnchor_.reset();
|
|
seenDiags1_.reset();
|
|
seenDiags2_.reset();
|
|
seedExRangeFw_[0].clear(); // mate 1 fw
|
|
seedExRangeFw_[1].clear(); // mate 2 fw
|
|
seedExRangeRc_[0].clear(); // mate 1 rc
|
|
seedExRangeRc_[1].clear(); // mate 2 rc
|
|
size_t maxlen = mate1len;
|
|
if(paired) {
|
|
redMate1_.reset();
|
|
redMate1_.init(mate1len);
|
|
redMate2_.reset();
|
|
redMate2_.init(mate2len);
|
|
if(mate2len > maxlen) {
|
|
maxlen = mate2len;
|
|
}
|
|
}
|
|
redAnchor_.init(maxlen);
|
|
}
|
|
|
|
protected:
|
|
|
|
bool eeSaTups(
|
|
const Read& rd, // read
|
|
SeedResults<index_t>& sh, // seed hits to extend into full alignments
|
|
const GFM<index_t>& gfm, // BWT
|
|
const BitPairReference& ref, // Reference strings
|
|
RandomSource& rnd, // pseudo-random generator
|
|
WalkMetrics& wlm, // group walk left metrics
|
|
SwMetrics& swmSeed, // metrics for seed extensions
|
|
index_t& nelt_out, // out: # elements total
|
|
index_t maxelts, // max # elts to report
|
|
bool all); // report all hits?
|
|
|
|
void extend(
|
|
const Read& rd, // read
|
|
const GFM<index_t>& gfmFw, // Forward Bowtie index
|
|
const GFM<index_t>* gfmBw, // Backward Bowtie index
|
|
index_t topf, // top in fw index
|
|
index_t botf, // bot in fw index
|
|
index_t topb, // top in bw index
|
|
index_t botb, // bot in bw index
|
|
bool fw, // seed orientation
|
|
index_t off, // seed offset from 5' end
|
|
index_t len, // seed length
|
|
PerReadMetrics& prm, // per-read metrics
|
|
index_t& nlex, // # positions we can extend to left w/o edit
|
|
index_t& nrex); // # positions we can extend to right w/o edit
|
|
|
|
void prioritizeSATups(
|
|
const Read& rd, // read
|
|
SeedResults<index_t>& sh, // seed hits to extend into full alignments
|
|
const GFM<index_t>& gfmFw, // BWT
|
|
const GFM<index_t>* gfmBw, // BWT'
|
|
const BitPairReference& ref, // Reference strings
|
|
int seedmms, // # seed mismatches allowed
|
|
index_t maxelt, // max elts we'll consider
|
|
bool doExtend, // extend out seeds
|
|
bool lensq, // square extended length
|
|
bool szsq, // square SA range size
|
|
index_t nsm, // if range as <= nsm elts, it's "small"
|
|
AlignmentCacheIface<index_t>& ca, // alignment cache for seed hits
|
|
RandomSource& rnd, // pseudo-random generator
|
|
WalkMetrics& wlm, // group walk left metrics
|
|
PerReadMetrics& prm, // per-read metrics
|
|
index_t& nelt_out, // out: # elements total
|
|
bool all); // report all hits?
|
|
|
|
Random1toN rand_; // random number generators
|
|
EList<Random1toN, 16> rands_; // random number generators
|
|
EList<Random1toN, 16> rands2_; // random number generators
|
|
EList<EEHit<index_t>, 16> eehits_; // holds end-to-end hits
|
|
EList<SATupleAndPos<index_t>, 16> satpos_; // holds SATuple, SeedPos pairs
|
|
EList<SATupleAndPos<index_t>, 16> satpos2_; // holds SATuple, SeedPos pairs
|
|
EList<SATuple<index_t>, 16> satups_; // holds SATuples to explore elements from
|
|
EList<GroupWalk2S<index_t, TSlice, 16> > gws_; // list of GroupWalks; no particular order
|
|
EList<size_t> mateStreaks_; // mate-find fail streaks
|
|
RowSampler<index_t> rowsamp_; // row sampler
|
|
|
|
// Ranges that we've extended through when extending seed hits
|
|
EList<ExtendRange> seedExRangeFw_[2];
|
|
EList<ExtendRange> seedExRangeRc_[2];
|
|
|
|
// Data structures encapsulating the diagonals that have already been used
|
|
// to seed alignment for mate 1 and mate 2.
|
|
EIvalMergeListBinned seenDiags1_;
|
|
EIvalMergeListBinned seenDiags2_;
|
|
|
|
// For weeding out redundant alignments
|
|
RedundantAlns redAnchor_; // database of cells used for anchor alignments
|
|
RedundantAlns redMate1_; // database of cells used for mate 1 alignments
|
|
RedundantAlns redMate2_; // database of cells used for mate 2 alignments
|
|
|
|
// For holding results for anchor (res_) and opposite (ores_) mates
|
|
SwResult resGap_; // temp holder for alignment result
|
|
SwResult oresGap_; // temp holder for alignment result, opp mate
|
|
SwResult resUngap_; // temp holder for ungapped alignment result
|
|
SwResult oresUngap_; // temp holder for ungap. aln. opp mate
|
|
SwResult resEe_; // temp holder for ungapped alignment result
|
|
SwResult oresEe_; // temp holder for ungap. aln. opp mate
|
|
|
|
Pool pool_; // memory pages for salistExact_
|
|
TSAList salistEe_; // PList for offsets for end-to-end hits
|
|
GroupWalkState<index_t> gwstate_; // some per-thread state shared by all GroupWalks
|
|
|
|
// For AlnRes::matchesRef:
|
|
ASSERT_ONLY(SStringExpandable<char> raw_refbuf_);
|
|
ASSERT_ONLY(SStringExpandable<uint32_t> raw_destU32_);
|
|
ASSERT_ONLY(EList<bool> raw_matches_);
|
|
ASSERT_ONLY(BTDnaString tmp_rf_);
|
|
ASSERT_ONLY(BTDnaString tmp_rdseq_);
|
|
ASSERT_ONLY(BTString tmp_qseq_);
|
|
ASSERT_ONLY(EList<index_t> tmp_reflens_);
|
|
ASSERT_ONLY(EList<index_t> tmp_refoffs_);
|
|
};
|
|
|
|
#define TIMER_START() \
|
|
struct timeval tv_i, tv_f; \
|
|
struct timezone tz_i, tz_f; \
|
|
size_t total_usecs; \
|
|
gettimeofday(&tv_i, &tz_i)
|
|
|
|
#define IF_TIMER_END() \
|
|
gettimeofday(&tv_f, &tz_f); \
|
|
total_usecs = \
|
|
(tv_f.tv_sec - tv_i.tv_sec) * 1000000 + (tv_f.tv_usec - tv_i.tv_usec); \
|
|
if(total_usecs > 300000)
|
|
|
|
/*
|
|
* aligner_sw_driver.cpp
|
|
*
|
|
* Routines that drive the alignment process given a collection of seed hits.
|
|
* This is generally done in a few stages: extendSeeds visits the set of
|
|
* seed-hit BW elements in some order; for each element visited it resolves its
|
|
* reference offset; once the reference offset is known, bounds for a dynamic
|
|
* programming subproblem are established; if these bounds are distinct from
|
|
* the bounds we've already tried, we solve the dynamic programming subproblem
|
|
* and report the hit; if the AlnSinkWrap indicates that we can stop, we
|
|
* return, otherwise we continue on to the next BW element.
|
|
*/
|
|
|
|
|
|
/**
|
|
* Given end-to-end alignment results stored in the SeedResults structure, set
|
|
* up all of our state for resolving and keeping track of reference offsets for
|
|
* hits. Order the list of ranges to examine such that all exact end-to-end
|
|
* alignments are examined before any 1mm end-to-end alignments.
|
|
*
|
|
* Note: there might be a lot of hits and a lot of wide ranges to look for
|
|
* here. We use 'maxelt'.
|
|
*/
|
|
template <typename index_t>
|
|
bool SwDriver<index_t>::eeSaTups(
|
|
const Read& rd, // read
|
|
SeedResults<index_t>& sh, // seed hits to extend into full alignments
|
|
const GFM<index_t>& gfm, // BWT
|
|
const BitPairReference& ref, // Reference strings
|
|
RandomSource& rnd, // pseudo-random generator
|
|
WalkMetrics& wlm, // group walk left metrics
|
|
SwMetrics& swmSeed, // metrics for seed extensions
|
|
index_t& nelt_out, // out: # elements total
|
|
index_t maxelt, // max elts we'll consider
|
|
bool all) // report all hits?
|
|
{
|
|
assert_eq(0, nelt_out);
|
|
gws_.clear();
|
|
rands_.clear();
|
|
satpos_.clear();
|
|
eehits_.clear();
|
|
// First, count up the total number of satpos_, rands_, eehits_, and gws_
|
|
// we're going to tuse
|
|
index_t nobj = 0;
|
|
if(!sh.exactFwEEHit().empty()) nobj++;
|
|
if(!sh.exactRcEEHit().empty()) nobj++;
|
|
nobj += sh.mm1EEHits().size();
|
|
nobj = min(nobj, maxelt);
|
|
gws_.ensure(nobj);
|
|
rands_.ensure(nobj);
|
|
satpos_.ensure(nobj);
|
|
eehits_.ensure(nobj);
|
|
index_t tot = sh.exactFwEEHit().size() + sh.exactRcEEHit().size();
|
|
bool succ = false;
|
|
bool firstEe = true;
|
|
bool done = false;
|
|
if(tot > 0) {
|
|
bool fwFirst = true;
|
|
// Pick fw / rc to go first in a weighted random fashion
|
|
#ifdef BOWTIE_64BIT_INDEX
|
|
index_t rn64 = rnd.nextU64();
|
|
index_t rn = rn64 % (uint64_t)tot;
|
|
#else
|
|
index_t rn32 = rnd.nextU32();
|
|
index_t rn = rn32 % (uint32_t)tot;
|
|
#endif
|
|
if(rn >= sh.exactFwEEHit().size()) {
|
|
fwFirst = false;
|
|
}
|
|
for(int fwi = 0; fwi < 2 && !done; fwi++) {
|
|
bool fw = ((fwi == 0) == fwFirst);
|
|
EEHit<index_t> hit = fw ? sh.exactFwEEHit() : sh.exactRcEEHit();
|
|
if(hit.empty()) {
|
|
continue;
|
|
}
|
|
assert(hit.fw == fw);
|
|
if(hit.bot > hit.top) {
|
|
// Possibly adjust bot and width if we would have exceeded maxelt
|
|
index_t tops[2] = { hit.top, 0 };
|
|
index_t bots[2] = { hit.bot, 0 };
|
|
index_t width = hit.bot - hit.top;
|
|
if(nelt_out + width > maxelt) {
|
|
index_t trim = (index_t)((nelt_out + width) - maxelt);
|
|
#ifdef BOWTIE_64BIT_INDEX
|
|
index_t rn = rnd.nextU64() % width;
|
|
#else
|
|
index_t rn = rnd.nextU32() % width;
|
|
#endif
|
|
index_t newwidth = width - trim;
|
|
if(hit.top + rn + newwidth > hit.bot) {
|
|
// Two pieces
|
|
tops[0] = hit.top + rn;
|
|
bots[0] = hit.bot;
|
|
tops[1] = hit.top;
|
|
bots[1] = hit.top + newwidth - (bots[0] - tops[0]);
|
|
} else {
|
|
// One piece
|
|
tops[0] = hit.top + rn;
|
|
bots[0] = tops[0] + newwidth;
|
|
}
|
|
assert_leq(bots[0], hit.bot);
|
|
assert_leq(bots[1], hit.bot);
|
|
assert_geq(bots[0], tops[0]);
|
|
assert_geq(bots[1], tops[1]);
|
|
assert_eq(newwidth, (bots[0] - tops[0]) + (bots[1] - tops[1]));
|
|
}
|
|
for(int i = 0; i < 2 && !done; i++) {
|
|
if(bots[i] <= tops[i]) break;
|
|
index_t width = bots[i] - tops[i];
|
|
index_t top = tops[i];
|
|
// Clear list where resolved offsets are stored
|
|
swmSeed.exranges++;
|
|
swmSeed.exrows += width;
|
|
if(!succ) {
|
|
swmSeed.exsucc++;
|
|
succ = true;
|
|
}
|
|
if(firstEe) {
|
|
salistEe_.clear();
|
|
pool_.clear();
|
|
firstEe = false;
|
|
}
|
|
// We have to be careful not to allocate excessive amounts of memory here
|
|
TSlice o(salistEe_, (index_t)salistEe_.size(), width);
|
|
for(index_t i = 0; i < width; i++) {
|
|
if(!salistEe_.add(pool_, (index_t)OFF_MASK)) {
|
|
swmSeed.exooms++;
|
|
return false;
|
|
}
|
|
}
|
|
assert(!done);
|
|
eehits_.push_back(hit);
|
|
satpos_.expand();
|
|
satpos_.back().sat.init(SAKey(), top, (index_t)OFF_MASK, o);
|
|
satpos_.back().sat.key.seq = MAX_U64;
|
|
satpos_.back().sat.key.len = (index_t)rd.length();
|
|
satpos_.back().pos.init(fw, 0, 0, (index_t)rd.length());
|
|
satpos_.back().origSz = width;
|
|
rands_.expand();
|
|
rands_.back().init(width, all);
|
|
gws_.expand();
|
|
SARangeWithOffs<TSlice, index_t> sa;
|
|
sa.topf = satpos_.back().sat.topf;
|
|
sa.len = satpos_.back().sat.key.len;
|
|
sa.offs = satpos_.back().sat.offs;
|
|
gws_.back().init(
|
|
gfm, // forward Bowtie index
|
|
ref, // reference sequences
|
|
sa, // SATuple
|
|
rnd, // pseudo-random generator
|
|
wlm); // metrics
|
|
assert(gws_.back().repOk(sa));
|
|
nelt_out += width;
|
|
if(nelt_out >= maxelt) {
|
|
done = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
succ = false;
|
|
if(!done && !sh.mm1EEHits().empty()) {
|
|
sh.sort1mmEe(rnd);
|
|
index_t sz = sh.mm1EEHits().size();
|
|
for(index_t i = 0; i < sz && !done; i++) {
|
|
EEHit<index_t> hit = sh.mm1EEHits()[i];
|
|
assert(hit.repOk(rd));
|
|
assert(!hit.empty());
|
|
// Possibly adjust bot and width if we would have exceeded maxelt
|
|
index_t tops[2] = { hit.top, 0 };
|
|
index_t bots[2] = { hit.bot, 0 };
|
|
index_t width = hit.bot - hit.top;
|
|
if(nelt_out + width > maxelt) {
|
|
index_t trim = (index_t)((nelt_out + width) - maxelt);
|
|
#ifdef BOWTIE_64BIT_INDEX
|
|
index_t rn = rnd.nextU64() % width;
|
|
#else
|
|
index_t rn = rnd.nextU32() % width;
|
|
#endif
|
|
index_t newwidth = width - trim;
|
|
if(hit.top + rn + newwidth > hit.bot) {
|
|
// Two pieces
|
|
tops[0] = hit.top + rn;
|
|
bots[0] = hit.bot;
|
|
tops[1] = hit.top;
|
|
bots[1] = hit.top + newwidth - (bots[0] - tops[0]);
|
|
} else {
|
|
// One piece
|
|
tops[0] = hit.top + rn;
|
|
bots[0] = tops[0] + newwidth;
|
|
}
|
|
assert_leq(bots[0], hit.bot);
|
|
assert_leq(bots[1], hit.bot);
|
|
assert_geq(bots[0], tops[0]);
|
|
assert_geq(bots[1], tops[1]);
|
|
assert_eq(newwidth, (bots[0] - tops[0]) + (bots[1] - tops[1]));
|
|
}
|
|
for(int i = 0; i < 2 && !done; i++) {
|
|
if(bots[i] <= tops[i]) break;
|
|
index_t width = bots[i] - tops[i];
|
|
index_t top = tops[i];
|
|
// Clear list where resolved offsets are stored
|
|
swmSeed.mm1ranges++;
|
|
swmSeed.mm1rows += width;
|
|
if(!succ) {
|
|
swmSeed.mm1succ++;
|
|
succ = true;
|
|
}
|
|
if(firstEe) {
|
|
salistEe_.clear();
|
|
pool_.clear();
|
|
firstEe = false;
|
|
}
|
|
TSlice o(salistEe_, (index_t)salistEe_.size(), width);
|
|
for(size_t i = 0; i < width; i++) {
|
|
if(!salistEe_.add(pool_, (index_t)OFF_MASK)) {
|
|
swmSeed.mm1ooms++;
|
|
return false;
|
|
}
|
|
}
|
|
eehits_.push_back(hit);
|
|
satpos_.expand();
|
|
satpos_.back().sat.init(SAKey(), top, (index_t)OFF_MASK, o);
|
|
satpos_.back().sat.key.seq = MAX_U64;
|
|
satpos_.back().sat.key.len = (index_t)rd.length();
|
|
satpos_.back().pos.init(hit.fw, 0, 0, (index_t)rd.length());
|
|
satpos_.back().origSz = width;
|
|
rands_.expand();
|
|
rands_.back().init(width, all);
|
|
gws_.expand();
|
|
SARangeWithOffs<TSlice, index_t> sa;
|
|
sa.topf = satpos_.back().sat.topf;
|
|
sa.len = satpos_.back().sat.key.len;
|
|
sa.offs = satpos_.back().sat.offs;
|
|
gws_.back().init(
|
|
gfm, // forward Bowtie index
|
|
ref, // reference sequences
|
|
sa, // SATuple
|
|
rnd, // pseudo-random generator
|
|
wlm); // metrics
|
|
assert(gws_.back().repOk(sa));
|
|
nelt_out += width;
|
|
if(nelt_out >= maxelt) {
|
|
done = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Extend a seed hit out on either side. Requires that we know the seed hit's
|
|
* offset into the read and orientation. Also requires that we know top/bot
|
|
* for the seed hit in both the forward and (if we want to extend to the right)
|
|
* reverse index.
|
|
*/
|
|
template <typename index_t>
|
|
void SwDriver<index_t>::extend(
|
|
const Read& rd, // read
|
|
const GFM<index_t>& gfmFw, // Forward Bowtie index
|
|
const GFM<index_t>* gfmBw, // Backward Bowtie index
|
|
index_t topf, // top in fw index
|
|
index_t botf, // bot in fw index
|
|
index_t topb, // top in bw index
|
|
index_t botb, // bot in bw index
|
|
bool fw, // seed orientation
|
|
index_t off, // seed offset from 5' end
|
|
index_t len, // seed length
|
|
PerReadMetrics& prm, // per-read metrics
|
|
index_t& nlex, // # positions we can extend to left w/o edit
|
|
index_t& nrex) // # positions we can extend to right w/o edit
|
|
{
|
|
index_t t[4], b[4];
|
|
index_t tp[4], bp[4];
|
|
SideLocus<index_t> tloc, bloc;
|
|
index_t rdlen = (index_t)rd.length();
|
|
index_t lim = fw ? off : rdlen - len - off;
|
|
// We're about to add onto the beginning, so reverse it
|
|
#ifndef NDEBUG
|
|
if(false) {
|
|
// TODO: This will sometimes fail even when the extension is legitimate
|
|
// This is because contains() comes in from one extreme or the other,
|
|
// whereas we started from the inside and worked outwards. This
|
|
// affects which Ns are OK and which are not OK.
|
|
|
|
// Have to do both because whether we can get through an N depends on
|
|
// which direction we're coming in
|
|
bool fwContains = gfmFw.contains(tmp_rdseq_);
|
|
tmp_rdseq_.reverse();
|
|
bool bwContains = gfmBw != NULL && gfmBw->contains(tmp_rdseq_);
|
|
tmp_rdseq_.reverse();
|
|
assert(fwContains || bwContains);
|
|
}
|
|
#endif
|
|
ASSERT_ONLY(tmp_rdseq_.reverse());
|
|
if(lim > 0) {
|
|
const GFM<index_t> *gfm = &gfmFw;
|
|
assert(gfm != NULL);
|
|
// Extend left using forward index
|
|
const BTDnaString& seq = fw ? rd.patFw : rd.patRc;
|
|
// See what we get by extending
|
|
index_t top = topf, bot = botf;
|
|
t[0] = t[1] = t[2] = t[3] = 0;
|
|
b[0] = b[1] = b[2] = b[3] = 0;
|
|
tp[0] = tp[1] = tp[2] = tp[3] = topb;
|
|
bp[0] = bp[1] = bp[2] = bp[3] = botb;
|
|
SideLocus<index_t> tloc, bloc;
|
|
INIT_LOCS(top, bot, tloc, bloc, *gfm);
|
|
for(index_t ii = 0; ii < lim; ii++) {
|
|
// Starting to left of seed (<off) and moving left
|
|
index_t i = 0;
|
|
if(fw) {
|
|
i = off - ii - 1;
|
|
} else {
|
|
i = rdlen - off - len - 1 - ii;
|
|
}
|
|
// Get char from read
|
|
int rdc = seq.get(i);
|
|
// See what we get by extending
|
|
if(bloc.valid()) {
|
|
prm.nSdFmops++;
|
|
t[0] = t[1] = t[2] = t[3] =
|
|
b[0] = b[1] = b[2] = b[3] = 0;
|
|
gfm->mapBiLFEx(tloc, bloc, t, b, tp, bp);
|
|
SANITY_CHECK_4TUP(t, b, tp, bp);
|
|
int nonz = -1;
|
|
bool abort = false;
|
|
size_t origSz = bot - top;
|
|
for(int j = 0; j < 4; j++) {
|
|
if(b[j] > t[j]) {
|
|
if(nonz >= 0) {
|
|
abort = true;
|
|
break;
|
|
}
|
|
nonz = j;
|
|
top = t[j]; bot = b[j];
|
|
}
|
|
}
|
|
assert_leq(bot - top, origSz);
|
|
if(abort || (nonz != rdc && rdc <= 3) || bot - top < origSz) {
|
|
break;
|
|
}
|
|
} else {
|
|
assert_eq(bot, top+1);
|
|
prm.nSdFmops++;
|
|
int c = gfm->mapLF1(top, tloc);
|
|
if(c != rdc && rdc <= 3) {
|
|
break;
|
|
}
|
|
bot = top + 1;
|
|
}
|
|
ASSERT_ONLY(tmp_rdseq_.append(rdc));
|
|
if(++nlex == 255) {
|
|
break;
|
|
}
|
|
INIT_LOCS(top, bot, tloc, bloc, *gfm);
|
|
}
|
|
}
|
|
// We're about to add onto the end, so re-reverse
|
|
ASSERT_ONLY(tmp_rdseq_.reverse());
|
|
lim = fw ? rdlen - len - off : off;
|
|
if(lim > 0 && gfmBw != NULL) {
|
|
const GFM<index_t> *gfm = gfmBw;
|
|
assert(gfm != NULL);
|
|
// Extend right using backward index
|
|
const BTDnaString& seq = fw ? rd.patFw : rd.patRc;
|
|
// See what we get by extending
|
|
index_t top = topb, bot = botb;
|
|
t[0] = t[1] = t[2] = t[3] = 0;
|
|
b[0] = b[1] = b[2] = b[3] = 0;
|
|
tp[0] = tp[1] = tp[2] = tp[3] = topf;
|
|
bp[0] = bp[1] = bp[2] = bp[3] = botf;
|
|
INIT_LOCS(top, bot, tloc, bloc, *gfm);
|
|
for(index_t ii = 0; ii < lim; ii++) {
|
|
// Starting to right of seed (<off) and moving right
|
|
index_t i;
|
|
if(fw) {
|
|
i = ii + len + off;
|
|
} else {
|
|
i = rdlen - off + ii;
|
|
}
|
|
// Get char from read
|
|
int rdc = seq.get(i);
|
|
// See what we get by extending
|
|
if(bloc.valid()) {
|
|
prm.nSdFmops++;
|
|
t[0] = t[1] = t[2] = t[3] =
|
|
b[0] = b[1] = b[2] = b[3] = 0;
|
|
gfm->mapBiLFEx(tloc, bloc, t, b, tp, bp);
|
|
SANITY_CHECK_4TUP(t, b, tp, bp);
|
|
int nonz = -1;
|
|
bool abort = false;
|
|
size_t origSz = bot - top;
|
|
for(int j = 0; j < 4; j++) {
|
|
if(b[j] > t[j]) {
|
|
if(nonz >= 0) {
|
|
abort = true;
|
|
break;
|
|
}
|
|
nonz = j;
|
|
top = t[j]; bot = b[j];
|
|
}
|
|
}
|
|
assert_leq(bot - top, origSz);
|
|
if(abort || (nonz != rdc && rdc <= 3) || bot - top < origSz) {
|
|
break;
|
|
}
|
|
} else {
|
|
assert_eq(bot, top+1);
|
|
prm.nSdFmops++;
|
|
int c = gfm->mapLF1(top, tloc);
|
|
if(c != rdc && rdc <= 3) {
|
|
break;
|
|
}
|
|
bot = top + 1;
|
|
}
|
|
ASSERT_ONLY(tmp_rdseq_.append(rdc));
|
|
if(++nrex == 255) {
|
|
break;
|
|
}
|
|
INIT_LOCS(top, bot, tloc, bloc, *gfm);
|
|
}
|
|
}
|
|
#ifndef NDEBUG
|
|
if(false) {
|
|
// TODO: This will sometimes fail even when the extension is legitimate
|
|
// This is because contains() comes in from one extreme or the other,
|
|
// whereas we started from the inside and worked outwards. This
|
|
// affects which Ns are OK and which are not OK.
|
|
|
|
// Have to do both because whether we can get through an N depends on
|
|
// which direction we're coming in
|
|
bool fwContains = gfmFw.contains(tmp_rdseq_);
|
|
tmp_rdseq_.reverse();
|
|
bool bwContains = gfmBw != NULL && gfmBw->contains(tmp_rdseq_);
|
|
tmp_rdseq_.reverse();
|
|
assert(fwContains || bwContains);
|
|
}
|
|
#endif
|
|
assert_lt(nlex, rdlen);
|
|
assert_lt(nrex, rdlen);
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* Given seed results, set up all of our state for resolving and keeping
|
|
* track of reference offsets for hits.
|
|
*/
|
|
template <typename index_t>
|
|
void SwDriver<index_t>::prioritizeSATups(
|
|
const Read& read, // read
|
|
SeedResults<index_t>& sh, // seed hits to extend into full alignments
|
|
const GFM<index_t>& gfmFw, // BWT
|
|
const GFM<index_t>* gfmBw, // BWT
|
|
const BitPairReference& ref, // Reference strings
|
|
int seedmms, // # mismatches allowed in seed
|
|
index_t maxelt, // max elts we'll consider
|
|
bool doExtend, // do extension of seed hits?
|
|
bool lensq, // square length in weight calculation
|
|
bool szsq, // square range size in weight calculation
|
|
index_t nsm, // if range as <= nsm elts, it's "small"
|
|
AlignmentCacheIface<index_t>& ca, // alignment cache for seed hits
|
|
RandomSource& rnd, // pseudo-random generator
|
|
WalkMetrics& wlm, // group walk left metrics
|
|
PerReadMetrics& prm, // per-read metrics
|
|
index_t& nelt_out, // out: # elements total
|
|
bool all) // report all hits?
|
|
{
|
|
const index_t nonz = sh.nonzeroOffsets(); // non-zero positions
|
|
const int matei = (read.mate <= 1 ? 0 : 1);
|
|
satups_.clear();
|
|
gws_.clear();
|
|
rands_.clear();
|
|
rands2_.clear();
|
|
satpos_.clear();
|
|
satpos2_.clear();
|
|
index_t nrange = 0, nelt = 0, nsmall = 0, nsmall_elts = 0;
|
|
bool keepWhole = false;
|
|
EList<SATupleAndPos<index_t>, 16>& satpos = keepWhole ? satpos_ : satpos2_;
|
|
for(index_t i = 0; i < nonz; i++) {
|
|
bool fw = true;
|
|
index_t offidx = 0, rdoff = 0, seedlen = 0;
|
|
QVal<index_t> qv = sh.hitsByRank(i, offidx, rdoff, fw, seedlen);
|
|
assert(qv.valid());
|
|
assert(!qv.empty());
|
|
assert(qv.repOk(ca.current()));
|
|
ca.queryQval(qv, satups_, nrange, nelt);
|
|
for(size_t j = 0; j < satups_.size(); j++) {
|
|
const index_t sz = satups_[j].size();
|
|
// Check whether this hit occurs inside the extended boundaries of
|
|
// another hit we already processed for this read.
|
|
if(seedmms == 0) {
|
|
// See if we're covered by a previous extended seed hit
|
|
EList<ExtendRange>& range =
|
|
fw ? seedExRangeFw_[matei] : seedExRangeRc_[matei];
|
|
bool skip = false;
|
|
for(index_t k = 0; k < range.size(); k++) {
|
|
index_t p5 = range[k].off;
|
|
index_t len = range[k].len;
|
|
if(p5 <= rdoff && p5 + len >= (rdoff + seedlen)) {
|
|
if(sz <= range[k].sz) {
|
|
skip = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if(skip) {
|
|
assert_gt(nrange, 0);
|
|
nrange--;
|
|
assert_geq(nelt, sz);
|
|
nelt -= sz;
|
|
continue; // Skip this seed
|
|
}
|
|
}
|
|
satpos.expand();
|
|
satpos.back().sat = satups_[j];
|
|
satpos.back().origSz = sz;
|
|
satpos.back().pos.init(fw, offidx, rdoff, seedlen);
|
|
if(sz <= nsm) {
|
|
nsmall++;
|
|
nsmall_elts += sz;
|
|
}
|
|
satpos.back().nlex = satpos.back().nrex = 0;
|
|
#ifndef NDEBUG
|
|
tmp_rdseq_.clear();
|
|
uint64_t key = satpos.back().sat.key.seq;
|
|
for(size_t k = 0; k < seedlen; k++) {
|
|
int c = (int)(key & 3);
|
|
tmp_rdseq_.append(c);
|
|
key >>= 2;
|
|
}
|
|
tmp_rdseq_.reverse();
|
|
#endif
|
|
index_t nlex = 0, nrex = 0;
|
|
if(doExtend) {
|
|
extend(
|
|
read,
|
|
gfmFw,
|
|
gfmBw,
|
|
satpos.back().sat.topf,
|
|
(index_t)(satpos.back().sat.topf + sz),
|
|
satpos.back().sat.topb,
|
|
(index_t)(satpos.back().sat.topb + sz),
|
|
fw,
|
|
rdoff,
|
|
seedlen,
|
|
prm,
|
|
nlex,
|
|
nrex);
|
|
}
|
|
satpos.back().nlex = nlex;
|
|
satpos.back().nrex = nrex;
|
|
if(seedmms == 0 && (nlex > 0 || nrex > 0)) {
|
|
assert_geq(rdoff, (fw ? nlex : nrex));
|
|
index_t p5 = rdoff - (fw ? nlex : nrex);
|
|
EList<ExtendRange>& range =
|
|
fw ? seedExRangeFw_[matei] : seedExRangeRc_[matei];
|
|
range.expand();
|
|
range.back().off = p5;
|
|
range.back().len = seedlen + nlex + nrex;
|
|
range.back().sz = sz;
|
|
}
|
|
}
|
|
satups_.clear();
|
|
}
|
|
assert_leq(nsmall, nrange);
|
|
nelt_out = nelt; // return the total number of elements
|
|
assert_eq(nrange, satpos.size());
|
|
satpos.sort();
|
|
if(keepWhole) {
|
|
gws_.ensure(nrange);
|
|
rands_.ensure(nrange);
|
|
for(index_t i = 0; i < nrange; i++) {
|
|
gws_.expand();
|
|
SARangeWithOffs<TSlice, index_t> sa;
|
|
sa.topf = satpos_.back().sat.topf;
|
|
sa.len = satpos_.back().sat.key.len;
|
|
sa.offs = satpos_.back().sat.offs;
|
|
gws_.back().init(
|
|
gfmFw, // forward Bowtie index
|
|
ref, // reference sequences
|
|
sa, // SA tuples: ref hit, salist range
|
|
rnd, // pseudo-random generator
|
|
wlm); // metrics
|
|
assert(gws_.back().initialized());
|
|
rands_.expand();
|
|
rands_.back().init(satpos_[i].sat.size(), all);
|
|
}
|
|
return;
|
|
}
|
|
// Resize satups_ list so that ranges having elements that we might
|
|
// possibly explore are present
|
|
satpos_.ensure(min(maxelt, nelt));
|
|
gws_.ensure(min(maxelt, nelt));
|
|
rands_.ensure(min(maxelt, nelt));
|
|
rands2_.ensure(min(maxelt, nelt));
|
|
size_t nlarge_elts = nelt - nsmall_elts;
|
|
if(maxelt < nelt) {
|
|
size_t diff = nelt - maxelt;
|
|
if(diff >= nlarge_elts) {
|
|
nlarge_elts = 0;
|
|
} else {
|
|
nlarge_elts -= diff;
|
|
}
|
|
}
|
|
index_t nelt_added = 0;
|
|
// Now we have a collection of ranges in satpos2_. Now we want to decide
|
|
// how we explore elements from them. The basic idea is that: for very
|
|
// small guys, where "very small" means that the size of the range is less
|
|
// than or equal to the parameter 'nsz', we explore them in their entirety
|
|
// right away. For the rest, we want to select in a way that is (a)
|
|
// random, and (b) weighted toward examining elements from the smaller
|
|
// ranges more frequently (and first).
|
|
//
|
|
// 1. do the smalls
|
|
for(index_t j = 0; j < nsmall && nelt_added < maxelt; j++) {
|
|
satpos_.expand();
|
|
satpos_.back() = satpos2_[j];
|
|
gws_.expand();
|
|
SARangeWithOffs<TSlice, index_t> sa;
|
|
sa.topf = satpos_.back().sat.topf;
|
|
sa.len = satpos_.back().sat.key.len;
|
|
sa.offs = satpos_.back().sat.offs;
|
|
gws_.back().init(
|
|
gfmFw, // forward Bowtie index
|
|
ref, // reference sequences
|
|
sa, // SA tuples: ref hit, salist range
|
|
rnd, // pseudo-random generator
|
|
wlm); // metrics
|
|
assert(gws_.back().initialized());
|
|
rands_.expand();
|
|
rands_.back().init(satpos_.back().sat.size(), all);
|
|
nelt_added += satpos_.back().sat.size();
|
|
#ifndef NDEBUG
|
|
for(size_t k = 0; k < satpos_.size()-1; k++) {
|
|
assert(!(satpos_[k] == satpos_.back()));
|
|
}
|
|
#endif
|
|
}
|
|
if(nelt_added >= maxelt || nsmall == satpos2_.size()) {
|
|
nelt_out = nelt_added;
|
|
return;
|
|
}
|
|
// 2. do the non-smalls
|
|
// Initialize the row sampler
|
|
rowsamp_.init(satpos2_, nsmall, satpos2_.size(), lensq, szsq);
|
|
// Initialize the random choosers
|
|
rands2_.resize(satpos2_.size());
|
|
for(index_t j = 0; j < satpos2_.size(); j++) {
|
|
rands2_[j].reset();
|
|
}
|
|
while(nelt_added < maxelt && nelt_added < nelt) {
|
|
// Pick a non-small range to sample from
|
|
index_t ri = rowsamp_.next(rnd) + nsmall;
|
|
assert_geq(ri, nsmall);
|
|
assert_lt(ri, satpos2_.size());
|
|
// Initialize random element chooser for that range
|
|
if(!rands2_[ri].inited()) {
|
|
rands2_[ri].init(satpos2_[ri].sat.size(), all);
|
|
assert(!rands2_[ri].done());
|
|
}
|
|
assert(!rands2_[ri].done());
|
|
// Choose an element from the range
|
|
uint32_t r = rands2_[ri].next(rnd);
|
|
if(rands2_[ri].done()) {
|
|
// Tell the row sampler this range is done
|
|
rowsamp_.finishedRange(ri - nsmall);
|
|
}
|
|
// Add the element to the satpos_ list
|
|
SATuple<index_t> sat;
|
|
TSlice o;
|
|
o.init(satpos2_[ri].sat.offs, r, r+1);
|
|
sat.init(satpos2_[ri].sat.key, (index_t)(satpos2_[ri].sat.topf + r), (index_t)OFF_MASK, o);
|
|
satpos_.expand();
|
|
satpos_.back().sat = sat;
|
|
satpos_.back().origSz = satpos2_[ri].origSz;
|
|
satpos_.back().pos = satpos2_[ri].pos;
|
|
// Initialize GroupWalk object
|
|
gws_.expand();
|
|
SARangeWithOffs<TSlice, index_t> sa;
|
|
sa.topf = sat.topf;
|
|
sa.len = sat.key.len;
|
|
sa.offs = sat.offs;
|
|
gws_.back().init(
|
|
gfmFw, // forward Bowtie index
|
|
ref, // reference sequences
|
|
sa, // SA tuples: ref hit, salist range
|
|
rnd, // pseudo-random generator
|
|
wlm); // metrics
|
|
assert(gws_.back().initialized());
|
|
// Initialize random selector
|
|
rands_.expand();
|
|
rands_.back().init(1, all);
|
|
nelt_added++;
|
|
}
|
|
nelt_out = nelt_added;
|
|
return;
|
|
}
|
|
|
|
enum {
|
|
FOUND_NONE = 0,
|
|
FOUND_EE,
|
|
FOUND_UNGAPPED,
|
|
};
|
|
|
|
/**
|
|
* Given a collection of SeedHits for a single read, extend seed alignments
|
|
* into full alignments. Where possible, try to avoid redundant offset lookups
|
|
* and dynamic programming wherever possible. Optionally report alignments to
|
|
* a AlnSinkWrap object as they are discovered.
|
|
*
|
|
* If 'reportImmediately' is true, returns true iff a call to msink->report()
|
|
* returned true (indicating that the reporting policy is satisfied and we can
|
|
* stop). Otherwise, returns false.
|
|
*/
|
|
template <typename index_t>
|
|
int SwDriver<index_t>::extendSeeds(
|
|
Read& rd, // read to align
|
|
bool mate1, // true iff rd is mate #1
|
|
SeedResults<index_t>& sh, // seed hits to extend into full alignments
|
|
const GFM<index_t>& gfmFw, // BWT
|
|
const GFM<index_t>* gfmBw, // BWT'
|
|
const BitPairReference& ref, // Reference strings
|
|
SwAligner& swa, // dynamic programming aligner
|
|
const Scoring& sc, // scoring scheme
|
|
int seedmms, // # mismatches allowed in seed
|
|
int seedlen, // length of seed
|
|
int seedival, // interval between seeds
|
|
TAlScore& minsc, // minimum score for anchor
|
|
int nceil, // maximum # Ns permitted in reference portion
|
|
size_t maxhalf, // max width in either direction for DP tables
|
|
bool doUngapped, // do ungapped alignment
|
|
size_t maxIters, // stop after this many seed-extend loop iters
|
|
size_t maxUg, // stop after this many ungaps
|
|
size_t maxDp, // stop after this many dps
|
|
size_t maxUgStreak, // stop after streak of this many ungap fails
|
|
size_t maxDpStreak, // stop after streak of this many dp fails
|
|
bool doExtend, // do seed extension
|
|
bool enable8, // use 8-bit SSE where possible
|
|
size_t cminlen, // use checkpointer if read longer than this
|
|
size_t cpow2, // interval between diagonals to checkpoint
|
|
bool doTri, // triangular mini-fills?
|
|
int tighten, // -M score tightening mode
|
|
AlignmentCacheIface<index_t>& ca, // alignment cache for seed hits
|
|
RandomSource& rnd, // pseudo-random source
|
|
WalkMetrics& wlm, // group walk left metrics
|
|
SwMetrics& swmSeed, // DP metrics for seed-extend
|
|
PerReadMetrics& prm, // per-read metrics
|
|
AlnSinkWrap<index_t>* msink, // AlnSink wrapper for multiseed-style aligner
|
|
bool reportImmediately, // whether to report hits immediately to msink
|
|
bool& exhaustive) // set to true iff we searched all seeds exhaustively
|
|
{
|
|
bool all = msink->allHits();
|
|
// typedef std::pair<index_t, index_t> UPair;
|
|
|
|
assert(!reportImmediately || msink != NULL);
|
|
assert(!reportImmediately || !msink->maxed());
|
|
|
|
assert_geq(nceil, 0);
|
|
assert_leq((size_t)nceil, rd.length());
|
|
|
|
// Calculate the largest possible number of read and reference gaps
|
|
const index_t rdlen = (index_t)rd.length();
|
|
TAlScore perfectScore = sc.perfectScore(rdlen);
|
|
|
|
DynProgFramer dpframe(!gReportOverhangs);
|
|
swa.reset();
|
|
|
|
// Initialize a set of GroupWalks, one for each seed. Also, intialize the
|
|
// accompanying lists of reference seed hits (satups*)
|
|
const index_t nsm = 5;
|
|
const index_t nonz = sh.nonzeroOffsets(); // non-zero positions
|
|
index_t eeHits = sh.numE2eHits();
|
|
bool eeMode = eeHits > 0;
|
|
bool firstEe = true;
|
|
bool firstExtend = true;
|
|
|
|
// Reset all the counters related to streaks
|
|
prm.nEeFail = 0;
|
|
prm.nUgFail = 0;
|
|
prm.nDpFail = 0;
|
|
|
|
index_t nelt = 0, neltLeft = 0;
|
|
index_t rows = rdlen;
|
|
index_t eltsDone = 0;
|
|
// cerr << "===" << endl;
|
|
while(true) {
|
|
if(eeMode) {
|
|
if(firstEe) {
|
|
firstEe = false;
|
|
eeMode = eeSaTups(
|
|
rd, // read
|
|
sh, // seed hits to extend into full alignments
|
|
gfmFw, // BWT
|
|
ref, // Reference strings
|
|
rnd, // pseudo-random generator
|
|
wlm, // group walk left metrics
|
|
swmSeed, // seed-extend metrics
|
|
nelt, // out: # elements total
|
|
maxIters, // max # to report
|
|
all); // report all hits?
|
|
assert_eq(gws_.size(), rands_.size());
|
|
assert_eq(gws_.size(), satpos_.size());
|
|
} else {
|
|
eeMode = false;
|
|
}
|
|
}
|
|
if(!eeMode) {
|
|
if(nonz == 0) {
|
|
return EXTEND_EXHAUSTED_CANDIDATES; // No seed hits! Bail.
|
|
}
|
|
if(minsc == perfectScore) {
|
|
return EXTEND_PERFECT_SCORE; // Already found all perfect hits!
|
|
}
|
|
if(firstExtend) {
|
|
nelt = 0;
|
|
prioritizeSATups(
|
|
rd, // read
|
|
sh, // seed hits to extend into full alignments
|
|
gfmFw, // BWT
|
|
gfmBw, // BWT'
|
|
ref, // Reference strings
|
|
seedmms, // # seed mismatches allowed
|
|
maxIters, // max rows to consider per position
|
|
doExtend, // extend out seeds
|
|
true, // square extended length
|
|
true, // square SA range size
|
|
nsm, // smallness threshold
|
|
ca, // alignment cache for seed hits
|
|
rnd, // pseudo-random generator
|
|
wlm, // group walk left metrics
|
|
prm, // per-read metrics
|
|
nelt, // out: # elements total
|
|
all); // report all hits?
|
|
assert_eq(gws_.size(), rands_.size());
|
|
assert_eq(gws_.size(), satpos_.size());
|
|
neltLeft = nelt;
|
|
firstExtend = false;
|
|
}
|
|
if(neltLeft == 0) {
|
|
// Finished examining gapped candidates
|
|
break;
|
|
}
|
|
}
|
|
for(size_t i = 0; i < gws_.size(); i++) {
|
|
if(eeMode && eehits_[i].score < minsc) {
|
|
return EXTEND_PERFECT_SCORE;
|
|
}
|
|
bool is_small = satpos_[i].sat.size() < nsm;
|
|
bool fw = satpos_[i].pos.fw;
|
|
index_t rdoff = satpos_[i].pos.rdoff;
|
|
index_t seedhitlen = satpos_[i].pos.seedlen;
|
|
if(!fw) {
|
|
// 'rdoff' and 'offidx' are with respect to the 5' end of
|
|
// the read. Here we convert rdoff to be with respect to
|
|
// the upstream (3') end of ther read.
|
|
rdoff = (index_t)(rdlen - rdoff - seedhitlen);
|
|
}
|
|
bool first = true;
|
|
// If the range is small, investigate all elements now. If the
|
|
// range is large, just investigate one and move on - we might come
|
|
// back to this range later.
|
|
index_t riter = 0;
|
|
while(!rands_[i].done() && (first || is_small || eeMode)) {
|
|
assert(!gws_[i].done());
|
|
riter++;
|
|
if(minsc == perfectScore) {
|
|
if(!eeMode || eehits_[i].score < perfectScore) {
|
|
return EXTEND_PERFECT_SCORE;
|
|
}
|
|
} else if(eeMode && eehits_[i].score < minsc) {
|
|
break;
|
|
}
|
|
if(prm.nExDps >= maxDp || prm.nMateDps >= maxDp) {
|
|
return EXTEND_EXCEEDED_HARD_LIMIT;
|
|
}
|
|
if(prm.nExUgs >= maxUg || prm.nMateUgs >= maxUg) {
|
|
return EXTEND_EXCEEDED_HARD_LIMIT;
|
|
}
|
|
if(prm.nExIters >= maxIters) {
|
|
return EXTEND_EXCEEDED_HARD_LIMIT;
|
|
}
|
|
prm.nExIters++;
|
|
first = false;
|
|
// Resolve next element offset
|
|
WalkResult<index_t> wr;
|
|
uint32_t elt = rands_[i].next(rnd);
|
|
//cerr << "elt=" << elt << endl;
|
|
SARangeWithOffs<TSlice, index_t> sa;
|
|
sa.topf = satpos_[i].sat.topf;
|
|
sa.len = satpos_[i].sat.key.len;
|
|
sa.offs = satpos_[i].sat.offs;
|
|
gws_[i].advanceElement((index_t)elt, gfmFw, ref, sa, gwstate_, wr, wlm, prm);
|
|
eltsDone++;
|
|
if(!eeMode) {
|
|
assert_gt(neltLeft, 0);
|
|
neltLeft--;
|
|
}
|
|
assert_neq((index_t)OFF_MASK, wr.toff);
|
|
index_t tidx = 0, toff = 0, tlen = 0;
|
|
bool straddled = false;
|
|
gfmFw.joinedToTextOff(
|
|
wr.elt.len,
|
|
wr.toff,
|
|
tidx,
|
|
toff,
|
|
tlen,
|
|
eeMode, // reject straddlers?
|
|
straddled); // did it straddle?
|
|
if(tidx == (index_t)OFF_MASK) {
|
|
// The seed hit straddled a reference boundary so the seed hit
|
|
// isn't valid
|
|
continue;
|
|
}
|
|
#ifndef NDEBUG
|
|
if(!eeMode && !straddled) { // Check that seed hit matches reference
|
|
uint64_t key = satpos_[i].sat.key.seq;
|
|
for(index_t k = 0; k < wr.elt.len; k++) {
|
|
int c = ref.getBase(tidx, toff + wr.elt.len - k - 1);
|
|
assert_leq(c, 3);
|
|
int ck = (int)(key & 3);
|
|
key >>= 2;
|
|
assert_eq(c, ck);
|
|
}
|
|
}
|
|
#endif
|
|
// Find offset of alignment's upstream base assuming net gaps=0
|
|
// between beginning of read and beginning of seed hit
|
|
int64_t refoff = (int64_t)toff - rdoff;
|
|
// Coordinate of the seed hit w/r/t the pasted reference string
|
|
Coord refcoord(tidx, refoff, fw);
|
|
if(seenDiags1_.locusPresent(refcoord)) {
|
|
// Already handled alignments seeded on this diagonal
|
|
prm.nRedundants++;
|
|
swmSeed.rshit++;
|
|
continue;
|
|
}
|
|
// Now that we have a seed hit, there are many issues to solve
|
|
// before we have a completely framed dynamic programming problem.
|
|
// They include:
|
|
//
|
|
// 1. Setting reference offsets on either side of the seed hit,
|
|
// accounting for where the seed occurs in the read
|
|
// 2. Adjusting the width of the banded dynamic programming problem
|
|
// and adjusting reference bounds to allow for gaps in the
|
|
// alignment
|
|
// 3. Accounting for the edges of the reference, which can impact
|
|
// the width of the DP problem and reference bounds.
|
|
// 4. Perhaps filtering the problem down to a smaller problem based
|
|
// on what DPs we've already solved for this read
|
|
//
|
|
// We do #1 here, since it is simple and we have all the seed-hit
|
|
// information here. #2 and #3 are handled in the DynProgFramer.
|
|
int readGaps = 0, refGaps = 0;
|
|
bool ungapped = false;
|
|
if(!eeMode) {
|
|
readGaps = sc.maxReadGaps(minsc, rdlen);
|
|
refGaps = sc.maxRefGaps(minsc, rdlen);
|
|
ungapped = (readGaps == 0 && refGaps == 0);
|
|
}
|
|
int state = FOUND_NONE;
|
|
bool found = false;
|
|
if(eeMode) {
|
|
resEe_.reset();
|
|
resEe_.alres.reset();
|
|
const EEHit<index_t>& h = eehits_[i];
|
|
assert_leq(h.score, perfectScore);
|
|
resEe_.alres.setScore(AlnScore(h.score, h.ns(), 0));
|
|
resEe_.alres.setShape(
|
|
refcoord.ref(), // ref id
|
|
refcoord.off(), // 0-based ref offset
|
|
tlen, // length of reference
|
|
fw, // aligned to Watson?
|
|
rdlen, // read length
|
|
true, // pretrim soft?
|
|
0, // pretrim 5' end
|
|
0, // pretrim 3' end
|
|
true, // alignment trim soft?
|
|
0, // alignment trim 5' end
|
|
0); // alignment trim 3' end
|
|
resEe_.alres.setRefNs(h.refns());
|
|
if(h.mms() > 0) {
|
|
assert_eq(1, h.mms());
|
|
assert_lt(h.e1.pos, rd.length());
|
|
resEe_.alres.ned().push_back(h.e1);
|
|
}
|
|
assert(resEe_.repOk(rd));
|
|
state = FOUND_EE;
|
|
found = true;
|
|
Interval refival(refcoord, 1);
|
|
seenDiags1_.add(refival);
|
|
} else if(doUngapped && ungapped) {
|
|
resUngap_.reset();
|
|
int al = swa.ungappedAlign(
|
|
fw ? rd.patFw : rd.patRc,
|
|
fw ? rd.qual : rd.qualRev,
|
|
refcoord,
|
|
ref,
|
|
tlen,
|
|
sc,
|
|
gReportOverhangs,
|
|
minsc,
|
|
resUngap_);
|
|
Interval refival(refcoord, 1);
|
|
seenDiags1_.add(refival);
|
|
prm.nExUgs++;
|
|
if(al == 0) {
|
|
prm.nExUgFails++;
|
|
prm.nUgFail++;
|
|
if(prm.nUgFail >= maxUgStreak) {
|
|
return EXTEND_EXCEEDED_SOFT_LIMIT;
|
|
}
|
|
swmSeed.ungapfail++;
|
|
continue;
|
|
} else if(al == -1) {
|
|
prm.nExUgFails++;
|
|
prm.nUgFail++; // count this as failure
|
|
if(prm.nUgFail >= maxUgStreak) {
|
|
return EXTEND_EXCEEDED_SOFT_LIMIT;
|
|
}
|
|
swmSeed.ungapnodec++;
|
|
} else {
|
|
prm.nExUgSuccs++;
|
|
prm.nUgLastSucc = prm.nExUgs-1;
|
|
if(prm.nUgFail > prm.nUgFailStreak) {
|
|
prm.nUgFailStreak = prm.nUgFail;
|
|
}
|
|
prm.nUgFail = 0;
|
|
found = true;
|
|
state = FOUND_UNGAPPED;
|
|
swmSeed.ungapsucc++;
|
|
}
|
|
}
|
|
int64_t pastedRefoff = (int64_t)wr.toff - rdoff;
|
|
DPRect rect;
|
|
if(state == FOUND_NONE) {
|
|
found = dpframe.frameSeedExtensionRect(
|
|
refoff, // ref offset implied by seed hit assuming no gaps
|
|
rows, // length of read sequence used in DP table
|
|
tlen, // length of reference
|
|
readGaps, // max # of read gaps permitted in opp mate alignment
|
|
refGaps, // max # of ref gaps permitted in opp mate alignment
|
|
(size_t)nceil, // # Ns permitted
|
|
maxhalf, // max width in either direction
|
|
rect); // DP rectangle
|
|
assert(rect.repOk());
|
|
// Add the seed diagonal at least
|
|
seenDiags1_.add(Interval(refcoord, 1));
|
|
if(!found) {
|
|
continue;
|
|
}
|
|
}
|
|
int64_t leftShift = refoff - rect.refl;
|
|
size_t nwindow = 0;
|
|
if(toff >= rect.refl) {
|
|
nwindow = (size_t)(toff - rect.refl);
|
|
}
|
|
// NOTE: We might be taking off more than we should because the
|
|
// pasted string omits non-A/C/G/T characters, but we included them
|
|
// when calculating leftShift. We'll account for this later.
|
|
pastedRefoff -= leftShift;
|
|
size_t nsInLeftShift = 0;
|
|
if(state == FOUND_NONE) {
|
|
if(!swa.initedRead()) {
|
|
// Initialize the aligner with a new read
|
|
swa.initRead(
|
|
rd.patFw, // fw version of query
|
|
rd.patRc, // rc version of query
|
|
rd.qual, // fw version of qualities
|
|
rd.qualRev,// rc version of qualities
|
|
0, // off of first char in 'rd' to consider
|
|
rdlen, // off of last char (excl) in 'rd' to consider
|
|
sc); // scoring scheme
|
|
}
|
|
swa.initRef(
|
|
fw, // whether to align forward or revcomp read
|
|
tidx, // reference aligned against
|
|
rect, // DP rectangle
|
|
ref, // Reference strings
|
|
tlen, // length of reference sequence
|
|
sc, // scoring scheme
|
|
minsc, // minimum score permitted
|
|
enable8, // use 8-bit SSE if possible?
|
|
cminlen, // minimum length for using checkpointing scheme
|
|
cpow2, // interval b/t checkpointed diags; 1 << this
|
|
doTri, // triangular mini-fills?
|
|
true, // this is a seed extension - not finding a mate
|
|
nwindow,
|
|
nsInLeftShift);
|
|
// Because of how we framed the problem, we can say that we've
|
|
// exhaustively scored the seed diagonal as well as maxgaps
|
|
// diagonals on either side
|
|
Interval refival(tidx, 0, fw, 0);
|
|
rect.initIval(refival);
|
|
seenDiags1_.add(refival);
|
|
// Now fill the dynamic programming matrix and return true iff
|
|
// there is at least one valid alignment
|
|
TAlScore bestCell = std::numeric_limits<TAlScore>::min();
|
|
found = swa.align(rnd, bestCell);
|
|
swmSeed.tallyGappedDp(readGaps, refGaps);
|
|
prm.nExDps++;
|
|
if(!found) {
|
|
prm.nExDpFails++;
|
|
prm.nDpFail++;
|
|
if(prm.nDpFail >= maxDpStreak) {
|
|
return EXTEND_EXCEEDED_SOFT_LIMIT;
|
|
}
|
|
if(bestCell > std::numeric_limits<TAlScore>::min() && bestCell > prm.bestLtMinscMate1) {
|
|
prm.bestLtMinscMate1 = bestCell;
|
|
}
|
|
continue; // Look for more anchor alignments
|
|
} else {
|
|
prm.nExDpSuccs++;
|
|
prm.nDpLastSucc = prm.nExDps-1;
|
|
if(prm.nDpFail > prm.nDpFailStreak) {
|
|
prm.nDpFailStreak = prm.nDpFail;
|
|
}
|
|
prm.nDpFail = 0;
|
|
}
|
|
}
|
|
bool firstInner = true;
|
|
while(true) {
|
|
assert(found);
|
|
SwResult *res = NULL;
|
|
if(state == FOUND_EE) {
|
|
if(!firstInner) {
|
|
break;
|
|
}
|
|
res = &resEe_;
|
|
} else if(state == FOUND_UNGAPPED) {
|
|
if(!firstInner) {
|
|
break;
|
|
}
|
|
res = &resUngap_;
|
|
} else {
|
|
resGap_.reset();
|
|
assert(resGap_.empty());
|
|
if(swa.done()) {
|
|
break;
|
|
}
|
|
swa.nextAlignment(resGap_, minsc, rnd);
|
|
found = !resGap_.empty();
|
|
if(!found) {
|
|
break;
|
|
}
|
|
res = &resGap_;
|
|
}
|
|
assert(res != NULL);
|
|
firstInner = false;
|
|
assert(res->alres.matchesRef(
|
|
rd,
|
|
ref,
|
|
tmp_rf_,
|
|
tmp_rdseq_,
|
|
tmp_qseq_,
|
|
raw_refbuf_,
|
|
raw_destU32_,
|
|
raw_matches_,
|
|
tmp_reflens_,
|
|
tmp_refoffs_));
|
|
Interval refival(tidx, 0, fw, tlen);
|
|
assert_gt(res->alres.refExtent(), 0);
|
|
if(gReportOverhangs &&
|
|
!refival.containsIgnoreOrient(res->alres.refival()))
|
|
{
|
|
res->alres.clipOutside(true, 0, tlen);
|
|
if(res->alres.refExtent() == 0) {
|
|
continue;
|
|
}
|
|
}
|
|
assert(gReportOverhangs ||
|
|
refival.containsIgnoreOrient(res->alres.refival()));
|
|
// Did the alignment fall entirely outside the reference?
|
|
if(!refival.overlapsIgnoreOrient(res->alres.refival())) {
|
|
continue;
|
|
}
|
|
// Is this alignment redundant with one we've seen previously?
|
|
if(redAnchor_.overlap(res->alres)) {
|
|
// Redundant with an alignment we found already
|
|
continue;
|
|
}
|
|
redAnchor_.add(res->alres);
|
|
// Annotate the AlnRes object with some key parameters
|
|
// that were used to obtain the alignment.
|
|
res->alres.setParams(
|
|
seedmms, // # mismatches allowed in seed
|
|
seedlen, // length of seed
|
|
seedival, // interval between seeds
|
|
minsc); // minimum score for valid alignment
|
|
|
|
if(reportImmediately) {
|
|
assert(msink != NULL);
|
|
assert(res->repOk());
|
|
// Check that alignment accurately reflects the
|
|
// reference characters aligned to
|
|
assert(res->alres.matchesRef(
|
|
rd,
|
|
ref,
|
|
tmp_rf_,
|
|
tmp_rdseq_,
|
|
tmp_qseq_,
|
|
raw_refbuf_,
|
|
raw_destU32_,
|
|
raw_matches_,
|
|
tmp_reflens_,
|
|
tmp_refoffs_));
|
|
// Report an unpaired alignment
|
|
assert(!msink->maxed());
|
|
if(msink->report(
|
|
0,
|
|
mate1 ? &res->alres : NULL,
|
|
mate1 ? NULL : &res->alres))
|
|
{
|
|
// Short-circuited because a limit, e.g. -k, -m or
|
|
// -M, was exceeded
|
|
return EXTEND_POLICY_FULFILLED;
|
|
}
|
|
if(tighten > 0 &&
|
|
msink->Mmode() &&
|
|
msink->hasSecondBestUnp1())
|
|
{
|
|
if(tighten == 1) {
|
|
if(msink->bestUnp1() >= minsc) {
|
|
minsc = msink->bestUnp1();
|
|
if(minsc < perfectScore &&
|
|
msink->bestUnp1() == msink->secondBestUnp1())
|
|
{
|
|
minsc++;
|
|
}
|
|
}
|
|
} else if(tighten == 2) {
|
|
if(msink->secondBestUnp1() >= minsc) {
|
|
minsc = msink->secondBestUnp1();
|
|
if(minsc < perfectScore) {
|
|
minsc++;
|
|
}
|
|
}
|
|
} else {
|
|
TAlScore diff = msink->bestUnp1() - msink->secondBestUnp1();
|
|
TAlScore bot = msink->secondBestUnp1() + ((diff*3)/4);
|
|
if(bot >= minsc) {
|
|
minsc = bot;
|
|
if(minsc < perfectScore) {
|
|
minsc++;
|
|
}
|
|
}
|
|
}
|
|
assert_leq(minsc, perfectScore);
|
|
}
|
|
}
|
|
}
|
|
|
|
// At this point we know that we aren't bailing, and will
|
|
// continue to resolve seed hits.
|
|
|
|
} // while(!gws_[i].done())
|
|
}
|
|
}
|
|
// Short-circuited because a limit, e.g. -k, -m or -M, was exceeded
|
|
return EXTEND_EXHAUSTED_CANDIDATES;
|
|
}
|
|
|
|
/**
|
|
* Given a collection of SeedHits for both mates in a read pair, extend seed
|
|
* alignments into full alignments and then look for the opposite mate using
|
|
* dynamic programming. Where possible, try to avoid redundant offset lookups.
|
|
* Optionally report alignments to a AlnSinkWrap object as they are discovered.
|
|
*
|
|
* If 'reportImmediately' is true, returns true iff a call to
|
|
* msink->report() returned true (indicating that the reporting
|
|
* policy is satisfied and we can stop). Otherwise, returns false.
|
|
*
|
|
* REDUNDANT SEED HITS
|
|
*
|
|
* See notes at top of aligner_sw_driver.h.
|
|
*
|
|
* REDUNDANT ALIGNMENTS
|
|
*
|
|
* See notes at top of aligner_sw_driver.h.
|
|
*
|
|
* MIXING PAIRED AND UNPAIRED ALIGNMENTS
|
|
*
|
|
* There are distinct paired-end alignment modes for the cases where (a) the
|
|
* user does or does not want to see unpaired alignments for individual mates
|
|
* when there are no reportable paired-end alignments involving both mates, and
|
|
* (b) the user does or does not want to see discordant paired-end alignments.
|
|
* The modes have implications for this function and for the AlnSinkWrap, since
|
|
* it affects when we're "done." Also, whether the user has asked us to report
|
|
* discordant alignments affects whether and how much searching for unpaired
|
|
* alignments we must do (i.e. if there are no paired-end alignments, we must
|
|
* at least do -m 1 for both mates).
|
|
*
|
|
* Mode 1: Just concordant paired-end. Print only concordant paired-end
|
|
* alignments. As soon as any limits (-k/-m/-M) are reached, stop.
|
|
*
|
|
* Mode 2: Concordant and discordant paired-end. If -k/-m/-M limits are
|
|
* reached for paired-end alignments, stop. Otherwise, if no paired-end
|
|
* alignments are found, align both mates in an unpaired -m 1 fashion. If
|
|
* there is exactly one unpaired alignment for each mate, report the
|
|
* combination as a discordant alignment.
|
|
*
|
|
* Mode 3: Concordant paired-end if possible, otherwise unpaired. If -k/-M
|
|
* limit is reached for paired-end alignmnts, stop. If -m limit is reached for
|
|
* paired-end alignments or no paired-end alignments are found, align both
|
|
* mates in an unpaired fashion. All the same settings governing validity and
|
|
* reportability in paired-end mode apply here too (-k/-m/-M/etc).
|
|
*
|
|
* Mode 4: Concordant or discordant paired-end if possible, otherwise unpaired.
|
|
* If -k/-M limit is reached for paired-end alignmnts, stop. If -m limit is
|
|
* reached for paired-end alignments or no paired-end alignments are found,
|
|
* align both mates in an unpaired fashion. If the -m limit was reached, there
|
|
* is no need to search for a discordant alignment, and unapired alignment can
|
|
* proceed as in Mode 3. If no paired-end alignments were found, then unpaired
|
|
* alignment proceeds as in Mode 3 but with this caveat: alignment must be at
|
|
* least as thorough as dictated by -m 1 up until the point where
|
|
*
|
|
* Print paired-end alignments when there are reportable paired-end
|
|
* alignments, otherwise report reportable unpaired alignments. If -k limit is
|
|
* reached for paired-end alignments, stop. If -m/-M limit is reached for
|
|
* paired-end alignments, stop searching for paired-end alignments and look
|
|
* only for unpaired alignments. If searching only for unpaired alignments,
|
|
* respect -k/-m/-M limits separately for both mates.
|
|
*
|
|
* The return value from the AlnSinkWrap's report member function must be
|
|
* specific enough to distinguish between:
|
|
*
|
|
* 1. Stop searching for paired-end alignments
|
|
* 2. Stop searching for alignments for unpaired alignments for mate #1
|
|
* 3. Stop searching for alignments for unpaired alignments for mate #2
|
|
* 4. Stop searching for any alignments
|
|
*
|
|
* Note that in Mode 2, options affecting validity and reportability of
|
|
* alignments apply . E.g. if -m 1 is specified
|
|
*
|
|
* WORKFLOW
|
|
*
|
|
* Our general approach to finding paired and unpaired alignments here
|
|
* is as follows:
|
|
*
|
|
* - For mate in mate1, mate2:
|
|
* - For each seed hit in mate:
|
|
* - Try to extend it into a full alignment; if we can't, continue
|
|
* to the next seed hit
|
|
* - Look for alignment for opposite mate; if we can't find one,
|
|
* -
|
|
* -
|
|
*
|
|
*/
|
|
template <typename index_t>
|
|
int SwDriver<index_t>::extendSeedsPaired(
|
|
Read& rd, // mate to align as anchor
|
|
Read& ord, // mate to align as opposite
|
|
bool anchor1, // true iff anchor mate is mate1
|
|
bool oppFilt, // true iff opposite mate was filtered out
|
|
SeedResults<index_t>& sh, // seed hits for anchor
|
|
const GFM<index_t>& gfmFw, // BWT
|
|
const GFM<index_t>* gfmBw, // BWT'
|
|
const BitPairReference& ref, // Reference strings
|
|
SwAligner& swa, // dynamic programming aligner for anchor
|
|
SwAligner& oswa, // dynamic programming aligner for opposite
|
|
const Scoring& sc, // scoring scheme
|
|
const PairedEndPolicy& pepol,// paired-end policy
|
|
int seedmms, // # mismatches allowed in seed
|
|
int seedlen, // length of seed
|
|
int seedival, // interval between seeds
|
|
TAlScore& minsc, // minimum score for valid anchor aln
|
|
TAlScore& ominsc, // minimum score for valid opposite aln
|
|
int nceil, // max # Ns permitted in ref for anchor
|
|
int onceil, // max # Ns permitted in ref for opposite
|
|
bool nofw, // don't align forward read
|
|
bool norc, // don't align revcomp read
|
|
size_t maxhalf, // max width in either direction for DP tables
|
|
bool doUngapped, // do ungapped alignment
|
|
size_t maxIters, // stop after this many seed-extend loop iters
|
|
size_t maxUg, // stop after this many ungaps
|
|
size_t maxDp, // stop after this many dps
|
|
size_t maxEeStreak, // stop after streak of this many end-to-end fails
|
|
size_t maxUgStreak, // stop after streak of this many ungap fails
|
|
size_t maxDpStreak, // stop after streak of this many dp fails
|
|
size_t maxMateStreak, // stop seed range after N mate-find fails
|
|
bool doExtend, // do seed extension
|
|
bool enable8, // use 8-bit SSE where possible
|
|
size_t cminlen, // use checkpointer if read longer than this
|
|
size_t cpow2, // interval between diagonals to checkpoint
|
|
bool doTri, // triangular mini-fills?
|
|
int tighten, // -M score tightening mode
|
|
AlignmentCacheIface<index_t>& ca, // alignment cache for seed hits
|
|
RandomSource& rnd, // pseudo-random source
|
|
WalkMetrics& wlm, // group walk left metrics
|
|
SwMetrics& swmSeed, // DP metrics for seed-extend
|
|
SwMetrics& swmMate, // DP metrics for mate finidng
|
|
PerReadMetrics& prm, // per-read metrics
|
|
AlnSinkWrap<index_t>* msink, // AlnSink wrapper for multiseed-style aligner
|
|
bool swMateImmediately, // whether to look for mate immediately
|
|
bool reportImmediately, // whether to report hits immediately to msink
|
|
bool discord, // look for discordant alignments?
|
|
bool mixed, // look for unpaired as well as paired alns?
|
|
bool& exhaustive)
|
|
{
|
|
bool all = msink->allHits();
|
|
// typedef std::pair<uint32_t, uint32_t> U32Pair;
|
|
|
|
assert(!reportImmediately || msink != NULL);
|
|
assert(!reportImmediately || !msink->maxed());
|
|
assert(!msink->state().doneWithMate(anchor1));
|
|
|
|
assert_geq(nceil, 0);
|
|
assert_geq(onceil, 0);
|
|
assert_leq((size_t)nceil, rd.length());
|
|
assert_leq((size_t)onceil, ord.length());
|
|
|
|
const index_t rdlen = rd.length();
|
|
const index_t ordlen = ord.length();
|
|
const TAlScore perfectScore = sc.perfectScore(rdlen);
|
|
const TAlScore operfectScore = sc.perfectScore(ordlen);
|
|
|
|
assert_leq(minsc, perfectScore);
|
|
assert(oppFilt || ominsc <= operfectScore);
|
|
|
|
TAlScore bestPairScore = perfectScore + operfectScore;
|
|
if(tighten > 0 && msink->Mmode() && msink->hasSecondBestPair()) {
|
|
// Paired-end alignments should have at least this score from now
|
|
TAlScore ps;
|
|
if(tighten == 1) {
|
|
ps = msink->bestPair();
|
|
} else if(tighten == 2) {
|
|
ps = msink->secondBestPair();
|
|
} else {
|
|
TAlScore diff = msink->bestPair() - msink->secondBestPair();
|
|
ps = msink->secondBestPair() + (diff * 3)/4;
|
|
}
|
|
if(tighten == 1 && ps < bestPairScore &&
|
|
msink->bestPair() == msink->secondBestPair())
|
|
{
|
|
ps++;
|
|
}
|
|
if(tighten >= 2 && ps < bestPairScore) {
|
|
ps++;
|
|
}
|
|
// Anchor mate must have score at least 'ps' minus the best possible
|
|
// score for the opposite mate.
|
|
TAlScore nc = ps - operfectScore;
|
|
if(nc > minsc) {
|
|
minsc = nc;
|
|
}
|
|
assert_leq(minsc, perfectScore);
|
|
}
|
|
|
|
DynProgFramer dpframe(!gReportOverhangs);
|
|
swa.reset();
|
|
oswa.reset();
|
|
|
|
// Initialize a set of GroupWalks, one for each seed. Also, intialize the
|
|
// accompanying lists of reference seed hits (satups*)
|
|
const index_t nsm = 5;
|
|
const index_t nonz = sh.nonzeroOffsets(); // non-zero positions
|
|
index_t eeHits = sh.numE2eHits();
|
|
bool eeMode = eeHits > 0;
|
|
bool firstEe = true;
|
|
bool firstExtend = true;
|
|
|
|
// Reset all the counters related to streaks
|
|
prm.nEeFail = 0;
|
|
prm.nUgFail = 0;
|
|
prm.nDpFail = 0;
|
|
|
|
index_t nelt = 0, neltLeft = 0;
|
|
const index_t rows = rdlen;
|
|
const index_t orows = ordlen;
|
|
index_t eltsDone = 0;
|
|
while(true) {
|
|
if(eeMode) {
|
|
if(firstEe) {
|
|
firstEe = false;
|
|
eeMode = eeSaTups(
|
|
rd, // read
|
|
sh, // seed hits to extend into full alignments
|
|
gfmFw, // BWT
|
|
ref, // Reference strings
|
|
rnd, // pseudo-random generator
|
|
wlm, // group walk left metrics
|
|
swmSeed, // seed-extend metrics
|
|
nelt, // out: # elements total
|
|
maxIters, // max elts to report
|
|
all); // report all hits
|
|
assert_eq(gws_.size(), rands_.size());
|
|
assert_eq(gws_.size(), satpos_.size());
|
|
neltLeft = nelt;
|
|
// Initialize list that contains the mate-finding failure
|
|
// streak for each range
|
|
mateStreaks_.resize(gws_.size());
|
|
mateStreaks_.fill(0);
|
|
} else {
|
|
eeMode = false;
|
|
}
|
|
}
|
|
if(!eeMode) {
|
|
if(nonz == 0) {
|
|
// No seed hits! Bail.
|
|
return EXTEND_EXHAUSTED_CANDIDATES;
|
|
}
|
|
if(msink->Mmode() && minsc == perfectScore) {
|
|
// Already found all perfect hits!
|
|
return EXTEND_PERFECT_SCORE;
|
|
}
|
|
if(firstExtend) {
|
|
nelt = 0;
|
|
prioritizeSATups(
|
|
rd, // read
|
|
sh, // seed hits to extend into full alignments
|
|
gfmFw, // BWT
|
|
gfmBw, // BWT'
|
|
ref, // Reference strings
|
|
seedmms, // # seed mismatches allowed
|
|
maxIters, // max rows to consider per position
|
|
doExtend, // extend out seeds
|
|
true, // square extended length
|
|
true, // square SA range size
|
|
nsm, // smallness threshold
|
|
ca, // alignment cache for seed hits
|
|
rnd, // pseudo-random generator
|
|
wlm, // group walk left metrics
|
|
prm, // per-read metrics
|
|
nelt, // out: # elements total
|
|
all); // report all hits?
|
|
assert_eq(gws_.size(), rands_.size());
|
|
assert_eq(gws_.size(), satpos_.size());
|
|
neltLeft = nelt;
|
|
firstExtend = false;
|
|
mateStreaks_.resize(gws_.size());
|
|
mateStreaks_.fill(0);
|
|
}
|
|
if(neltLeft == 0) {
|
|
// Finished examining gapped candidates
|
|
break;
|
|
}
|
|
}
|
|
for(index_t i = 0; i < gws_.size(); i++) {
|
|
if(eeMode && eehits_[i].score < minsc) {
|
|
return EXTEND_PERFECT_SCORE;
|
|
}
|
|
bool is_small = satpos_[i].sat.size() < nsm;
|
|
bool fw = satpos_[i].pos.fw;
|
|
index_t rdoff = satpos_[i].pos.rdoff;
|
|
index_t seedhitlen = satpos_[i].pos.seedlen;
|
|
if(!fw) {
|
|
// 'rdoff' and 'offidx' are with respect to the 5' end of
|
|
// the read. Here we convert rdoff to be with respect to
|
|
// the upstream (3') end of ther read.
|
|
rdoff = (index_t)(rdlen - rdoff - seedhitlen);
|
|
}
|
|
bool first = true;
|
|
// If the range is small, investigate all elements now. If the
|
|
// range is large, just investigate one and move on - we might come
|
|
// back to this range later.
|
|
while(!rands_[i].done() && (first || is_small || eeMode)) {
|
|
if(minsc == perfectScore) {
|
|
if(!eeMode || eehits_[i].score < perfectScore) {
|
|
return EXTEND_PERFECT_SCORE;
|
|
}
|
|
} else if(eeMode && eehits_[i].score < minsc) {
|
|
break;
|
|
}
|
|
if(prm.nExDps >= maxDp || prm.nMateDps >= maxDp) {
|
|
return EXTEND_EXCEEDED_HARD_LIMIT;
|
|
}
|
|
if(prm.nExUgs >= maxUg || prm.nMateUgs >= maxUg) {
|
|
return EXTEND_EXCEEDED_HARD_LIMIT;
|
|
}
|
|
if(prm.nExIters >= maxIters) {
|
|
return EXTEND_EXCEEDED_HARD_LIMIT;
|
|
}
|
|
if(eeMode && prm.nEeFail >= maxEeStreak) {
|
|
return EXTEND_EXCEEDED_SOFT_LIMIT;
|
|
}
|
|
if(!eeMode && prm.nDpFail >= maxDpStreak) {
|
|
return EXTEND_EXCEEDED_SOFT_LIMIT;
|
|
}
|
|
if(!eeMode && prm.nUgFail >= maxUgStreak) {
|
|
return EXTEND_EXCEEDED_SOFT_LIMIT;
|
|
}
|
|
if(mateStreaks_[i] >= maxMateStreak) {
|
|
// Don't try this seed range anymore
|
|
rands_[i].setDone();
|
|
assert(rands_[i].done());
|
|
break;
|
|
}
|
|
prm.nExIters++;
|
|
first = false;
|
|
assert(!gws_[i].done());
|
|
// Resolve next element offset
|
|
WalkResult<index_t> wr;
|
|
uint32_t elt = rands_[i].next(rnd);
|
|
SARangeWithOffs<TSlice, index_t> sa;
|
|
sa.topf = satpos_[i].sat.topf;
|
|
sa.len = satpos_[i].sat.key.len;
|
|
sa.offs = satpos_[i].sat.offs;
|
|
gws_[i].advanceElement((index_t)elt, gfmFw, ref, sa, gwstate_, wr, wlm, prm);
|
|
eltsDone++;
|
|
assert_gt(neltLeft, 0);
|
|
neltLeft--;
|
|
assert_neq((index_t)OFF_MASK, wr.toff);
|
|
index_t tidx = 0, toff = 0, tlen = 0;
|
|
bool straddled = false;
|
|
gfmFw.joinedToTextOff(
|
|
wr.elt.len,
|
|
wr.toff,
|
|
tidx,
|
|
toff,
|
|
tlen,
|
|
eeMode, // reject straddlers?
|
|
straddled); // straddled?
|
|
if(tidx == (index_t)OFF_MASK) {
|
|
// The seed hit straddled a reference boundary so the seed hit
|
|
// isn't valid
|
|
continue;
|
|
}
|
|
#ifndef NDEBUG
|
|
if(!eeMode && !straddled) { // Check that seed hit matches reference
|
|
uint64_t key = satpos_[i].sat.key.seq;
|
|
for(index_t k = 0; k < wr.elt.len; k++) {
|
|
int c = ref.getBase(tidx, toff + wr.elt.len - k - 1);
|
|
assert_leq(c, 3);
|
|
int ck = (int)(key & 3);
|
|
key >>= 2;
|
|
assert_eq(c, ck);
|
|
}
|
|
}
|
|
#endif
|
|
// Find offset of alignment's upstream base assuming net gaps=0
|
|
// between beginning of read and beginning of seed hit
|
|
int64_t refoff = (int64_t)toff - rdoff;
|
|
EIvalMergeListBinned& seenDiags = anchor1 ? seenDiags1_ : seenDiags2_;
|
|
// Coordinate of the seed hit w/r/t the pasted reference string
|
|
Coord refcoord(tidx, refoff, fw);
|
|
if(seenDiags.locusPresent(refcoord)) {
|
|
// Already handled alignments seeded on this diagonal
|
|
prm.nRedundants++;
|
|
swmSeed.rshit++;
|
|
continue;
|
|
}
|
|
// Now that we have a seed hit, there are many issues to solve
|
|
// before we have a completely framed dynamic programming problem.
|
|
// They include:
|
|
//
|
|
// 1. Setting reference offsets on either side of the seed hit,
|
|
// accounting for where the seed occurs in the read
|
|
// 2. Adjusting the width of the banded dynamic programming problem
|
|
// and adjusting reference bounds to allow for gaps in the
|
|
// alignment
|
|
// 3. Accounting for the edges of the reference, which can impact
|
|
// the width of the DP problem and reference bounds.
|
|
// 4. Perhaps filtering the problem down to a smaller problem based
|
|
// on what DPs we've already solved for this read
|
|
//
|
|
// We do #1 here, since it is simple and we have all the seed-hit
|
|
// information here. #2 and #3 are handled in the DynProgFramer.
|
|
int readGaps = 0, refGaps = 0;
|
|
bool ungapped = false;
|
|
if(!eeMode) {
|
|
readGaps = sc.maxReadGaps(minsc, rdlen);
|
|
refGaps = sc.maxRefGaps(minsc, rdlen);
|
|
ungapped = (readGaps == 0 && refGaps == 0);
|
|
}
|
|
int state = FOUND_NONE;
|
|
bool found = false;
|
|
// In unpaired mode, a seed extension is successful if it
|
|
// results in a full alignment that meets the minimum score
|
|
// threshold. In paired-end mode, a seed extension is
|
|
// successful if it results in a *full paired-end* alignment
|
|
// that meets the minimum score threshold.
|
|
if(eeMode) {
|
|
resEe_.reset();
|
|
resEe_.alres.reset();
|
|
const EEHit<index_t>& h = eehits_[i];
|
|
assert_leq(h.score, perfectScore);
|
|
resEe_.alres.setScore(AlnScore(h.score, h.ns(), 0));
|
|
resEe_.alres.setShape(
|
|
refcoord.ref(), // ref id
|
|
refcoord.off(), // 0-based ref offset
|
|
tlen, // reference length
|
|
fw, // aligned to Watson?
|
|
rdlen, // read length
|
|
true, // pretrim soft?
|
|
0, // pretrim 5' end
|
|
0, // pretrim 3' end
|
|
true, // alignment trim soft?
|
|
0, // alignment trim 5' end
|
|
0); // alignment trim 3' end
|
|
resEe_.alres.setRefNs(h.refns());
|
|
if(h.mms() > 0) {
|
|
assert_eq(1, h.mms());
|
|
assert_lt(h.e1.pos, rd.length());
|
|
resEe_.alres.ned().push_back(h.e1);
|
|
}
|
|
assert(resEe_.repOk(rd));
|
|
state = FOUND_EE;
|
|
found = true;
|
|
Interval refival(refcoord, 1);
|
|
seenDiags.add(refival);
|
|
prm.nExEes++;
|
|
prm.nEeFail++; // say it's failed until proven successful
|
|
prm.nExEeFails++;
|
|
} else if(doUngapped && ungapped) {
|
|
resUngap_.reset();
|
|
int al = swa.ungappedAlign(
|
|
fw ? rd.patFw : rd.patRc,
|
|
fw ? rd.qual : rd.qualRev,
|
|
refcoord,
|
|
ref,
|
|
tlen,
|
|
sc,
|
|
gReportOverhangs,
|
|
minsc, // minimum
|
|
resUngap_);
|
|
Interval refival(refcoord, 1);
|
|
seenDiags.add(refival);
|
|
prm.nExUgs++;
|
|
prm.nUgFail++; // say it's failed until proven successful
|
|
prm.nExUgFails++;
|
|
if(al == 0) {
|
|
swmSeed.ungapfail++;
|
|
continue;
|
|
} else if(al == -1) {
|
|
swmSeed.ungapnodec++;
|
|
} else {
|
|
found = true;
|
|
state = FOUND_UNGAPPED;
|
|
swmSeed.ungapsucc++;
|
|
}
|
|
}
|
|
int64_t pastedRefoff = (int64_t)wr.toff - rdoff;
|
|
DPRect rect;
|
|
if(state == FOUND_NONE) {
|
|
found = dpframe.frameSeedExtensionRect(
|
|
refoff, // ref offset implied by seed hit assuming no gaps
|
|
rows, // length of read sequence used in DP table
|
|
tlen, // length of reference
|
|
readGaps, // max # of read gaps permitted in opp mate alignment
|
|
refGaps, // max # of ref gaps permitted in opp mate alignment
|
|
(size_t)nceil, // # Ns permitted
|
|
maxhalf, // max width in either direction
|
|
rect); // DP rectangle
|
|
assert(rect.repOk());
|
|
// Add the seed diagonal at least
|
|
seenDiags.add(Interval(refcoord, 1));
|
|
if(!found) {
|
|
continue;
|
|
}
|
|
}
|
|
int64_t leftShift = refoff - rect.refl;
|
|
size_t nwindow = 0;
|
|
if(toff >= rect.refl) {
|
|
nwindow = (size_t)(toff - rect.refl);
|
|
}
|
|
// NOTE: We might be taking off more than we should because the
|
|
// pasted string omits non-A/C/G/T characters, but we included them
|
|
// when calculating leftShift. We'll account for this later.
|
|
pastedRefoff -= leftShift;
|
|
size_t nsInLeftShift = 0;
|
|
if(state == FOUND_NONE) {
|
|
if(!swa.initedRead()) {
|
|
// Initialize the aligner with a new read
|
|
swa.initRead(
|
|
rd.patFw, // fw version of query
|
|
rd.patRc, // rc version of query
|
|
rd.qual, // fw version of qualities
|
|
rd.qualRev,// rc version of qualities
|
|
0, // off of first char in 'rd' to consider
|
|
rdlen, // off of last char (excl) in 'rd' to consider
|
|
sc); // scoring scheme
|
|
}
|
|
swa.initRef(
|
|
fw, // whether to align forward or revcomp read
|
|
tidx, // reference aligned against
|
|
rect, // DP rectangle
|
|
ref, // Reference strings
|
|
tlen, // length of reference sequence
|
|
sc, // scoring scheme
|
|
minsc, // minimum score permitted
|
|
enable8, // use 8-bit SSE if possible?
|
|
cminlen, // minimum length for using checkpointing scheme
|
|
cpow2, // interval b/t checkpointed diags; 1 << this
|
|
doTri, // triangular mini-fills?
|
|
true, // this is a seed extension - not finding a mate
|
|
nwindow,
|
|
nsInLeftShift);
|
|
// Because of how we framed the problem, we can say that we've
|
|
// exhaustively scored the seed diagonal as well as maxgaps
|
|
// diagonals on either side
|
|
Interval refival(tidx, 0, fw, 0);
|
|
rect.initIval(refival);
|
|
seenDiags.add(refival);
|
|
// Now fill the dynamic programming matrix and return true iff
|
|
// there is at least one valid alignment
|
|
TAlScore bestCell = std::numeric_limits<TAlScore>::min();
|
|
found = swa.align(rnd, bestCell);
|
|
swmSeed.tallyGappedDp(readGaps, refGaps);
|
|
prm.nExDps++;
|
|
prm.nDpFail++; // failed until proven successful
|
|
prm.nExDpFails++; // failed until proven successful
|
|
if(!found) {
|
|
TAlScore bestLast = anchor1 ? prm.bestLtMinscMate1 : prm.bestLtMinscMate2;
|
|
if(bestCell > std::numeric_limits<TAlScore>::min() && bestCell > bestLast) {
|
|
if(anchor1) {
|
|
prm.bestLtMinscMate1 = bestCell;
|
|
} else {
|
|
prm.bestLtMinscMate2 = bestCell;
|
|
}
|
|
}
|
|
continue; // Look for more anchor alignments
|
|
}
|
|
}
|
|
bool firstInner = true;
|
|
bool foundConcordant = false;
|
|
while(true) {
|
|
assert(found);
|
|
SwResult *res = NULL;
|
|
if(state == FOUND_EE) {
|
|
if(!firstInner) {
|
|
break;
|
|
}
|
|
res = &resEe_;
|
|
assert(res->repOk(rd));
|
|
} else if(state == FOUND_UNGAPPED) {
|
|
if(!firstInner) {
|
|
break;
|
|
}
|
|
res = &resUngap_;
|
|
assert(res->repOk(rd));
|
|
} else {
|
|
resGap_.reset();
|
|
assert(resGap_.empty());
|
|
if(swa.done()) {
|
|
break;
|
|
}
|
|
swa.nextAlignment(resGap_, minsc, rnd);
|
|
found = !resGap_.empty();
|
|
if(!found) {
|
|
break;
|
|
}
|
|
res = &resGap_;
|
|
assert(res->repOk(rd));
|
|
}
|
|
// TODO: If we're just taking anchor alignments out of the
|
|
// same rectangle, aren't we getting very similar
|
|
// rectangles for the opposite mate each time? Seems like
|
|
// we could save some work by detecting this.
|
|
assert(res != NULL);
|
|
firstInner = false;
|
|
assert(res->alres.matchesRef(
|
|
rd,
|
|
ref,
|
|
tmp_rf_,
|
|
tmp_rdseq_,
|
|
tmp_qseq_,
|
|
raw_refbuf_,
|
|
raw_destU32_,
|
|
raw_matches_,
|
|
tmp_reflens_,
|
|
tmp_refoffs_));
|
|
Interval refival(tidx, 0, fw, tlen);
|
|
assert_gt(res->alres.refExtent(), 0);
|
|
if(gReportOverhangs &&
|
|
!refival.containsIgnoreOrient(res->alres.refival()))
|
|
{
|
|
res->alres.clipOutside(true, 0, tlen);
|
|
if(res->alres.refExtent() == 0) {
|
|
continue;
|
|
}
|
|
}
|
|
assert(gReportOverhangs ||
|
|
refival.containsIgnoreOrient(res->alres.refival()));
|
|
// Did the alignment fall entirely outside the reference?
|
|
if(!refival.overlapsIgnoreOrient(res->alres.refival())) {
|
|
continue;
|
|
}
|
|
// Is this alignment redundant with one we've seen previously?
|
|
if(redAnchor_.overlap(res->alres)) {
|
|
continue;
|
|
}
|
|
redAnchor_.add(res->alres);
|
|
// Annotate the AlnRes object with some key parameters
|
|
// that were used to obtain the alignment.
|
|
res->alres.setParams(
|
|
seedmms, // # mismatches allowed in seed
|
|
seedlen, // length of seed
|
|
seedival, // interval between seeds
|
|
minsc); // minimum score for valid alignment
|
|
bool foundMate = false;
|
|
TRefOff off = res->alres.refoff();
|
|
if( msink->state().doneWithMate(!anchor1) &&
|
|
!msink->state().doneWithMate( anchor1))
|
|
{
|
|
// We're done with the opposite mate but not with the
|
|
// anchor mate; don't try to mate up the anchor.
|
|
swMateImmediately = false;
|
|
}
|
|
if(found && swMateImmediately) {
|
|
assert(!msink->state().doneWithMate(!anchor1));
|
|
bool oleft = false, ofw = false;
|
|
int64_t oll = 0, olr = 0, orl = 0, orr = 0;
|
|
assert(!msink->state().done());
|
|
foundMate = !oppFilt;
|
|
TAlScore ominsc_cur = ominsc;
|
|
//bool oungapped = false;
|
|
int oreadGaps = 0, orefGaps = 0;
|
|
//int oungappedAlign = -1; // defer
|
|
if(foundMate) {
|
|
// Adjust ominsc given the alignment score of the
|
|
// anchor mate
|
|
ominsc_cur = ominsc;
|
|
if(tighten > 0 && msink->Mmode() && msink->hasSecondBestPair()) {
|
|
// Paired-end alignments should have at least this score from now
|
|
TAlScore ps;
|
|
if(tighten == 1) {
|
|
ps = msink->bestPair();
|
|
} else if(tighten == 2) {
|
|
ps = msink->secondBestPair();
|
|
} else {
|
|
TAlScore diff = msink->bestPair() - msink->secondBestPair();
|
|
ps = msink->secondBestPair() + (diff * 3)/4;
|
|
}
|
|
if(tighten == 1 && ps < bestPairScore &&
|
|
msink->bestPair() == msink->secondBestPair())
|
|
{
|
|
ps++;
|
|
}
|
|
if(tighten >= 2 && ps < bestPairScore) {
|
|
ps++;
|
|
}
|
|
// Anchor mate must have score at least 'ps' minus the best possible
|
|
// score for the opposite mate.
|
|
TAlScore nc = ps - res->alres.score().score();
|
|
if(nc > ominsc_cur) {
|
|
ominsc_cur = nc;
|
|
assert_leq(ominsc_cur, operfectScore);
|
|
}
|
|
}
|
|
oreadGaps = sc.maxReadGaps(ominsc_cur, ordlen);
|
|
orefGaps = sc.maxRefGaps (ominsc_cur, ordlen);
|
|
//oungapped = (oreadGaps == 0 && orefGaps == 0);
|
|
// TODO: Something lighter-weight than DP to scan
|
|
// for other mate??
|
|
//if(oungapped) {
|
|
// oresUngap_.reset();
|
|
// oungappedAlign = oswa.ungappedAlign(
|
|
// ofw ? ord.patFw : ord.patRc,
|
|
// ofw ? ord.qual : ord.qualRev,
|
|
// orefcoord,
|
|
// ref,
|
|
// otlen,
|
|
// sc,
|
|
// gReportOverhangs,
|
|
// ominsc_cur,
|
|
// 0,
|
|
// oresUngap_);
|
|
//}
|
|
foundMate = pepol.otherMate(
|
|
anchor1, // anchor mate is mate #1?
|
|
fw, // anchor aligned to Watson?
|
|
off, // offset of anchor mate
|
|
orows + oreadGaps, // max # columns spanned by alignment
|
|
tlen, // reference length
|
|
anchor1 ? rd.length() : ord.length(), // mate 1 len
|
|
anchor1 ? ord.length() : rd.length(), // mate 2 len
|
|
oleft, // out: look left for opposite mate?
|
|
oll,
|
|
olr,
|
|
orl,
|
|
orr,
|
|
ofw);
|
|
}
|
|
DPRect orect;
|
|
if(foundMate) {
|
|
foundMate = dpframe.frameFindMateRect(
|
|
!oleft, // true iff anchor alignment is to the left
|
|
oll, // leftmost Watson off for LHS of opp aln
|
|
olr, // rightmost Watson off for LHS of opp aln
|
|
orl, // leftmost Watson off for RHS of opp aln
|
|
orr, // rightmost Watson off for RHS of opp aln
|
|
orows, // length of opposite mate
|
|
tlen, // length of reference sequence aligned to
|
|
oreadGaps, // max # of read gaps in opp mate aln
|
|
orefGaps, // max # of ref gaps in opp mate aln
|
|
(size_t)onceil, // max # Ns on opp mate
|
|
maxhalf, // max width in either direction
|
|
orect); // DP rectangle
|
|
assert(!foundMate || orect.refr >= orect.refl);
|
|
}
|
|
if(foundMate) {
|
|
oresGap_.reset();
|
|
assert(oresGap_.empty());
|
|
if(!oswa.initedRead()) {
|
|
oswa.initRead(
|
|
ord.patFw, // read to align
|
|
ord.patRc, // qualities
|
|
ord.qual, // read to align
|
|
ord.qualRev,// qualities
|
|
0, // off of first char to consider
|
|
ordlen, // off of last char (ex) to consider
|
|
sc); // scoring scheme
|
|
}
|
|
// Given the boundaries defined by refi and reff, initilize
|
|
// the SwAligner with the dynamic programming problem that
|
|
// aligns the read to this reference stretch.
|
|
size_t onsInLeftShift = 0;
|
|
assert_geq(orect.refr, orect.refl);
|
|
oswa.initRef(
|
|
ofw, // align forward or revcomp read?
|
|
tidx, // reference aligned against
|
|
orect, // DP rectangle
|
|
ref, // Reference strings
|
|
tlen, // length of reference sequence
|
|
sc, // scoring scheme
|
|
ominsc_cur,// min score for valid alignments
|
|
enable8, // use 8-bit SSE if possible?
|
|
cminlen, // minimum length for using checkpointing scheme
|
|
cpow2, // interval b/t checkpointed diags; 1 << this
|
|
doTri, // triangular mini-fills?
|
|
false, // this is finding a mate - not seed ext
|
|
0, // nwindow?
|
|
onsInLeftShift);
|
|
// TODO: Can't we add some diagonals to the
|
|
// opposite mate's seenDiags when we fill in the
|
|
// opposite mate's DP? Or can we? We might want
|
|
// to use this again as an anchor - will that still
|
|
// happen? Also, isn't there a problem with
|
|
// consistency of the minimum score? Minimum score
|
|
// here depends in part on the score of the anchor
|
|
// alignment here, but it won't when the current
|
|
// opposite becomes the anchor.
|
|
|
|
// Because of how we framed the problem, we can say
|
|
// that we've exhaustively explored the "core"
|
|
// diagonals
|
|
//Interval orefival(tidx, 0, ofw, 0);
|
|
//orect.initIval(orefival);
|
|
//oseenDiags.add(orefival);
|
|
|
|
// Now fill the dynamic programming matrix, return true
|
|
// iff there is at least one valid alignment
|
|
TAlScore bestCell = std::numeric_limits<TAlScore>::min();
|
|
foundMate = oswa.align(rnd, bestCell);
|
|
prm.nMateDps++;
|
|
swmMate.tallyGappedDp(oreadGaps, orefGaps);
|
|
if(!foundMate) {
|
|
TAlScore bestLast = anchor1 ? prm.bestLtMinscMate2 : prm.bestLtMinscMate1;
|
|
if(bestCell > std::numeric_limits<TAlScore>::min() && bestCell > bestLast) {
|
|
if(anchor1) {
|
|
prm.bestLtMinscMate2 = bestCell;
|
|
} else {
|
|
prm.bestLtMinscMate1 = bestCell;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
bool didAnchor = false;
|
|
do {
|
|
oresGap_.reset();
|
|
assert(oresGap_.empty());
|
|
if(foundMate && oswa.done()) {
|
|
foundMate = false;
|
|
} else if(foundMate) {
|
|
oswa.nextAlignment(oresGap_, ominsc_cur, rnd);
|
|
foundMate = !oresGap_.empty();
|
|
assert(!foundMate || oresGap_.alres.matchesRef(
|
|
ord,
|
|
ref,
|
|
tmp_rf_,
|
|
tmp_rdseq_,
|
|
tmp_qseq_,
|
|
raw_refbuf_,
|
|
raw_destU32_,
|
|
raw_matches_,
|
|
tmp_reflens_,
|
|
tmp_refoffs_));
|
|
}
|
|
if(foundMate) {
|
|
// Redundant with one we've seen previously?
|
|
if(!redAnchor_.overlap(oresGap_.alres)) {
|
|
redAnchor_.add(oresGap_.alres);
|
|
}
|
|
assert_eq(ofw, oresGap_.alres.fw());
|
|
// Annotate the AlnRes object with some key parameters
|
|
// that were used to obtain the alignment.
|
|
oresGap_.alres.setParams(
|
|
seedmms, // # mismatches allowed in seed
|
|
seedlen, // length of seed
|
|
seedival, // interval between seeds
|
|
ominsc); // minimum score for valid alignment
|
|
assert_gt(oresGap_.alres.refExtent(), 0);
|
|
if(gReportOverhangs &&
|
|
!refival.containsIgnoreOrient(oresGap_.alres.refival()))
|
|
{
|
|
oresGap_.alres.clipOutside(true, 0, tlen);
|
|
foundMate = oresGap_.alres.refExtent() > 0;
|
|
}
|
|
if(foundMate &&
|
|
((!gReportOverhangs &&
|
|
!refival.containsIgnoreOrient(oresGap_.alres.refival())) ||
|
|
!refival.overlapsIgnoreOrient(oresGap_.alres.refival())))
|
|
{
|
|
foundMate = false;
|
|
}
|
|
}
|
|
ASSERT_ONLY(TRefId refid);
|
|
TRefOff off1, off2;
|
|
size_t len1, len2;
|
|
bool fw1, fw2;
|
|
int pairCl = PE_ALS_DISCORD;
|
|
if(foundMate) {
|
|
ASSERT_ONLY(refid =) res->alres.refid();
|
|
assert_eq(refid, oresGap_.alres.refid());
|
|
off1 = anchor1 ? off : oresGap_.alres.refoff();
|
|
off2 = anchor1 ? oresGap_.alres.refoff() : off;
|
|
len1 = anchor1 ?
|
|
res->alres.refExtent() : oresGap_.alres.refExtent();
|
|
len2 = anchor1 ?
|
|
oresGap_.alres.refExtent() : res->alres.refExtent();
|
|
fw1 = anchor1 ? res->alres.fw() : oresGap_.alres.fw();
|
|
fw2 = anchor1 ? oresGap_.alres.fw() : res->alres.fw();
|
|
// Check that final mate alignments are consistent with
|
|
// paired-end fragment constraints
|
|
pairCl = pepol.peClassifyPair(
|
|
off1,
|
|
len1,
|
|
fw1,
|
|
off2,
|
|
len2,
|
|
fw2);
|
|
// Instead of trying
|
|
//foundMate = pairCl != PE_ALS_DISCORD;
|
|
}
|
|
if(msink->state().doneConcordant()) {
|
|
foundMate = false;
|
|
}
|
|
if(reportImmediately) {
|
|
if(foundMate) {
|
|
// Report pair to the AlnSinkWrap
|
|
assert(!msink->state().doneConcordant());
|
|
assert(msink != NULL);
|
|
assert(res->repOk());
|
|
assert(oresGap_.repOk());
|
|
// Report an unpaired alignment
|
|
assert(!msink->maxed());
|
|
assert(!msink->state().done());
|
|
bool doneUnpaired = false;
|
|
//if(mixed || discord) {
|
|
// Report alignment for mate #1 as an
|
|
// unpaired alignment.
|
|
if(!anchor1 || !didAnchor) {
|
|
if(anchor1) {
|
|
didAnchor = true;
|
|
}
|
|
const AlnRes& r1 = anchor1 ?
|
|
res->alres : oresGap_.alres;
|
|
if(!redMate1_.overlap(r1)) {
|
|
redMate1_.add(r1);
|
|
if(msink->report(0, &r1, NULL)) {
|
|
doneUnpaired = true; // Short-circuited
|
|
}
|
|
}
|
|
}
|
|
// Report alignment for mate #2 as an
|
|
// unpaired alignment.
|
|
if(anchor1 || !didAnchor) {
|
|
if(!anchor1) {
|
|
didAnchor = true;
|
|
}
|
|
const AlnRes& r2 = anchor1 ?
|
|
oresGap_.alres : res->alres;
|
|
if(!redMate2_.overlap(r2)) {
|
|
redMate2_.add(r2);
|
|
if(msink->report(0, NULL, &r2)) {
|
|
doneUnpaired = true; // Short-circuited
|
|
}
|
|
}
|
|
}
|
|
//} // if(mixed || discord)
|
|
bool donePaired = false;
|
|
if(pairCl != PE_ALS_DISCORD) {
|
|
foundConcordant = true;
|
|
if(msink->report(
|
|
0,
|
|
anchor1 ? &res->alres : &oresGap_.alres,
|
|
anchor1 ? &oresGap_.alres : &res->alres))
|
|
{
|
|
// Short-circuited because a limit, e.g.
|
|
// -k, -m or -M, was exceeded
|
|
donePaired = true;
|
|
} else {
|
|
if(tighten > 0 && msink->Mmode() && msink->hasSecondBestPair()) {
|
|
// Paired-end alignments should have at least this score from now
|
|
TAlScore ps;
|
|
if(tighten == 1) {
|
|
ps = msink->bestPair();
|
|
} else if(tighten == 2) {
|
|
ps = msink->secondBestPair();
|
|
} else {
|
|
TAlScore diff = msink->bestPair() - msink->secondBestPair();
|
|
ps = msink->secondBestPair() + (diff * 3)/4;
|
|
}
|
|
if(tighten == 1 && ps < bestPairScore &&
|
|
msink->bestPair() == msink->secondBestPair())
|
|
{
|
|
ps++;
|
|
}
|
|
if(tighten >= 2 && ps < bestPairScore) {
|
|
ps++;
|
|
}
|
|
// Anchor mate must have score at least 'ps' minus the best possible
|
|
// score for the opposite mate.
|
|
TAlScore nc = ps - operfectScore;
|
|
if(nc > minsc) {
|
|
minsc = nc;
|
|
assert_leq(minsc, perfectScore);
|
|
if(minsc > res->alres.score().score()) {
|
|
// We're done with this anchor
|
|
break;
|
|
}
|
|
}
|
|
assert_leq(minsc, perfectScore);
|
|
}
|
|
}
|
|
} // if(pairCl != PE_ALS_DISCORD)
|
|
if(donePaired || doneUnpaired) {
|
|
return EXTEND_POLICY_FULFILLED;
|
|
}
|
|
if(msink->state().doneWithMate(anchor1)) {
|
|
// We're now done with the mate that we're
|
|
// currently using as our anchor. We're not
|
|
// with the read overall.
|
|
return EXTEND_POLICY_FULFILLED;
|
|
}
|
|
} else if((mixed || discord) && !didAnchor) {
|
|
didAnchor = true;
|
|
// Report unpaired hit for anchor
|
|
assert(msink != NULL);
|
|
assert(res->repOk());
|
|
// Check that alignment accurately reflects the
|
|
// reference characters aligned to
|
|
assert(res->alres.matchesRef(
|
|
rd,
|
|
ref,
|
|
tmp_rf_,
|
|
tmp_rdseq_,
|
|
tmp_qseq_,
|
|
raw_refbuf_,
|
|
raw_destU32_,
|
|
raw_matches_,
|
|
tmp_reflens_,
|
|
tmp_refoffs_));
|
|
// Report an unpaired alignment
|
|
assert(!msink->maxed());
|
|
assert(!msink->state().done());
|
|
// Report alignment for mate #1 as an
|
|
// unpaired alignment.
|
|
if(!msink->state().doneUnpaired(anchor1)) {
|
|
const AlnRes& r = res->alres;
|
|
RedundantAlns& red = anchor1 ? redMate1_ : redMate2_;
|
|
const AlnRes* r1 = anchor1 ? &res->alres : NULL;
|
|
const AlnRes* r2 = anchor1 ? NULL : &res->alres;
|
|
if(!red.overlap(r)) {
|
|
red.add(r);
|
|
if(msink->report(0, r1, r2)) {
|
|
return EXTEND_POLICY_FULFILLED; // Short-circuited
|
|
}
|
|
}
|
|
}
|
|
if(msink->state().doneWithMate(anchor1)) {
|
|
// Done with mate, but not read overall
|
|
return EXTEND_POLICY_FULFILLED;
|
|
}
|
|
}
|
|
}
|
|
} while(!oresGap_.empty());
|
|
} // if(found && swMateImmediately)
|
|
else if(found) {
|
|
assert(!msink->state().doneWithMate(anchor1));
|
|
// We found an anchor alignment but did not attempt to find
|
|
// an alignment for the opposite mate (probably because
|
|
// we're done with it)
|
|
if(reportImmediately && (mixed || discord)) {
|
|
// Report unpaired hit for anchor
|
|
assert(msink != NULL);
|
|
assert(res->repOk());
|
|
// Check that alignment accurately reflects the
|
|
// reference characters aligned to
|
|
assert(res->alres.matchesRef(
|
|
rd,
|
|
ref,
|
|
tmp_rf_,
|
|
tmp_rdseq_,
|
|
tmp_qseq_,
|
|
raw_refbuf_,
|
|
raw_destU32_,
|
|
raw_matches_,
|
|
tmp_reflens_,
|
|
tmp_refoffs_));
|
|
// Report an unpaired alignment
|
|
assert(!msink->maxed());
|
|
assert(!msink->state().done());
|
|
// Report alignment for mate #1 as an
|
|
// unpaired alignment.
|
|
if(!msink->state().doneUnpaired(anchor1)) {
|
|
const AlnRes& r = res->alres;
|
|
RedundantAlns& red = anchor1 ? redMate1_ : redMate2_;
|
|
const AlnRes* r1 = anchor1 ? &res->alres : NULL;
|
|
const AlnRes* r2 = anchor1 ? NULL : &res->alres;
|
|
if(!red.overlap(r)) {
|
|
red.add(r);
|
|
if(msink->report(0, r1, r2)) {
|
|
return EXTEND_POLICY_FULFILLED; // Short-circuited
|
|
}
|
|
}
|
|
}
|
|
if(msink->state().doneWithMate(anchor1)) {
|
|
// Done with mate, but not read overall
|
|
return EXTEND_POLICY_FULFILLED;
|
|
}
|
|
}
|
|
}
|
|
} // while(true)
|
|
|
|
if(foundConcordant) {
|
|
prm.nMateDpSuccs++;
|
|
mateStreaks_[i] = 0;
|
|
// Register this as a success. Now we need to
|
|
// make the streak variables reflect the
|
|
// success.
|
|
if(state == FOUND_UNGAPPED) {
|
|
assert_gt(prm.nUgFail, 0);
|
|
assert_gt(prm.nExUgFails, 0);
|
|
prm.nExUgFails--;
|
|
prm.nExUgSuccs++;
|
|
prm.nUgLastSucc = prm.nExUgs-1;
|
|
if(prm.nUgFail > prm.nUgFailStreak) {
|
|
prm.nUgFailStreak = prm.nUgFail;
|
|
}
|
|
prm.nUgFail = 0;
|
|
} else if(state == FOUND_EE) {
|
|
assert_gt(prm.nEeFail, 0);
|
|
assert_gt(prm.nExEeFails, 0);
|
|
prm.nExEeFails--;
|
|
prm.nExEeSuccs++;
|
|
prm.nEeLastSucc = prm.nExEes-1;
|
|
if(prm.nEeFail > prm.nEeFailStreak) {
|
|
prm.nEeFailStreak = prm.nEeFail;
|
|
}
|
|
prm.nEeFail = 0;
|
|
} else {
|
|
assert_gt(prm.nDpFail, 0);
|
|
assert_gt(prm.nExDpFails, 0);
|
|
prm.nExDpFails--;
|
|
prm.nExDpSuccs++;
|
|
prm.nDpLastSucc = prm.nExDps-1;
|
|
if(prm.nDpFail > prm.nDpFailStreak) {
|
|
prm.nDpFailStreak = prm.nDpFail;
|
|
}
|
|
prm.nDpFail = 0;
|
|
}
|
|
} else {
|
|
prm.nMateDpFails++;
|
|
mateStreaks_[i]++;
|
|
}
|
|
// At this point we know that we aren't bailing, and will continue to resolve seed hits.
|
|
|
|
} // while(!gw.done())
|
|
} // for(size_t i = 0; i < gws_.size(); i++)
|
|
}
|
|
return EXTEND_EXHAUSTED_CANDIDATES;
|
|
}
|
|
|
|
#endif /*ALIGNER_SW_DRIVER_H_*/
|