1801 lines
44 KiB
C++
1801 lines
44 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/>.
|
|
*/
|
|
|
|
#ifndef PAT_H_
|
|
#define PAT_H_
|
|
|
|
#include <cassert>
|
|
#include <cmath>
|
|
#include <stdexcept>
|
|
#include <vector>
|
|
#include <string>
|
|
#include <cstring>
|
|
#include <ctype.h>
|
|
#include <fstream>
|
|
#include "alphabet.h"
|
|
#include "assert_helpers.h"
|
|
#include "tokenize.h"
|
|
#include "random_source.h"
|
|
#include "threading.h"
|
|
#include "filebuf.h"
|
|
#include "qual.h"
|
|
#include "search_globals.h"
|
|
#include "sstring.h"
|
|
#include "ds.h"
|
|
#include "read.h"
|
|
#include "util.h"
|
|
|
|
extern bool threeN;
|
|
/**
|
|
* Classes and routines for reading reads from various input sources.
|
|
*/
|
|
|
|
using namespace std;
|
|
|
|
/**
|
|
* Calculate a per-read random seed based on a combination of
|
|
* the read data (incl. sequence, name, quals) and the global
|
|
* seed in '_randSeed'.
|
|
*/
|
|
static inline uint32_t genRandSeed(const BTDnaString& qry,
|
|
const BTString& qual,
|
|
const BTString& name,
|
|
uint32_t seed)
|
|
{
|
|
// Calculate a per-read random seed based on a combination of
|
|
// the read data (incl. sequence, name, quals) and the global
|
|
// seed
|
|
uint32_t rseed = (seed + 101) * 59 * 61 * 67 * 71 * 73 * 79 * 83;
|
|
size_t qlen = qry.length();
|
|
// Throw all the characters of the read into the random seed
|
|
for(size_t i = 0; i < qlen; i++) {
|
|
int p = (int)qry[i];
|
|
assert_leq(p, 4);
|
|
size_t off = ((i & 15) << 1);
|
|
rseed ^= (p << off);
|
|
}
|
|
// Throw all the quality values for the read into the random
|
|
// seed
|
|
for(size_t i = 0; i < qlen; i++) {
|
|
int p = (int)qual[i];
|
|
assert_leq(p, 255);
|
|
size_t off = ((i & 3) << 3);
|
|
rseed ^= (p << off);
|
|
}
|
|
// Throw all the characters in the read name into the random
|
|
// seed
|
|
size_t namelen = name.length();
|
|
for(size_t i = 0; i < namelen; i++) {
|
|
int p = (int)name[i];
|
|
if(p == '/') break;
|
|
assert_leq(p, 255);
|
|
size_t off = ((i & 3) << 3);
|
|
rseed ^= (p << off);
|
|
}
|
|
return rseed;
|
|
}
|
|
|
|
/**
|
|
* Parameters affecting how reads and read in.
|
|
*/
|
|
struct PatternParams {
|
|
PatternParams(
|
|
int format_,
|
|
bool fileParallel_,
|
|
uint32_t seed_,
|
|
bool useSpinlock_,
|
|
bool solexa64_,
|
|
bool phred64_,
|
|
bool intQuals_,
|
|
bool fuzzy_,
|
|
int sampleLen_,
|
|
int sampleFreq_,
|
|
uint32_t skip_) :
|
|
format(format_),
|
|
fileParallel(fileParallel_),
|
|
seed(seed_),
|
|
useSpinlock(useSpinlock_),
|
|
solexa64(solexa64_),
|
|
phred64(phred64_),
|
|
intQuals(intQuals_),
|
|
fuzzy(fuzzy_),
|
|
sampleLen(sampleLen_),
|
|
sampleFreq(sampleFreq_),
|
|
skip(skip_) { }
|
|
|
|
int format; // file format
|
|
bool fileParallel; // true -> wrap files with separate PairedPatternSources
|
|
uint32_t seed; // pseudo-random seed
|
|
bool useSpinlock; // use spin locks instead of pthreads
|
|
bool solexa64; // true -> qualities are on solexa64 scale
|
|
bool phred64; // true -> qualities are on phred64 scale
|
|
bool intQuals; // true -> qualities are space-separated numbers
|
|
bool fuzzy; // true -> try to parse fuzzy fastq
|
|
int sampleLen; // length of sampled reads for FastaContinuous...
|
|
int sampleFreq; // frequency of sampled reads for FastaContinuous...
|
|
uint32_t skip; // skip the first 'skip' patterns
|
|
};
|
|
|
|
/**
|
|
* Encapsulates a synchronized source of patterns; usually a file.
|
|
* Optionally reverses reads and quality strings before returning them,
|
|
* though that is usually more efficiently done by the concrete
|
|
* subclass. Concrete subclasses should delimit critical sections with
|
|
* calls to lock() and unlock().
|
|
*/
|
|
class PatternSource {
|
|
|
|
public:
|
|
|
|
PatternSource(const PatternParams& p) :
|
|
seed_(p.seed),
|
|
readCnt_(0),
|
|
numWrappers_(0),
|
|
doLocking_(true),
|
|
useSpinlock_(p.useSpinlock),
|
|
mutex()
|
|
{
|
|
}
|
|
|
|
virtual ~PatternSource() { }
|
|
|
|
/**
|
|
* Call this whenever this PatternSource is wrapped by a new
|
|
* WrappedPatternSourcePerThread. This helps us keep track of
|
|
* whether locks will be contended.
|
|
*/
|
|
void addWrapper() {
|
|
lock();
|
|
numWrappers_++;
|
|
unlock();
|
|
}
|
|
|
|
/**
|
|
* The main member function for dispensing patterns.
|
|
*
|
|
* Returns true iff a pair was parsed succesfully.
|
|
*/
|
|
virtual bool nextReadPair(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired,
|
|
bool fixName);
|
|
|
|
/**
|
|
* The main member function for dispensing patterns.
|
|
*/
|
|
virtual bool nextRead(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done);
|
|
|
|
/**
|
|
* Implementation to be provided by concrete subclasses. An
|
|
* implementation for this member is only relevant for formats that
|
|
* can read in a pair of reads in a single transaction with a
|
|
* single input source. If paired-end input is given as a pair of
|
|
* parallel files, this member should throw an error and exit.
|
|
*/
|
|
virtual bool nextReadPairImpl(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired) = 0;
|
|
|
|
/**
|
|
* Implementation to be provided by concrete subclasses. An
|
|
* implementation for this member is only relevant for formats
|
|
* where individual input sources look like single-end-read
|
|
* sources, e.g., formats where paired-end reads are specified in
|
|
* parallel read files.
|
|
*/
|
|
virtual bool nextReadImpl(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done) = 0;
|
|
|
|
/// Reset state to start over again with the first read
|
|
virtual void reset() { readCnt_ = 0; }
|
|
|
|
/**
|
|
* Concrete subclasses call lock() to enter a critical region.
|
|
* What constitutes a critical region depends on the subclass.
|
|
*/
|
|
void lock() {
|
|
if(!doLocking_) return; // no contention
|
|
mutex.lock();
|
|
}
|
|
|
|
/**
|
|
* Concrete subclasses call unlock() to exit a critical region
|
|
* What constitutes a critical region depends on the subclass.
|
|
*/
|
|
void unlock() {
|
|
if(!doLocking_) return; // no contention
|
|
mutex.unlock();
|
|
}
|
|
|
|
/**
|
|
* Return a new dynamically allocated PatternSource for the given
|
|
* format, using the given list of strings as the filenames to read
|
|
* from or as the sequences themselves (i.e. if -c was used).
|
|
*/
|
|
static PatternSource* patsrcFromStrings(
|
|
const PatternParams& p,
|
|
const EList<string>& qs,
|
|
size_t nthreads = 1);
|
|
|
|
/**
|
|
* Return the number of reads attempted.
|
|
*/
|
|
TReadId readCnt() const { return readCnt_ - 1; }
|
|
|
|
int paired_type; // 1 - left or unpaird, 2-right
|
|
// int align_times = 0;
|
|
|
|
protected:
|
|
|
|
uint32_t seed_;
|
|
|
|
/// The number of reads read by this PatternSource
|
|
TReadId readCnt_;
|
|
|
|
int numWrappers_; /// # threads that own a wrapper for this PatternSource
|
|
bool doLocking_; /// override whether to lock (true = don't override)
|
|
/// User can ask to use the normal pthreads-style lock even if
|
|
/// spinlocks is enabled and compiled in. This is sometimes better
|
|
/// if we expect bad I/O latency on some reads.
|
|
bool useSpinlock_;
|
|
MUTEX_T mutex;
|
|
};
|
|
|
|
/**
|
|
* Abstract parent class for synhconized sources of paired-end reads
|
|
* (and possibly also single-end reads).
|
|
*/
|
|
class PairedPatternSource {
|
|
public:
|
|
PairedPatternSource(const PatternParams& p) : mutex_m(), seed_(p.seed) {}
|
|
virtual ~PairedPatternSource() { }
|
|
|
|
virtual void addWrapper() = 0;
|
|
virtual void reset() = 0;
|
|
|
|
virtual bool nextReadPair(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired,
|
|
bool fixName) = 0;
|
|
|
|
virtual pair<TReadId, TReadId> readCnt() const = 0;
|
|
|
|
/**
|
|
* Lock this PairedPatternSource, usually because one of its shared
|
|
* fields is being updated.
|
|
*/
|
|
void lock() {
|
|
mutex_m.lock();
|
|
}
|
|
|
|
/**
|
|
* Unlock this PairedPatternSource.
|
|
*/
|
|
void unlock() {
|
|
mutex_m.unlock();
|
|
}
|
|
|
|
/**
|
|
* Given the values for all of the various arguments used to specify
|
|
* the read and quality input, create a list of pattern sources to
|
|
* dispense them.
|
|
*/
|
|
static PairedPatternSource* setupPatternSources(
|
|
const EList<string>& si, // singles, from argv
|
|
const EList<string>& m1, // mate1's, from -1 arg
|
|
const EList<string>& m2, // mate2's, from -2 arg
|
|
const EList<string>& m12, // both mates on each line, from --12 arg
|
|
#ifdef USE_SRA
|
|
const EList<string>& sra_accs,
|
|
#endif
|
|
const EList<string>& q, // qualities associated with singles
|
|
const EList<string>& q1, // qualities associated with m1
|
|
const EList<string>& q2, // qualities associated with m2
|
|
const PatternParams& p, // read-in params
|
|
size_t nthreads,
|
|
bool verbose); // be talkative?
|
|
|
|
protected:
|
|
|
|
MUTEX_T mutex_m; /// mutex for syncing over critical regions
|
|
uint32_t seed_;
|
|
};
|
|
|
|
/**
|
|
* Encapsulates a synchronized source of both paired-end reads and
|
|
* unpaired reads, where the paired-end must come from parallel files.
|
|
*/
|
|
class PairedSoloPatternSource : public PairedPatternSource {
|
|
|
|
public:
|
|
|
|
PairedSoloPatternSource(
|
|
const EList<PatternSource*>* src,
|
|
const PatternParams& p) :
|
|
PairedPatternSource(p),
|
|
cur_(0),
|
|
src_(src)
|
|
{
|
|
assert(src_ != NULL);
|
|
for(size_t i = 0; i < src_->size(); i++) {
|
|
assert((*src_)[i] != NULL);
|
|
}
|
|
}
|
|
|
|
virtual ~PairedSoloPatternSource() { delete src_; }
|
|
|
|
/**
|
|
* Call this whenever this PairedPatternSource is wrapped by a new
|
|
* WrappedPatternSourcePerThread. This helps us keep track of
|
|
* whether locks within PatternSources will be contended.
|
|
*/
|
|
virtual void addWrapper() {
|
|
for(size_t i = 0; i < src_->size(); i++) {
|
|
(*src_)[i]->addWrapper();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reset this object and all the PatternSources under it so that
|
|
* the next call to nextReadPair gets the very first read pair.
|
|
*/
|
|
virtual void reset() {
|
|
for(size_t i = 0; i < src_->size(); i++) {
|
|
(*src_)[i]->reset();
|
|
}
|
|
cur_ = 0;
|
|
}
|
|
|
|
/**
|
|
* The main member function for dispensing pairs of reads or
|
|
* singleton reads. Returns true iff ra and rb contain a new
|
|
* pair; returns false if ra contains a new unpaired read.
|
|
*/
|
|
virtual bool nextReadPair(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired,
|
|
bool fixName);
|
|
|
|
/**
|
|
* Return the number of reads attempted.
|
|
*/
|
|
virtual pair<TReadId, TReadId> readCnt() const {
|
|
uint64_t ret = 0llu;
|
|
for(size_t i = 0; i < src_->size(); i++) ret += (*src_)[i]->readCnt();
|
|
return make_pair(ret, 0llu);
|
|
}
|
|
|
|
protected:
|
|
|
|
volatile uint32_t cur_; // current element in parallel srca_, srcb_ vectors
|
|
const EList<PatternSource*>* src_; /// PatternSources for paired-end reads
|
|
};
|
|
|
|
/**
|
|
* Encapsulates a synchronized source of both paired-end reads and
|
|
* unpaired reads, where the paired-end must come from parallel files.
|
|
*/
|
|
class PairedDualPatternSource : public PairedPatternSource {
|
|
|
|
public:
|
|
|
|
PairedDualPatternSource(
|
|
const EList<PatternSource*>* srca,
|
|
const EList<PatternSource*>* srcb,
|
|
const PatternParams& p) :
|
|
PairedPatternSource(p), cur_(0), srca_(srca), srcb_(srcb)
|
|
{
|
|
assert(srca_ != NULL);
|
|
assert(srcb_ != NULL);
|
|
// srca_ and srcb_ must be parallel
|
|
assert_eq(srca_->size(), srcb_->size());
|
|
for(size_t i = 0; i < srca_->size(); i++) {
|
|
// Can't have NULL first-mate sources. Second-mate sources
|
|
// can be NULL, in the case when the corresponding first-
|
|
// mate source is unpaired.
|
|
assert((*srca_)[i] != NULL);
|
|
for(size_t j = 0; j < srcb_->size(); j++) {
|
|
assert_neq((*srca_)[i], (*srcb_)[j]);
|
|
}
|
|
}
|
|
}
|
|
|
|
virtual ~PairedDualPatternSource() {
|
|
delete srca_;
|
|
delete srcb_;
|
|
}
|
|
|
|
/**
|
|
* Call this whenever this PairedPatternSource is wrapped by a new
|
|
* WrappedPatternSourcePerThread. This helps us keep track of
|
|
* whether locks within PatternSources will be contended.
|
|
*/
|
|
virtual void addWrapper() {
|
|
for(size_t i = 0; i < srca_->size(); i++) {
|
|
(*srca_)[i]->addWrapper();
|
|
if((*srcb_)[i] != NULL) {
|
|
(*srcb_)[i]->addWrapper();
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reset this object and all the PatternSources under it so that
|
|
* the next call to nextReadPair gets the very first read pair.
|
|
*/
|
|
virtual void reset() {
|
|
for(size_t i = 0; i < srca_->size(); i++) {
|
|
(*srca_)[i]->reset();
|
|
if((*srcb_)[i] != NULL) {
|
|
(*srcb_)[i]->reset();
|
|
}
|
|
}
|
|
cur_ = 0;
|
|
}
|
|
|
|
/**
|
|
* The main member function for dispensing pairs of reads or
|
|
* singleton reads. Returns true iff ra and rb contain a new
|
|
* pair; returns false if ra contains a new unpaired read.
|
|
*/
|
|
virtual bool nextReadPair(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired,
|
|
bool fixName);
|
|
|
|
/**
|
|
* Return the number of reads attempted.
|
|
*/
|
|
virtual pair<TReadId, TReadId> readCnt() const;
|
|
|
|
protected:
|
|
|
|
volatile uint32_t cur_; // current element in parallel srca_, srcb_ vectors
|
|
const EList<PatternSource*>* srca_; /// PatternSources for 1st mates and/or unpaired reads
|
|
const EList<PatternSource*>* srcb_; /// PatternSources for 2nd mates
|
|
};
|
|
|
|
/**
|
|
* Encapsulates a single thread's interaction with the PatternSource.
|
|
* Most notably, this class holds the buffers into which the
|
|
* PatterSource will write sequences. This class is *not* threadsafe
|
|
* - it doesn't need to be since there's one per thread. PatternSource
|
|
* is thread-safe.
|
|
*/
|
|
class PatternSourcePerThread {
|
|
|
|
public:
|
|
|
|
PatternSourcePerThread() :
|
|
buf1_(), buf2_(), rdid_(0xffffffff), endid_(0xffffffff) { }
|
|
|
|
virtual ~PatternSourcePerThread() { }
|
|
|
|
/**
|
|
* change 3N plan for both mate1 and mate2
|
|
*/
|
|
void changePlan3N(int mappingCycle) {
|
|
buf1_.changePlan3N(mappingCycle);
|
|
buf2_.changePlan3N(3-mappingCycle);
|
|
}
|
|
|
|
/**
|
|
* Read the next read pair.
|
|
*/
|
|
virtual bool nextReadPair(
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired,
|
|
bool fixName)
|
|
{
|
|
return success;
|
|
}
|
|
|
|
Read& bufa() { return buf1_; }
|
|
Read& bufb() { return buf2_; }
|
|
const Read& bufa() const { return buf1_; }
|
|
const Read& bufb() const { return buf2_; }
|
|
|
|
TReadId rdid() const { return rdid_; }
|
|
TReadId endid() const { return endid_; }
|
|
virtual void reset() { rdid_ = endid_ = 0xffffffff; }
|
|
|
|
/**
|
|
* Return the length of mate 1 or mate 2.
|
|
*/
|
|
size_t length(int mate) const {
|
|
return (mate == 1) ? buf1_.length() : buf2_.length();
|
|
}
|
|
|
|
protected:
|
|
|
|
Read buf1_; // read buffer for mate a
|
|
Read buf2_; // read buffer for mate b
|
|
TReadId rdid_; // index of read just read
|
|
TReadId endid_; // index of read just read
|
|
};
|
|
|
|
/**
|
|
* Abstract parent factory for PatternSourcePerThreads.
|
|
*/
|
|
class PatternSourcePerThreadFactory {
|
|
public:
|
|
virtual ~PatternSourcePerThreadFactory() { }
|
|
virtual PatternSourcePerThread* create() const = 0;
|
|
virtual EList<PatternSourcePerThread*>* create(uint32_t n) const = 0;
|
|
|
|
/// Free memory associated with a pattern source
|
|
virtual void destroy(PatternSourcePerThread* patsrc) const {
|
|
assert(patsrc != NULL);
|
|
// Free the PatternSourcePerThread
|
|
delete patsrc;
|
|
}
|
|
|
|
/// Free memory associated with a pattern source list
|
|
virtual void destroy(EList<PatternSourcePerThread*>* patsrcs) const {
|
|
assert(patsrcs != NULL);
|
|
// Free all of the PatternSourcePerThreads
|
|
for(size_t i = 0; i < patsrcs->size(); i++) {
|
|
if((*patsrcs)[i] != NULL) {
|
|
delete (*patsrcs)[i];
|
|
(*patsrcs)[i] = NULL;
|
|
}
|
|
}
|
|
// Free the vector
|
|
delete patsrcs;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* A per-thread wrapper for a PairedPatternSource.
|
|
*/
|
|
class WrappedPatternSourcePerThread : public PatternSourcePerThread {
|
|
public:
|
|
WrappedPatternSourcePerThread(PairedPatternSource& __patsrc) :
|
|
patsrc_(__patsrc)
|
|
{
|
|
patsrc_.addWrapper();
|
|
}
|
|
|
|
/**
|
|
* Get the next paired or unpaired read from the wrapped
|
|
* PairedPatternSource.
|
|
*/
|
|
virtual bool nextReadPair(
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired,
|
|
bool fixName);
|
|
|
|
private:
|
|
|
|
/// Container for obtaining paired reads from PatternSources
|
|
PairedPatternSource& patsrc_;
|
|
};
|
|
|
|
/**
|
|
* Abstract parent factory for PatternSourcePerThreads.
|
|
*/
|
|
class WrappedPatternSourcePerThreadFactory : public PatternSourcePerThreadFactory {
|
|
public:
|
|
WrappedPatternSourcePerThreadFactory(PairedPatternSource& patsrc) :
|
|
patsrc_(patsrc) { }
|
|
|
|
/**
|
|
* Create a new heap-allocated WrappedPatternSourcePerThreads.
|
|
*/
|
|
virtual PatternSourcePerThread* create() const {
|
|
return new WrappedPatternSourcePerThread(patsrc_);
|
|
}
|
|
|
|
/**
|
|
* Create a new heap-allocated vector of heap-allocated
|
|
* WrappedPatternSourcePerThreads.
|
|
*/
|
|
virtual EList<PatternSourcePerThread*>* create(uint32_t n) const {
|
|
EList<PatternSourcePerThread*>* v = new EList<PatternSourcePerThread*>;
|
|
for(size_t i = 0; i < n; i++) {
|
|
v->push_back(new WrappedPatternSourcePerThread(patsrc_));
|
|
assert(v->back() != NULL);
|
|
}
|
|
return v;
|
|
}
|
|
|
|
private:
|
|
/// Container for obtaining paired reads from PatternSources
|
|
PairedPatternSource& patsrc_;
|
|
};
|
|
|
|
/// Skip to the end of the current string of newline chars and return
|
|
/// the first character after the newline chars, or -1 for EOF
|
|
static inline int getOverNewline(FileBuf& in) {
|
|
int c;
|
|
while(isspace(c = in.get()));
|
|
return c;
|
|
}
|
|
|
|
/// Skip to the end of the current string of newline chars such that
|
|
/// the next call to get() returns the first character after the
|
|
/// whitespace
|
|
static inline int peekOverNewline(FileBuf& in) {
|
|
while(true) {
|
|
int c = in.peek();
|
|
if(c != '\r' && c != '\n') {
|
|
return c;
|
|
}
|
|
in.get();
|
|
}
|
|
}
|
|
|
|
/// Skip to the end of the current line; return the first character
|
|
/// of the next line or -1 for EOF
|
|
static inline int getToEndOfLine(FileBuf& in) {
|
|
while(true) {
|
|
int c = in.get(); if(c < 0) return -1;
|
|
if(c == '\n' || c == '\r') {
|
|
while(c == '\n' || c == '\r') {
|
|
c = in.get(); if(c < 0) return -1;
|
|
}
|
|
// c now holds first character of next line
|
|
return c;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Skip to the end of the current line such that the next call to
|
|
/// get() returns the first character on the next line
|
|
static inline int peekToEndOfLine(FileBuf& in) {
|
|
while(true) {
|
|
int c = in.get(); if(c < 0) return c;
|
|
if(c == '\n' || c == '\r') {
|
|
c = in.peek();
|
|
while(c == '\n' || c == '\r') {
|
|
in.get(); if(c < 0) return c; // consume \r or \n
|
|
c = in.peek();
|
|
}
|
|
// next get() gets first character of next line
|
|
return c;
|
|
}
|
|
}
|
|
}
|
|
|
|
extern void wrongQualityFormat(const BTString& read_name);
|
|
extern void tooFewQualities(const BTString& read_name);
|
|
extern void tooManyQualities(const BTString& read_name);
|
|
|
|
/**
|
|
* Encapsulates a source of patterns which is an in-memory vector.
|
|
*/
|
|
class VectorPatternSource : public PatternSource {
|
|
|
|
public:
|
|
|
|
VectorPatternSource(
|
|
const EList<string>& v,
|
|
const PatternParams& p);
|
|
|
|
virtual ~VectorPatternSource() { }
|
|
|
|
virtual bool nextReadImpl(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done);
|
|
|
|
/**
|
|
* This is unused, but implementation is given for completeness.
|
|
*/
|
|
virtual bool nextReadPairImpl(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired);
|
|
|
|
virtual void reset() {
|
|
PatternSource::reset();
|
|
cur_ = skip_;
|
|
paired_ = false;
|
|
}
|
|
|
|
private:
|
|
|
|
size_t cur_;
|
|
uint32_t skip_;
|
|
bool paired_;
|
|
EList<BTDnaString> v_; // forward sequences
|
|
EList<BTString> quals_; // forward qualities
|
|
EList<BTString> names_; // names
|
|
EList<int> trimmed3_; // names
|
|
EList<int> trimmed5_; // names
|
|
};
|
|
|
|
/**
|
|
*
|
|
*/
|
|
class BufferedFilePatternSource : public PatternSource {
|
|
public:
|
|
BufferedFilePatternSource(
|
|
const EList<string>& infiles,
|
|
const PatternParams& p) :
|
|
PatternSource(p),
|
|
infiles_(infiles),
|
|
filecur_(0),
|
|
fb_(),
|
|
skip_(p.skip),
|
|
first_(true)
|
|
{
|
|
assert_gt(infiles.size(), 0);
|
|
errs_.resize(infiles_.size());
|
|
errs_.fill(0, infiles_.size(), false);
|
|
assert(!fb_.isOpen());
|
|
open(); // open first file in the list
|
|
filecur_++;
|
|
}
|
|
|
|
virtual ~BufferedFilePatternSource() {
|
|
if(fb_.isOpen()) fb_.close();
|
|
}
|
|
|
|
/**
|
|
* Fill Read with the sequence, quality and name for the next
|
|
* read in the list of read files. This function gets called by
|
|
* all the search threads, so we must handle synchronization.
|
|
*/
|
|
virtual bool nextReadImpl(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done)
|
|
{
|
|
// We'll be manipulating our file handle/filecur_ state
|
|
lock();
|
|
while(true) {
|
|
do { read(r, rdid, endid, success, done); }
|
|
while(!success && !done);
|
|
if(!success && filecur_ < infiles_.size()) {
|
|
assert(done);
|
|
open();
|
|
resetForNextFile(); // reset state to handle a fresh file
|
|
filecur_++;
|
|
continue;
|
|
}
|
|
break;
|
|
}
|
|
assert(r.repOk());
|
|
// Leaving critical region
|
|
unlock();
|
|
return success;
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
virtual bool nextReadPairImpl(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired)
|
|
{
|
|
// We'll be manipulating our file handle/filecur_ state
|
|
lock();
|
|
while(true) {
|
|
do { readPair(ra, rb, rdid, endid, success, done, paired); }
|
|
while(!success && !done);
|
|
if(!success && filecur_ < infiles_.size()) {
|
|
assert(done);
|
|
open();
|
|
resetForNextFile(); // reset state to handle a fresh file
|
|
filecur_++;
|
|
continue;
|
|
}
|
|
break;
|
|
}
|
|
assert(ra.repOk());
|
|
assert(rb.repOk());
|
|
// Leaving critical region
|
|
unlock();
|
|
return success;
|
|
}
|
|
|
|
/**
|
|
* Reset state so that we read start reading again from the
|
|
* beginning of the first file. Should only be called by the
|
|
* master thread.
|
|
*/
|
|
virtual void reset() {
|
|
PatternSource::reset();
|
|
filecur_ = 0;
|
|
open();
|
|
filecur_++;
|
|
}
|
|
|
|
protected:
|
|
|
|
/// Read another pattern from the input file; this is overridden
|
|
/// to deal with specific file formats
|
|
virtual bool read(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done) = 0;
|
|
|
|
/// Read another pattern pair from the input file; this is
|
|
/// overridden to deal with specific file formats
|
|
virtual bool readPair(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired) = 0;
|
|
|
|
/// Reset state to handle a fresh file
|
|
virtual void resetForNextFile() { }
|
|
|
|
void open() {
|
|
if(fb_.isOpen()) fb_.close();
|
|
while(filecur_ < infiles_.size()) {
|
|
// Open read
|
|
FILE *in;
|
|
if(infiles_[filecur_] == "-") {
|
|
in = stdin;
|
|
} else if((in = fopen(infiles_[filecur_].c_str(), "rb")) == NULL) {
|
|
if(!errs_[filecur_]) {
|
|
cerr << "Warning: Could not open read file \"" << infiles_[filecur_].c_str() << "\" for reading; skipping..." << endl;
|
|
errs_[filecur_] = true;
|
|
}
|
|
filecur_++;
|
|
continue;
|
|
}
|
|
fb_.newFile(in);
|
|
return;
|
|
}
|
|
cerr << "Error: No input read files were valid" << endl;
|
|
exit(1);
|
|
return;
|
|
}
|
|
|
|
EList<string> infiles_; // filenames for read files
|
|
EList<bool> errs_; // whether we've already printed an error for each file
|
|
size_t filecur_; // index into infiles_ of next file to read
|
|
FileBuf fb_; // read file currently being read from
|
|
TReadId skip_; // number of reads to skip
|
|
bool first_;
|
|
};
|
|
|
|
/**
|
|
* Parse a single quality string from fb and store qualities in r.
|
|
* Assume the next character obtained via fb.get() is the first
|
|
* character of the quality string. When returning, the next
|
|
* character returned by fb.peek() or fb.get() should be the first
|
|
* character of the following line.
|
|
*/
|
|
int parseQuals(
|
|
Read& r,
|
|
FileBuf& fb,
|
|
int firstc,
|
|
int readLen,
|
|
int trim3,
|
|
int trim5,
|
|
bool intQuals,
|
|
bool phred64,
|
|
bool solexa64);
|
|
|
|
/**
|
|
* Synchronized concrete pattern source for a list of FASTA or CSFASTA
|
|
* (if color = true) files.
|
|
*/
|
|
class FastaPatternSource : public BufferedFilePatternSource {
|
|
public:
|
|
FastaPatternSource(const EList<string>& infiles,
|
|
const PatternParams& p) :
|
|
BufferedFilePatternSource(infiles, p),
|
|
first_(true), solexa64_(p.solexa64), phred64_(p.phred64), intQuals_(p.intQuals)
|
|
{ }
|
|
virtual void reset() {
|
|
first_ = true;
|
|
BufferedFilePatternSource::reset();
|
|
}
|
|
protected:
|
|
/**
|
|
* Scan to the next FASTA record (starting with >) and return the first
|
|
* character of the record (which will always be >).
|
|
*/
|
|
static int skipToNextFastaRecord(FileBuf& in) {
|
|
int c;
|
|
while((c = in.get()) != '>') {
|
|
if(in.eof()) return -1;
|
|
}
|
|
return c;
|
|
}
|
|
|
|
/// Called when we have to bail without having parsed a read.
|
|
void bail(Read& r) {
|
|
r.reset();
|
|
fb_.resetLastN();
|
|
}
|
|
|
|
/// Read another pattern from a FASTA input file
|
|
virtual bool read(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done);
|
|
|
|
/// Read another pair of patterns from a FASTA input file
|
|
virtual bool readPair(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired)
|
|
{
|
|
// (For now, we shouldn't ever be here)
|
|
cerr << "In FastaPatternSource.readPair()" << endl;
|
|
throw 1;
|
|
return false;
|
|
}
|
|
|
|
virtual void resetForNextFile() {
|
|
first_ = true;
|
|
}
|
|
|
|
private:
|
|
bool first_;
|
|
|
|
public:
|
|
bool solexa64_;
|
|
bool phred64_;
|
|
bool intQuals_;
|
|
};
|
|
|
|
|
|
/**
|
|
* Tokenize a line of space-separated integer quality values.
|
|
*/
|
|
static inline bool tokenizeQualLine(
|
|
FileBuf& filebuf,
|
|
char *buf,
|
|
size_t buflen,
|
|
EList<string>& toks)
|
|
{
|
|
size_t rd = filebuf.gets(buf, buflen);
|
|
if(rd == 0) return false;
|
|
assert(NULL == strrchr(buf, '\n'));
|
|
tokenize(string(buf), " ", toks);
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Synchronized concrete pattern source for a list of files with tab-
|
|
* delimited name, seq, qual fields (or, for paired-end reads,
|
|
* basename, seq1, qual1, seq2, qual2).
|
|
*/
|
|
class TabbedPatternSource : public BufferedFilePatternSource {
|
|
|
|
public:
|
|
|
|
TabbedPatternSource(
|
|
const EList<string>& infiles,
|
|
const PatternParams& p,
|
|
bool secondName) :
|
|
BufferedFilePatternSource(infiles, p),
|
|
solQuals_(p.solexa64),
|
|
phred64Quals_(p.phred64),
|
|
intQuals_(p.intQuals),
|
|
secondName_(secondName) { }
|
|
|
|
protected:
|
|
|
|
/// Read another pattern from a FASTA input file
|
|
virtual bool read(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done);
|
|
|
|
/// Read another pair of patterns from a FASTA input file
|
|
virtual bool readPair(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired);
|
|
|
|
private:
|
|
|
|
/**
|
|
* Parse a name from fb_ and store in r. Assume that the next
|
|
* character obtained via fb_.get() is the first character of
|
|
* the sequence and the string stops at the next char upto (could
|
|
* be tab, newline, etc.).
|
|
*/
|
|
int parseName(Read& r, Read* r2, char upto = '\t');
|
|
|
|
/**
|
|
* Parse a single sequence from fb_ and store in r. Assume
|
|
* that the next character obtained via fb_.get() is the first
|
|
* character of the sequence and the sequence stops at the next
|
|
* char upto (could be tab, newline, etc.).
|
|
*/
|
|
int parseSeq(Read& r, int& charsRead, int& trim5, char upto = '\t');
|
|
|
|
/**
|
|
* Parse a single quality string from fb_ and store in r.
|
|
* Assume that the next character obtained via fb_.get() is
|
|
* the first character of the quality string and the string stops
|
|
* at the next char upto (could be tab, newline, etc.).
|
|
*/
|
|
int parseQuals(Read& r, int charsRead, int dstLen, int trim5,
|
|
char& c2, char upto = '\t', char upto2 = -1);
|
|
|
|
bool solQuals_;
|
|
bool phred64Quals_;
|
|
bool intQuals_;
|
|
EList<string> qualToks_;
|
|
bool secondName_;
|
|
};
|
|
|
|
/**
|
|
* Synchronized concrete pattern source for Illumina Qseq files. In
|
|
* Qseq files, each read appears on a separate line and the tab-
|
|
* delimited fields are:
|
|
*
|
|
* 1. Machine name
|
|
* 2. Run number
|
|
* 3. Lane number
|
|
* 4. Tile number
|
|
* 5. X coordinate of spot
|
|
* 6. Y coordinate of spot
|
|
* 7. Index: "Index sequence or 0. For no indexing, or for a file that
|
|
* has not been demultiplexed yet, this field should have a value of
|
|
* 0."
|
|
* 8. Read number: 1 for unpaired, 1 or 2 for paired
|
|
* 9. Sequence
|
|
* 10. Quality
|
|
* 11. Filter: 1 = passed, 0 = didn't
|
|
*/
|
|
class QseqPatternSource : public BufferedFilePatternSource {
|
|
|
|
public:
|
|
|
|
QseqPatternSource(
|
|
const EList<string>& infiles,
|
|
const PatternParams& p) :
|
|
BufferedFilePatternSource(infiles, p),
|
|
solQuals_(p.solexa64),
|
|
phred64Quals_(p.phred64),
|
|
intQuals_(p.intQuals) { }
|
|
|
|
protected:
|
|
|
|
#define BAIL_UNPAIRED() { \
|
|
peekOverNewline(fb_); \
|
|
r.reset(); \
|
|
success = false; \
|
|
done = true; \
|
|
return success; \
|
|
}
|
|
|
|
/**
|
|
* Parse a name from fb_ and store in r. Assume that the next
|
|
* character obtained via fb_.get() is the first character of
|
|
* the sequence and the string stops at the next char upto (could
|
|
* be tab, newline, etc.).
|
|
*/
|
|
int parseName(
|
|
Read& r, // buffer for mate 1
|
|
Read* r2, // buffer for mate 2 (NULL if mate2 is read separately)
|
|
bool append, // true -> append characters, false -> skip them
|
|
bool clearFirst, // clear the name buffer first
|
|
bool warnEmpty, // emit a warning if nothing was added to the name
|
|
bool useDefault, // if nothing is read, put readCnt_ as a default value
|
|
int upto); // stop parsing when we first reach character 'upto'
|
|
|
|
/**
|
|
* Parse a single sequence from fb_ and store in r. Assume
|
|
* that the next character obtained via fb_.get() is the first
|
|
* character of the sequence and the sequence stops at the next
|
|
* char upto (could be tab, newline, etc.).
|
|
*/
|
|
int parseSeq(
|
|
Read& r, // buffer for read
|
|
int& charsRead,
|
|
int& trim5,
|
|
char upto);
|
|
|
|
/**
|
|
* Parse a single quality string from fb_ and store in r.
|
|
* Assume that the next character obtained via fb_.get() is
|
|
* the first character of the quality string and the string stops
|
|
* at the next char upto (could be tab, newline, etc.).
|
|
*/
|
|
int parseQuals(
|
|
Read& r, // buffer for read
|
|
int charsRead,
|
|
int dstLen,
|
|
int trim5,
|
|
char& c2,
|
|
char upto,
|
|
char upto2);
|
|
|
|
/**
|
|
* Read another pattern from a Qseq input file.
|
|
*/
|
|
virtual bool read(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done);
|
|
|
|
/**
|
|
* Read a pair of patterns from 1 Qseq file. Note: this is never used.
|
|
*/
|
|
virtual bool readPair(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired)
|
|
{
|
|
// (For now, we shouldn't ever be here)
|
|
cerr << "In QseqPatternSource.readPair()" << endl;
|
|
throw 1;
|
|
return false;
|
|
}
|
|
|
|
bool solQuals_;
|
|
bool phred64Quals_;
|
|
bool intQuals_;
|
|
EList<string> qualToks_;
|
|
};
|
|
|
|
/**
|
|
* Synchronized concrete pattern source for a list of FASTA files where
|
|
* reads need to be extracted from long continuous sequences.
|
|
*/
|
|
class FastaContinuousPatternSource : public BufferedFilePatternSource {
|
|
public:
|
|
FastaContinuousPatternSource(const EList<string>& infiles, const PatternParams& p) :
|
|
BufferedFilePatternSource(infiles, p),
|
|
length_(p.sampleLen), freq_(p.sampleFreq),
|
|
eat_(length_-1), beginning_(true),
|
|
bufCur_(0), subReadCnt_(0llu)
|
|
{
|
|
resetForNextFile();
|
|
}
|
|
|
|
virtual void reset() {
|
|
BufferedFilePatternSource::reset();
|
|
resetForNextFile();
|
|
}
|
|
|
|
protected:
|
|
|
|
/// Read another pattern from a FASTA input file
|
|
virtual bool read(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done)
|
|
{
|
|
success = true;
|
|
done = false;
|
|
r.reset();
|
|
while(true) {
|
|
r.color = gColor;
|
|
int c = fb_.get();
|
|
if(c < 0) { success = false; done = true; return success; }
|
|
if(c == '>') {
|
|
resetForNextFile();
|
|
c = fb_.peek();
|
|
bool sawSpace = false;
|
|
while(c != '\n' && c != '\r') {
|
|
if(!sawSpace) {
|
|
sawSpace = isspace(c);
|
|
}
|
|
if(!sawSpace) {
|
|
nameBuf_.append(c);
|
|
}
|
|
fb_.get();
|
|
c = fb_.peek();
|
|
}
|
|
while(c == '\n' || c == '\r') {
|
|
fb_.get();
|
|
c = fb_.peek();
|
|
}
|
|
nameBuf_.append('_');
|
|
} else {
|
|
int cat = asc2dnacat[c];
|
|
if(cat >= 2) c = 'N';
|
|
if(cat == 0) {
|
|
// Encountered non-DNA, non-IUPAC char; skip it
|
|
continue;
|
|
} else {
|
|
// DNA char
|
|
buf_[bufCur_++] = c;
|
|
if(bufCur_ == 1024) bufCur_ = 0;
|
|
if(eat_ > 0) {
|
|
eat_--;
|
|
// Try to keep readCnt_ aligned with the offset
|
|
// into the reference; that lets us see where
|
|
// the sampling gaps are by looking at the read
|
|
// name
|
|
if(!beginning_) readCnt_++;
|
|
continue;
|
|
}
|
|
for(size_t i = 0; i < length_; i++) {
|
|
if(length_ - i <= bufCur_) {
|
|
c = buf_[bufCur_ - (length_ - i)];
|
|
} else {
|
|
// Rotate
|
|
c = buf_[bufCur_ - (length_ - i) + 1024];
|
|
}
|
|
r.patFw.append(asc2dna[c]);
|
|
r.qual.append('I');
|
|
}
|
|
// Set up a default name if one hasn't been set
|
|
r.name = nameBuf_;
|
|
char cbuf[20];
|
|
itoa10<TReadId>(readCnt_ - subReadCnt_, cbuf);
|
|
r.name.append(cbuf);
|
|
eat_ = freq_-1;
|
|
readCnt_++;
|
|
beginning_ = false;
|
|
rdid = endid = readCnt_-1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/// Shouldn't ever be here; it's not sensible to obtain read pairs
|
|
// from a continuous input.
|
|
virtual bool readPair(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired)
|
|
{
|
|
cerr << "In FastaContinuousPatternSource.readPair()" << endl;
|
|
throw 1;
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Reset state to be read for the next file.
|
|
*/
|
|
virtual void resetForNextFile() {
|
|
eat_ = length_-1;
|
|
beginning_ = true;
|
|
bufCur_ = 0;
|
|
nameBuf_.clear();
|
|
subReadCnt_ = readCnt_;
|
|
}
|
|
|
|
private:
|
|
size_t length_; /// length of reads to generate
|
|
size_t freq_; /// frequency to sample reads
|
|
size_t eat_; /// number of characters we need to skip before
|
|
/// we have flushed all of the ambiguous or
|
|
/// non-existent characters out of our read
|
|
/// window
|
|
bool beginning_; /// skipping over the first read length?
|
|
char buf_[1024]; /// read buffer
|
|
BTString nameBuf_; /// read buffer for name of fasta record being
|
|
/// split into mers
|
|
size_t bufCur_; /// buffer cursor; points to where we should
|
|
/// insert the next character
|
|
uint64_t subReadCnt_;/// number to subtract from readCnt_ to get
|
|
/// the pat id to output (so it resets to 0 for
|
|
/// each new sequence)
|
|
};
|
|
|
|
/**
|
|
* Read a FASTQ-format file.
|
|
* See: http://maq.sourceforge.net/fastq.shtml
|
|
*/
|
|
class FastqPatternSource : public BufferedFilePatternSource {
|
|
|
|
public:
|
|
|
|
FastqPatternSource(const EList<string>& infiles, const PatternParams& p) :
|
|
BufferedFilePatternSource(infiles, p),
|
|
first_(true),
|
|
solQuals_(p.solexa64),
|
|
phred64Quals_(p.phred64),
|
|
intQuals_(p.intQuals),
|
|
fuzzy_(p.fuzzy)
|
|
{ }
|
|
|
|
virtual void reset() {
|
|
first_ = true;
|
|
fb_.resetLastN();
|
|
BufferedFilePatternSource::reset();
|
|
}
|
|
|
|
protected:
|
|
|
|
/**
|
|
* Scan to the next FASTQ record (starting with @) and return the first
|
|
* character of the record (which will always be @). Since the quality
|
|
* line may start with @, we keep scanning until we've seen a line
|
|
* beginning with @ where the line two lines back began with +.
|
|
*/
|
|
static int skipToNextFastqRecord(FileBuf& in, bool sawPlus) {
|
|
int line = 0;
|
|
int plusLine = -1;
|
|
int c = in.get();
|
|
int firstc = c;
|
|
while(true) {
|
|
if(line > 20) {
|
|
// If we couldn't find our desired '@' in the first 20
|
|
// lines, it's time to give up
|
|
if(firstc == '>') {
|
|
// That firstc is '>' may be a hint that this is
|
|
// actually a FASTA file, so return it intact
|
|
return '>';
|
|
}
|
|
// Return an error
|
|
return -1;
|
|
}
|
|
if(c == -1) return -1;
|
|
if(c == '\n') {
|
|
c = in.get();
|
|
if(c == '@' && sawPlus && plusLine == (line-2)) {
|
|
return '@';
|
|
}
|
|
else if(c == '+') {
|
|
// Saw a '+' at the beginning of a line; remember where
|
|
// we saw it
|
|
sawPlus = true;
|
|
plusLine = line;
|
|
}
|
|
else if(c == -1) {
|
|
return -1;
|
|
}
|
|
line++;
|
|
}
|
|
c = in.get();
|
|
}
|
|
}
|
|
|
|
/// Read another pattern from a FASTQ input file
|
|
virtual bool read(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done);
|
|
|
|
/// Read another read pair from a FASTQ input file
|
|
virtual bool readPair(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired)
|
|
{
|
|
// (For now, we shouldn't ever be here)
|
|
cerr << "In FastqPatternSource.readPair()" << endl;
|
|
throw 1;
|
|
return false;
|
|
}
|
|
|
|
virtual void resetForNextFile() {
|
|
first_ = true;
|
|
}
|
|
|
|
private:
|
|
|
|
/**
|
|
* Do things we need to do if we have to bail in the middle of a
|
|
* read, usually because we reached the end of the input without
|
|
* finishing.
|
|
*/
|
|
void bail(Read& r) {
|
|
r.patFw.clear();
|
|
fb_.resetLastN();
|
|
}
|
|
|
|
bool first_;
|
|
bool solQuals_;
|
|
bool phred64Quals_;
|
|
bool intQuals_;
|
|
bool fuzzy_;
|
|
EList<string> qualToks_;
|
|
};
|
|
|
|
/**
|
|
* Read a Raw-format file (one sequence per line). No quality strings
|
|
* allowed. All qualities are assumed to be 'I' (40 on the Phred-33
|
|
* scale).
|
|
*/
|
|
class RawPatternSource : public BufferedFilePatternSource {
|
|
|
|
public:
|
|
|
|
RawPatternSource(const EList<string>& infiles, const PatternParams& p) :
|
|
BufferedFilePatternSource(infiles, p), first_(true) { }
|
|
|
|
virtual void reset() {
|
|
first_ = true;
|
|
BufferedFilePatternSource::reset();
|
|
}
|
|
|
|
protected:
|
|
|
|
/// Read another pattern from a Raw input file
|
|
virtual bool read(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done)
|
|
{
|
|
int c;
|
|
success = true;
|
|
done = false;
|
|
r.reset();
|
|
c = getOverNewline(this->fb_);
|
|
if(c < 0) {
|
|
bail(r); success = false; done = true; return success;
|
|
}
|
|
assert(!isspace(c));
|
|
r.color = gColor;
|
|
int mytrim5 = gTrim5;
|
|
if(first_) {
|
|
// Check that the first character is sane for a raw file
|
|
int cc = c;
|
|
if(gColor) {
|
|
if(cc >= '0' && cc <= '4') cc = "ACGTN"[(int)cc - '0'];
|
|
if(cc == '.') cc = 'N';
|
|
}
|
|
if(asc2dnacat[cc] == 0) {
|
|
cerr << "Error: reads file does not look like a Raw file" << endl;
|
|
if(c == '>') {
|
|
cerr << "Reads file looks like a FASTA file; please use -f" << endl;
|
|
}
|
|
if(c == '@') {
|
|
cerr << "Reads file looks like a FASTQ file; please use -q" << endl;
|
|
}
|
|
throw 1;
|
|
}
|
|
first_ = false;
|
|
}
|
|
if(gColor) {
|
|
// This may be a primer character. If so, keep it in the
|
|
// 'primer' field of the read buf and parse the rest of the
|
|
// read without it.
|
|
c = toupper(c);
|
|
if(asc2dnacat[c] > 0) {
|
|
// First char is a DNA char
|
|
int c2 = toupper(fb_.peek());
|
|
// Second char is a color char
|
|
if(asc2colcat[c2] > 0) {
|
|
r.primer = c;
|
|
r.trimc = c2;
|
|
mytrim5 += 2; // trim primer and first color
|
|
}
|
|
}
|
|
if(c < 0) {
|
|
bail(r); success = false; done = true; return success;
|
|
}
|
|
}
|
|
// _in now points just past the first character of a sequence
|
|
// line, and c holds the first character
|
|
int chs = 0;
|
|
while(!isspace(c) && c >= 0) {
|
|
if(gColor) {
|
|
if(c >= '0' && c <= '4') c = "ACGTN"[(int)c - '0'];
|
|
if(c == '.') c = 'N';
|
|
}
|
|
// 5' trimming
|
|
if(isalpha(c) && chs >= mytrim5) {
|
|
//size_t len = chs - mytrim5;
|
|
//if(len >= 1024) tooManyQualities(BTString("(no name)"));
|
|
r.patFw.append(asc2dna[c]);
|
|
r.qual.append('I');
|
|
}
|
|
chs++;
|
|
if(isspace(fb_.peek())) break;
|
|
c = fb_.get();
|
|
}
|
|
// 3' trimming
|
|
r.patFw.trimEnd(gTrim3);
|
|
r.qual.trimEnd(gTrim3);
|
|
c = peekToEndOfLine(fb_);
|
|
r.trimmed3 = gTrim3;
|
|
r.trimmed5 = mytrim5;
|
|
r.readOrigBuf.install(fb_.lastN(), fb_.lastNLen());
|
|
fb_.resetLastN();
|
|
|
|
// Set up name
|
|
char cbuf[20];
|
|
itoa10<TReadId>(readCnt_, cbuf);
|
|
r.name.install(cbuf);
|
|
readCnt_++;
|
|
|
|
rdid = endid = readCnt_-1;
|
|
return success;
|
|
}
|
|
|
|
/// Read another read pair from a FASTQ input file
|
|
virtual bool readPair(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired)
|
|
{
|
|
// (For now, we shouldn't ever be here)
|
|
cerr << "In RawPatternSource.readPair()" << endl;
|
|
throw 1;
|
|
return false;
|
|
}
|
|
|
|
virtual void resetForNextFile() {
|
|
first_ = true;
|
|
}
|
|
|
|
private:
|
|
|
|
/**
|
|
* Do things we need to do if we have to bail in the middle of a
|
|
* read, usually because we reached the end of the input without
|
|
* finishing.
|
|
*/
|
|
void bail(Read& r) {
|
|
r.patFw.clear();
|
|
fb_.resetLastN();
|
|
}
|
|
|
|
bool first_;
|
|
};
|
|
|
|
#ifdef USE_SRA
|
|
|
|
namespace ngs {
|
|
class ReadCollection;
|
|
class ReadIterator;
|
|
}
|
|
|
|
namespace tthread {
|
|
class thread;
|
|
};
|
|
|
|
struct SRA_Data;
|
|
|
|
/**
|
|
*
|
|
*/
|
|
class SRAPatternSource : public PatternSource {
|
|
public:
|
|
SRAPatternSource(
|
|
const EList<string>& sra_accs,
|
|
const PatternParams& p,
|
|
const size_t nthreads = 1) :
|
|
PatternSource(p),
|
|
sra_accs_(sra_accs),
|
|
sra_acc_cur_(0),
|
|
skip_(p.skip),
|
|
first_(true),
|
|
nthreads_(nthreads),
|
|
sra_run_(NULL),
|
|
sra_it_(NULL),
|
|
sra_data_(NULL),
|
|
io_thread_(NULL)
|
|
{
|
|
assert_gt(sra_accs_.size(), 0);
|
|
errs_.resize(sra_accs_.size());
|
|
errs_.fill(0, sra_accs_.size(), false);
|
|
open(); // open first file in the list
|
|
sra_acc_cur_++;
|
|
}
|
|
|
|
virtual ~SRAPatternSource();
|
|
|
|
/**
|
|
* Fill Read with the sequence, quality and name for the next
|
|
* read in the list of read files. This function gets called by
|
|
* all the search threads, so we must handle synchronization.
|
|
*/
|
|
virtual bool nextReadImpl(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done)
|
|
{
|
|
// We'll be manipulating our file handle/filecur_ state
|
|
lock();
|
|
while(true) {
|
|
do { read(r, rdid, endid, success, done); }
|
|
while(!success && !done);
|
|
if(!success && sra_acc_cur_ < sra_accs_.size()) {
|
|
assert(done);
|
|
open();
|
|
resetForNextFile(); // reset state to handle a fresh file
|
|
sra_acc_cur_++;
|
|
continue;
|
|
}
|
|
break;
|
|
}
|
|
assert(r.repOk());
|
|
// Leaving critical region
|
|
unlock();
|
|
return success;
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
virtual bool nextReadPairImpl(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired)
|
|
{
|
|
// We'll be manipulating our file handle/filecur_ state
|
|
lock();
|
|
while(true) {
|
|
do { readPair(ra, rb, rdid, endid, success, done, paired); }
|
|
while(!success && !done);
|
|
if(!success && sra_acc_cur_ < sra_accs_.size()) {
|
|
assert(done);
|
|
open();
|
|
resetForNextFile(); // reset state to handle a fresh file
|
|
sra_acc_cur_++;
|
|
continue;
|
|
}
|
|
break;
|
|
}
|
|
assert(ra.repOk());
|
|
assert(rb.repOk());
|
|
// Leaving critical region
|
|
unlock();
|
|
return success;
|
|
}
|
|
|
|
/**
|
|
* Reset state so that we read start reading again from the
|
|
* beginning of the first file. Should only be called by the
|
|
* master thread.
|
|
*/
|
|
virtual void reset() {
|
|
PatternSource::reset();
|
|
sra_acc_cur_ = 0,
|
|
open();
|
|
sra_acc_cur_++;
|
|
}
|
|
|
|
/// Read another pattern from the input file; this is overridden
|
|
/// to deal with specific file formats
|
|
virtual bool read(
|
|
Read& r,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
/// Read another pattern pair from the input file; this is
|
|
/// overridden to deal with specific file formats
|
|
virtual bool readPair(
|
|
Read& ra,
|
|
Read& rb,
|
|
TReadId& rdid,
|
|
TReadId& endid,
|
|
bool& success,
|
|
bool& done,
|
|
bool& paired);
|
|
|
|
protected:
|
|
|
|
/// Reset state to handle a fresh file
|
|
virtual void resetForNextFile() { }
|
|
|
|
void open();
|
|
|
|
EList<string> sra_accs_; // filenames for read files
|
|
EList<bool> errs_; // whether we've already printed an error for each file
|
|
size_t sra_acc_cur_; // index into infiles_ of next file to read
|
|
TReadId skip_; // number of reads to skip
|
|
bool first_;
|
|
|
|
size_t nthreads_;
|
|
|
|
ngs::ReadCollection* sra_run_;
|
|
ngs::ReadIterator* sra_it_;
|
|
|
|
SRA_Data* sra_data_;
|
|
tthread::thread* io_thread_;
|
|
};
|
|
|
|
#endif
|
|
|
|
#endif /*PAT_H_*/
|