package aligner; import java.util.Arrays; import dna.AminoAcid; import dna.ChromosomeArray; import dna.Data; import shared.Tools; import stream.Read; import stream.SiteScore; /** * Based on MSA9ts, with transform scores tweaked for PacBio. */ public final class MultiStateAligner9PacBioAdapter3 { public MultiStateAligner9PacBioAdapter3(int maxRows_, int maxColumns_){ // assert(maxColumns_>=200); // assert(maxRows_>=200); maxRows=maxRows_; maxColumns=maxColumns_; packed=new int[3][maxRows+1][]; for(int i=0; i<3; i++){ packed[i][0]=new int[maxColumns+1]; packed[i][1]=new int[maxColumns+1]; packed[i][2]=new int[maxColumns+1]; for(int j=3; j0); rows=read.length; columns=refEndLoc-refStartLoc+1; if(/*read.length<40 || */false || minScore<=0 || columns>read.length+Tools.min(100, read.length)){ // assert(false) : minScore; return fillUnlimited(read, ref, refStartLoc, refEndLoc); } minScore-=100; //Increases quality trivially assert(rows<=maxRows) : "Check that values are in-bounds before calling this function: "+rows+", "+maxRows+"\n"+ refStartLoc+", "+refEndLoc+", "+rows+", "+maxRows+", "+columns+", "+maxColumns+"\n"+new String(read)+"\n"; assert(columns<=maxColumns) : "Check that values are in-bounds before calling this function: "+columns+", "+maxColumns+"\n"+ refStartLoc+", "+refEndLoc+", "+rows+", "+maxRows+", "+columns+", "+maxColumns+"\n"+new String(read)+"\n"; assert(refStartLoc>=0) : "Check that values are in-bounds before calling this function: "+refStartLoc+"\n"+ refStartLoc+", "+refEndLoc+", "+rows+", "+maxRows+", "+columns+", "+maxColumns+"\n"+new String(read)+"\n"; assert(refEndLocBADoff); //TODO: Actually, it needs to be substantially more. assert(subfloor=0; i--){ vertLimit[i]=Tools.max(vertLimit[i+1]-POINTSoff_MATCH2, floor); } horizLimit[columns]=minScore_off; for(int i=columns-1; i>=0; i--){ horizLimit[i]=Tools.max(horizLimit[i+1]-POINTSoff_MATCH2, floor); } for(int row=1; row<=rows; row++){ { int score=calcInsScoreOffset(row); packed[0][row][0]=score; packed[1][row][0]=score; packed[2][row][0]=score; } int colStart=minGoodCol; int colStop=maxGoodCol; minGoodCol=-1; maxGoodCol=-2; final int vlimit=vertLimit[row]; if(verbose2){ System.out.println(); System.out.println("row="+row); System.out.println("colStart="+colStart); System.out.println("colStop="+colStop); System.out.println("vlimit="+(vlimit>>SCOREOFFSET)+"\t"+(vlimit)); } if(colStart<0 || colStop1){ assert(row>0); packed[MODE_MS][row][colStart-1]=subfloor; packed[MODE_INS][row][colStart-1]=subfloor; packed[MODE_DEL][row][colStart-1]=subfloor; } for(int col=colStart; col<=columns; col++){ if(verbose2){ System.out.println("\ncol "+col); } final byte call0=(row<2 ? (byte)'?' : read[row-2]); final byte call1=read[row-1]; final byte ref0=(col<2 ? (byte)'!' : ref[refStartLoc+col-2]); final byte ref1=ref[refStartLoc+col-1]; // final boolean match=(read[row-1]==ref[refStartLoc+col-1]); // final boolean prevMatch=(row<2 || col<2 ? false : read[row-2]==ref[refStartLoc+col-2]); final boolean match=(call1==ref1); final boolean prevMatch=(call0==ref0); // System.err.println("") iterationsLimited++; final int limit=Tools.max(vlimit, horizLimit[col]); final int limit3=Tools.max(floor, (match ? limit-POINTSoff_MATCH2 : limit-POINTSoff_SUB3)); final int delNeeded=Tools.max(0, row-col-1); final int insNeeded=Tools.max(0, (rows-row)-(columns-col)-1); final int delPenalty=calcDelScoreOffset(delNeeded); final int insPenalty=calcInsScoreOffset(insNeeded); final int scoreFromDiag_MS=packed[MODE_MS][row-1][col-1]&SCOREMASK; final int scoreFromDel_MS=packed[MODE_DEL][row-1][col-1]&SCOREMASK; final int scoreFromIns_MS=packed[MODE_INS][row-1][col-1]&SCOREMASK; final int scoreFromDiag_DEL=packed[MODE_MS][row][col-1]&SCOREMASK; final int scoreFromDel_DEL=packed[MODE_DEL][row][col-1]&SCOREMASK; final int scoreFromDiag_INS=packed[MODE_MS][row-1][col]&SCOREMASK; final int scoreFromIns_INS=packed[MODE_INS][row-1][col]&SCOREMASK; // if(scoreFromDiag_MS=scoreD && scoreMS>=scoreI){ score=scoreMS; time=(prevMatch ? streak+1 : 1); prevState=MODE_MS; }else if(scoreD>=scoreI){ score=scoreD; time=1; prevState=MODE_DEL; }else{ score=scoreI; time=1; prevState=MODE_INS; } }else{ int scoreMS; if(ref1!='N' && call1!='N'){ scoreMS=scoreFromDiag_MS+(prevMatch ? (streak<=1 ? POINTSoff_SUBR : POINTSoff_SUB) : subScoreArray[streak]); }else{ scoreMS=scoreFromDiag_MS+POINTSoff_NOCALL; } int scoreD=scoreFromDel_MS+POINTSoff_SUB; //+2 to move it as close as possible to the deletion / insertion int scoreI=scoreFromIns_MS+POINTSoff_SUB; if(scoreMS>=scoreD && scoreMS>=scoreI){ score=scoreMS; time=(prevMatch ? 1 : streak+1); // time=(prevMatch ? (streak==1 ? 3 : 1) : streak+1); prevState=MODE_MS; }else if(scoreD>=scoreI){ score=scoreD; time=1; prevState=MODE_DEL; }else{ score=scoreI; time=1; prevState=MODE_INS; } } final int limit2; if(delNeeded>0){ limit2=limit-delPenalty; }else if(insNeeded>0){ limit2=limit-insPenalty; }else{ limit2=limit; } assert(limit2>=limit); if(verbose2){System.err.println("MS: \tlimit2="+(limit2>>SCOREOFFSET)+"\t, score="+(score>>SCOREOFFSET));} if(score>=limit2){ maxGoodCol=col; if(minGoodCol<0){minGoodCol=col;} }else{ score=subfloor; } assert(time<=MAX_TIME);//if(time>MAX_TIME){time=MAX_TIME-3;} assert(score>=MINoff_SCORE || score==BADoff) : "Score overflow - use MSA2 instead"; assert(score<=MAXoff_SCORE) : "Score overflow - use MSA2 instead"; // packed[MODE_MS][row][col]=(score|prevState|time); packed[MODE_MS][row][col]=(score|time); assert((score&SCOREMASK)==score); // assert((prevState&MODEMASK)==prevState); assert((time&TIMEMASK)==time); } } if((scoreFromDiag_DEL<=limit && scoreFromDel_DEL<=limit)){ // assert((scoreFromDiag_DEL<=limit && scoreFromDel_DEL<=limit)) : scoreFromDiag_DEL+", "+row; packed[MODE_DEL][row][col]=subfloor; }else{//Calculate DEL score final int streak=packed[MODE_DEL][row][col-1]&TIMEMASK; int scoreMS=scoreFromDiag_DEL+POINTSoff_DEL; int scoreD=scoreFromDel_DEL+delScoreArray[streak]; // int scoreI=scoreFromIns+POINTSoff_DEL; if(ref1=='N'){ scoreMS+=POINTSoff_DEL_REF_N; scoreD+=POINTSoff_DEL_REF_N; } //if(match){scoreMS=subfloor;} int score; int time; byte prevState; if(scoreMS>=scoreD){ score=scoreMS; time=1; prevState=MODE_MS; }else{ score=scoreD; time=streak+1; prevState=MODE_DEL; } final int limit2; if(insNeeded>0){ limit2=limit-insPenalty; }else if(delNeeded>0){ limit2=limit-calcDelScoreOffset(time+delNeeded)+calcDelScoreOffset(time); }else{ limit2=limit; } assert(limit2>=limit); if(verbose2){System.err.println("DEL: \tlimit2="+(limit2>>SCOREOFFSET)+"\t, score="+(score>>SCOREOFFSET));} if(score>=limit2){ maxGoodCol=col; if(minGoodCol<0){minGoodCol=col;} }else{ score=subfloor; } assert(time<=MAX_TIME);//if(time>MAX_TIME){time=MAX_TIME-3;} assert(score>=MINoff_SCORE || score==BADoff) : "Score overflow - use MSA2 instead"; assert(score<=MAXoff_SCORE) : "Score overflow - use MSA2 instead"; // packed[MODE_DEL][row][col]=(score|prevState|time); packed[MODE_DEL][row][col]=(score|time); assert((score&SCOREMASK)==score); // assert((prevState&MODEMASK)==prevState); assert((time&TIMEMASK)==time); } if(scoreFromDiag_INS<=limit && scoreFromIns_INS<=limit){ packed[MODE_INS][row][col]=subfloor; }else{//Calculate INS score final int streak=packed[MODE_INS][row-1][col]&TIMEMASK; int scoreMS=scoreFromDiag_INS+POINTSoff_INS; // int scoreD=scoreFromDel+POINTSoff_INS; int scoreI=scoreFromIns_INS+insScoreArray[streak]; // System.err.println("("+row+","+col+")\t"+scoreFromDiag+"+"+POINTSoff_INS+"="+scoreM+", "+ // scoreFromSub+"+"+POINTSoff_INS+"="+scoreS+", " // +scoreD+", "+scoreFromIns+"+"+ // (streak==0 ? POINTSoff_INS : streak=colStop){ if(col>colStop && maxGoodCol1){ packed[MODE_MS][row-1][col+1]=subfloor; packed[MODE_INS][row-1][col+1]=subfloor; packed[MODE_DEL][row-1][col+1]=subfloor; } } } } int maxCol=-1; int maxState=-1; int maxScore=Integer.MIN_VALUE; for(int state=0; statemaxScore){ maxScore=x; maxCol=col; maxState=state; } } } assert(maxScore>=BADoff); // if(maxScore==BADoff){ // return null; // } // if(maxScore>=SCOREOFFSET; // System.err.println("Returning "+rows+", "+maxCol+", "+maxState+", "+maxScore); return new int[] {rows, maxCol, maxState, maxScore}; } /** return new int[] {rows, maxC, maxS, max}; * Does not require a min score (ie, same as old method) */ private final int[] fillUnlimited(byte[] read, byte[] ref, int refStartLoc, int refEndLoc){ rows=read.length; columns=refEndLoc-refStartLoc+1; final int maxGain=(read.length-1)*POINTSoff_MATCH2+POINTSoff_MATCH; final int subfloor=0-2*maxGain; assert(subfloor>BADoff && subfloor*2>BADoff); //TODO: Actually, it needs to be substantially more. //temporary, for finding a bug if(rows>maxRows || columns>maxColumns){ throw new RuntimeException("rows="+rows+", maxRows="+maxRows+", cols="+columns+", maxCols="+maxColumns+"\n"+new String(read)+"\n"); } assert(rows<=maxRows) : "Check that values are in-bounds before calling this function: "+rows+", "+maxRows; assert(columns<=maxColumns) : "Check that values are in-bounds before calling this function: "+columns+", "+maxColumns; assert(refStartLoc>=0) : "Check that values are in-bounds before calling this function: "+refStartLoc; assert(refEndLoc=scoreD && scoreMS>=scoreI){ score=scoreMS; time=(prevMatch ? streak+1 : 1); // prevState=MODE_MS; }else if(scoreD>=scoreI){ score=scoreD; time=1; // prevState=MODE_DEL; }else{ score=scoreI; time=1; // prevState=MODE_INS; } assert(time<=MAX_TIME);//if(time>MAX_TIME){time=MAX_TIME-3;} assert(score>=MINoff_SCORE) : "Score overflow - use MSA2 instead"; assert(score<=MAXoff_SCORE) : "Score overflow - use MSA2 instead"; // packed[MODE_MS][row][col]=(score|prevState|time); packed[MODE_MS][row][col]=(score|time); assert((score&SCOREMASK)==score); // assert((prevState&MODEMASK)==prevState); assert((time&TIMEMASK)==time); }else{ int scoreMS; if(ref1!='N' && call1!='N'){ scoreMS=scoreFromDiag+(prevMatch ? (streak<=1 ? POINTSoff_SUBR : POINTSoff_SUB) : subScoreArray[streak]); }else{ scoreMS=scoreFromDiag+POINTSoff_NOCALL; } int scoreD=scoreFromDel+POINTSoff_SUB; //+2 to move it as close as possible to the deletion / insertion int scoreI=scoreFromIns+POINTSoff_SUB; int score; int time; byte prevState; if(scoreMS>=scoreD && scoreMS>=scoreI){ score=scoreMS; time=(prevMatch ? 1 : streak+1); // time=(prevMatch ? (streak==1 ? 3 : 1) : streak+1); prevState=MODE_MS; }else if(scoreD>=scoreI){ score=scoreD; time=1; prevState=MODE_DEL; }else{ score=scoreI; time=1; prevState=MODE_INS; } assert(time<=MAX_TIME);//if(time>MAX_TIME){time=MAX_TIME-3;} assert(score>=MINoff_SCORE) : "Score overflow - use MSA2 instead"; assert(score<=MAXoff_SCORE) : "Score overflow - use MSA2 instead"; // packed[MODE_MS][row][col]=(score|prevState|time); packed[MODE_MS][row][col]=(score|time); assert((score&SCOREMASK)==score); // assert((prevState&MODEMASK)==prevState); assert((time&TIMEMASK)==time); } } } {//Calculate DEL score final int streak=packed[MODE_DEL][row][col-1]&TIMEMASK; final int scoreFromDiag=packed[MODE_MS][row][col-1]&SCOREMASK; final int scoreFromDel=packed[MODE_DEL][row][col-1]&SCOREMASK; int scoreMS=scoreFromDiag+POINTSoff_DEL; int scoreD=scoreFromDel+delScoreArray[streak]; // int scoreI=scoreFromIns+POINTSoff_DEL; if(ref1=='N'){ scoreMS+=POINTSoff_DEL_REF_N; scoreD+=POINTSoff_DEL_REF_N; } //if(match){scoreMS=subfloor;} int score; int time; byte prevState; if(scoreMS>=scoreD){ score=scoreMS; time=1; prevState=MODE_MS; }else{ score=scoreD; time=streak+1; prevState=MODE_DEL; } assert(time<=MAX_TIME);//if(time>MAX_TIME){time=MAX_TIME-3;} assert(score>=MINoff_SCORE) : "Score overflow - use MSA2 instead"; assert(score<=MAXoff_SCORE) : "Score overflow - use MSA2 instead"; // packed[MODE_DEL][row][col]=(score|prevState|time); packed[MODE_DEL][row][col]=(score|time); assert((score&SCOREMASK)==score); // assert((prevState&MODEMASK)==prevState); assert((time&TIMEMASK)==time); } {//Calculate INS score final int streak=packed[MODE_INS][row-1][col]&TIMEMASK; final int scoreFromDiag=packed[MODE_MS][row-1][col]&SCOREMASK; final int scoreFromIns=packed[MODE_INS][row-1][col]&SCOREMASK; int scoreMS=scoreFromDiag+POINTSoff_INS; // int scoreD=scoreFromDel+POINTSoff_INS; int scoreI=scoreFromIns+insScoreArray[streak]; // System.err.println("("+row+","+col+")\t"+scoreFromDiag+"+"+POINTSoff_INS+"="+scoreM+", "+ // scoreFromSub+"+"+POINTSoff_INS+"="+scoreS+", " // +scoreD+", "+scoreFromIns+"+"+ // (streak==0 ? POINTSoff_INS : streakmaxScore){ maxScore=x; maxCol=col; maxState=state; } } } maxScore>>=SCOREOFFSET; // System.err.println("Returning "+rows+", "+maxCol+", "+maxState+", "+maxScore); return new int[] {rows, maxCol, maxState, maxScore}; } /** Generates the match string */ public final byte[] traceback(byte[] read, byte[] ref, int refStartLoc, int refEndLoc, int row, int col, int state){ // assert(false); assert(refStartLoc<=refEndLoc) : refStartLoc+", "+refEndLoc; assert(row==rows); byte[] out=new byte[row+col-1]; //TODO if an out of bound crash occurs, try removing the "-1". int outPos=0; if(state==MODE_INS){ //TODO ? Maybe not needed. } while(row>0 && col>0){ // byte prev0=(byte)(packed[state][row][col]&MODEMASK); final int time=packed[state][row][col]&TIMEMASK; final byte prev; // System.err.println("state="+state+", prev="+prev+", row="+row+", col="+col+", score="+scores[state][row][col]); if(state==MODE_MS){ if(time>1){prev=(byte)state;} else{ final int scoreFromDiag=packed[MODE_MS][row-1][col-1]&SCOREMASK; final int scoreFromDel=packed[MODE_DEL][row-1][col-1]&SCOREMASK; final int scoreFromIns=packed[MODE_INS][row-1][col-1]&SCOREMASK; if(scoreFromDiag>=scoreFromDel && scoreFromDiag>=scoreFromIns){prev=MODE_MS;} else if(scoreFromDel>=scoreFromIns){prev=MODE_DEL;} else{prev=MODE_INS;} } byte c=read[row-1]; byte r=ref[refStartLoc+col-1]; if(c==r){ out[outPos]='m'; }else{ if(!AminoAcid.isFullyDefined(c)){ out[outPos]='N'; }else if(!AminoAcid.isFullyDefined(r)){ // out[outPos]='X'; out[outPos]='N'; }else{ out[outPos]='S'; } } row--; col--; }else if(state==MODE_DEL){ if(time>1){prev=(byte)state;} else{ final int scoreFromDiag=packed[MODE_MS][row][col-1]&SCOREMASK; final int scoreFromDel=packed[MODE_DEL][row][col-1]&SCOREMASK; if(scoreFromDiag>=scoreFromDel){prev=MODE_MS;} else{prev=MODE_DEL;} } byte r=ref[refStartLoc+col-1]; out[outPos]='D'; col--; }else{ if(time>1){prev=(byte)state;} else{ final int scoreFromDiag=packed[MODE_MS][row-1][col]&SCOREMASK; final int scoreFromIns=packed[MODE_INS][row-1][col]&SCOREMASK; if(scoreFromDiag>=scoreFromIns){prev=MODE_MS;} else{prev=MODE_INS;} } assert(state==MODE_INS) : state; if(col==0){ out[outPos]='X'; }else if(col>=columns){ out[outPos]='Y'; }else{ out[outPos]='I'; } row--; } // assert(prev==prev0); state=prev; outPos++; } assert(row==0 || col==0); if(col!=row){ while(row>0){ out[outPos]='X'; outPos++; row--; col--; } if(col>0){ //do nothing } } //Shrink and reverse the string byte[] out2=new byte[outPos]; for(int i=0; i=0 && maxState=0 && maxRow=0 && maxColdifC){ score+=POINTSoff_NOREF; difR--; } row+=difR; col+=difR; } assert(refStartLoc<=refEndLoc); assert(row==rows); final int bestRefStop=refStartLoc+col-1; while(row>0 && col>0){ // System.err.println("state="+state+", row="+row+", col="+col); // byte prev0=(byte)(packed[state][row][col]&MODEMASK); final int time=packed[state][row][col]&TIMEMASK; final byte prev; if(state==MODE_MS){ if(time>1){prev=(byte)state;} else{ final int scoreFromDiag=packed[MODE_MS][row-1][col-1]&SCOREMASK; final int scoreFromDel=packed[MODE_DEL][row-1][col-1]&SCOREMASK; final int scoreFromIns=packed[MODE_INS][row-1][col-1]&SCOREMASK; if(scoreFromDiag>=scoreFromDel && scoreFromDiag>=scoreFromIns){prev=MODE_MS;} else if(scoreFromDel>=scoreFromIns){prev=MODE_DEL;} else{prev=MODE_INS;} } row--; col--; }else if(state==MODE_DEL){ if(time>1){prev=(byte)state;} else{ final int scoreFromDiag=packed[MODE_MS][row][col-1]&SCOREMASK; final int scoreFromDel=packed[MODE_DEL][row][col-1]&SCOREMASK; if(scoreFromDiag>=scoreFromDel){prev=MODE_MS;} else{prev=MODE_DEL;} } col--; }else{ assert(state==MODE_INS); if(time>1){prev=(byte)state;} else{ final int scoreFromDiag=packed[MODE_MS][row-1][col]&SCOREMASK; final int scoreFromIns=packed[MODE_INS][row-1][col]&SCOREMASK; if(scoreFromDiag>=scoreFromIns){prev=MODE_MS;} else{prev=MODE_INS;} } row--; } if(col<0){ System.err.println(row); break; //prevents an out of bounds access } // assert(prev==prev0); state=prev; // System.err.println("state2="+state+", row2="+row+", col2="+col+"\n"); } // assert(false) : row+", "+col; if(row>col){ col-=row; } final int bestRefStart=refStartLoc+col; score>>=SCOREOFFSET; int[] rvec; if(bestRefStartrefEndLoc){ //Suggest extra padding in cases of overflow int padLeft=Tools.max(0, refStartLoc-bestRefStart); int padRight=Tools.max(0, bestRefStop-refEndLoc); rvec=new int[] {score, bestRefStart, bestRefStop, padLeft, padRight}; }else{ rvec=new int[] {score, bestRefStart, bestRefStop}; } return rvec; } /** Will not fill areas that cannot match minScore. * @return {score, bestRefStart, bestRefStop} */ public final int[] fillAndScoreLimited(byte[] read, byte[] ref, int refStartLoc, int refEndLoc, int minScore){ int a=Tools.max(0, refStartLoc); int b=Tools.min(ref.length-1, refEndLoc); assert(b>=a); int[] score; if(b-a>=maxColumns){ System.err.println("Warning: Max alignment columns exceeded; restricting range. "+(b-a+1)+" > "+maxColumns); assert(false) : refStartLoc+", "+refEndLoc; b=Tools.min(ref.length-1, a+maxColumns-1); } int[] max=fillLimited(read, ref, a, b, minScore); score=(max==null ? null : score(read, ref, a, b, max[0], max[1], max[2])); return score; } public final static int scoreNoIndels(byte[] read, SiteScore ss){ ChromosomeArray cha=Data.getChromosome(ss.chrom); return scoreNoIndels(read, cha.array, ss.start, ss); } public final static int scoreNoIndels(byte[] read, final int chrom, final int refStart){ ChromosomeArray cha=Data.getChromosome(chrom); return scoreNoIndels(read, cha.array, refStart, null); } public final static int scoreNoIndels(byte[] read, SiteScore ss, byte[] baseScores){ ChromosomeArray cha=Data.getChromosome(ss.chrom); return scoreNoIndels(read, cha.array, baseScores, ss.start, ss); } public final static int scoreNoIndels(byte[] read, final int chrom, final int refStart, byte[] baseScores){ ChromosomeArray cha=Data.getChromosome(chrom); return scoreNoIndels(read, cha.array, baseScores, refStart, null); } public final static int scoreNoIndels(byte[] read, byte[] ref, final int refStart, final SiteScore ss){ int score=0; int mode=-1; int timeInMode=0; //This block handles cases where the read runs outside the reference //Of course, padding the reference with 'N' would be better, but... int readStart=0; int readStop=read.length; final int refStop=refStart+read.length; boolean semiperfect=true; int norefs=0; if(refStart<0){ readStart=0-refStart; score+=POINTS_NOREF*readStart; norefs+=readStart; } if(refStop>ref.length){ int dif=(refStop-ref.length); readStop-=dif; score+=POINTS_NOREF*dif; norefs+=dif; } // if(refStart<0 || refStart+read.length>ref.length){return -99999;} //No longer needed. for(int i=readStart; iref.length){ int dif=(refStop-ref.length); readStop-=dif; score+=POINTS_NOREF*dif; norefs+=dif; } // if(refStart<0 || refStart+read.length>ref.length){return -99999;} //No longer needed. for(int i=readStart; iref.length){ int dif=(refStop-ref.length); System.err.println("dif="+dif+", ref.length="+ref.length+", refStop="+refStop); readStop-=dif; score+=POINTS_NOREF*dif; } assert(refStart+readStop<=ref.length) : "readStart="+readStart+", readStop="+readStop+ ", refStart="+refStart+", refStop="+refStop+", ref.length="+ref.length+", read.length="+read.length; assert(matchReturn!=null); assert(matchReturn.length==1); if(matchReturn[0]==null || matchReturn[0].length!=read.length){ assert(matchReturn[0]==null || matchReturn[0].lengthref.length){ int dif=(refStop-ref.length); System.err.println("dif="+dif+", ref.length="+ref.length+", refStop="+refStop); readStop-=dif; score+=POINTS_NOREF*dif; } assert(refStart+readStop<=ref.length) : "readStart="+readStart+", readStop="+readStop+ ", refStart="+refStart+", refStop="+refStop+", ref.length="+ref.length+", read.length="+read.length; assert(matchReturn!=null); assert(matchReturn.length==1); if(matchReturn[0]==null || matchReturn[0].length!=read.length){ assert(matchReturn[0]==null || matchReturn[0].length=0) : width+", "+s.length()+", "+s+", "+num+", "+spaces; for(int i=0; i=0) : width+", "+s.length()+", "+s+", "+num+", "+spaces; for(int i=0; i>SCOREOFFSET; String s=" "+num; if(s.length()>width){s=num>0 ? maxString : minString;} int spaces=width-s.length(); assert(spaces>=0) : width+", "+s.length()+", "+s+", "+num+", "+spaces; for(int i=0; i=0); for(int i=0; iLIMIT_FOR_COST_4){ score+=(len-LIMIT_FOR_COST_4)*POINTS_DEL4; len=LIMIT_FOR_COST_4; } if(len>LIMIT_FOR_COST_3){ score+=(len-LIMIT_FOR_COST_3)*POINTS_DEL3; len=LIMIT_FOR_COST_3; } if(len>1){ score+=(len-1)*POINTS_DEL2; } return score; } private static int calcDelScoreOffset(int len){ if(len<=0){return 0;} int score=POINTSoff_DEL; if(len>LIMIT_FOR_COST_4){ score+=(len-LIMIT_FOR_COST_4)*POINTSoff_DEL4; len=LIMIT_FOR_COST_4; } if(len>LIMIT_FOR_COST_3){ score+=(len-LIMIT_FOR_COST_3)*POINTSoff_DEL3; len=LIMIT_FOR_COST_3; } if(len>1){ score+=(len-1)*POINTSoff_DEL2; } return score; } private static int calcMatchScoreOffset(int len){ if(len<=0){return 0;} int score=POINTSoff_MATCH; if(len>1){ score+=(len-1)*POINTSoff_MATCH2; } return score; } private static int calcSubScoreOffset(int len){ if(len<=0){return 0;} int score=POINTSoff_SUB; if(len>LIMIT_FOR_COST_3){ score+=(len-LIMIT_FOR_COST_3)*POINTSoff_SUB3; len=LIMIT_FOR_COST_3; } if(len>1){ score+=(len-1)*POINTSoff_SUB2; } return score; } public static int calcInsScore(int len){ if(len<=0){return 0;} int score=POINTS_INS; if(len>LIMIT_FOR_COST_4){ score+=(len-LIMIT_FOR_COST_4)*POINTS_INS4; len=LIMIT_FOR_COST_4; } if(len>LIMIT_FOR_COST_3){ score+=(len-LIMIT_FOR_COST_3)*POINTS_INS3; len=LIMIT_FOR_COST_3; } if(len>1){ score+=(len-1)*POINTS_INS2; } return score; } private static int calcInsScoreOffset(int len){ if(len<=0){return 0;} int score=POINTSoff_INS; if(len>LIMIT_FOR_COST_4){ score+=(len-LIMIT_FOR_COST_4)*POINTSoff_INS4; len=LIMIT_FOR_COST_4; } if(len>LIMIT_FOR_COST_3){ score+=(len-LIMIT_FOR_COST_3)*POINTSoff_INS3; len=LIMIT_FOR_COST_3; } if(len>1){ score+=(len-1)*POINTSoff_INS2; } return score; } private final int maxRows; private final int maxColumns; private final int[][][] packed; private final int[] vertLimit; private final int[] horizLimit; private final int[] insScoreArray; private final int[] delScoreArray; private final int[] matchScoreArray; private final int[] subScoreArray; CharSequence showVertLimit(){ StringBuilder sb=new StringBuilder(); for(int i=0; i<=rows; i++){sb.append(vertLimit[i]>>SCOREOFFSET).append(",");} return sb; } CharSequence showHorizLimit(){ StringBuilder sb=new StringBuilder(); for(int i=0; i<=columns; i++){sb.append(horizLimit[i]>>SCOREOFFSET).append(",");} return sb; } // public static final int MODEBITS=2; public static final int TIMEBITS=12; public static final int SCOREBITS=32-TIMEBITS; public static final int MAX_TIME=((1<