Skip to content

Latest commit

 

History

History
838 lines (770 loc) · 26.3 KB

File metadata and controls

838 lines (770 loc) · 26.3 KB

PoDemFan_N-detect_ATPG_Test_Compression

2023 VLSI Testing Final Project

In all three directories FAN_PODEM_PARALLEL, FANV2_PODEMV1, PODEM

  • type make to compile the code into executables
  • type make clean to clean the executables

Command :

./atpg -tdfatpg -ndet n -compression *.ckt > *.pat

PODEM

Implement and modify the PODEM algorithm to generate patterns for transition delay fault. This part is done by Hsin-Tzu Chang.

In this part, we modified the source code of PODEM provided by the VLSI Testing course in National Taiwan University and implemented the $N$-detect transition delay fault ATPG. For the detailed modification, please refer to the Appendix.

The following paragraphs introduce four main parts of the program:

Primary Fault Test Pattern Generation

For primary fault test pattern generation, we introduce two boolean flags, valid_v1 and valid_v2, to keep track of the states of V1 and V2 generation.

For a selected fault, we backtrace one level for fault activation and fault detection respectively. If a conflict occurs, the test generation fails. If no conflict occurs, we set valid_v1 (valid_v2) to true if the fault has been activated, and false if it hasn't.

Next, several rounds of concurrent backtrack are applied. We apply backtrack between the two decision trees according to the value of flags. The procedure is repeated until a test is successfully generated or the program reaches the user defined backtrack limit.

By performing concurrent backtrack, the algorithm has higher flexibility and is more likely to generate valid test patterns.

Primary Fault Selection

We implement two flows for selecting the primary fault for each round. In flow 1, we repeatedly choose the same fault until it is $N$-detected or the ATPG fails to generate test patterns.

In flow 2 (default), the process involves $N$ rounds of test pattern generation, from $K = 1$ to $K = N$. For each round, we traverse the fault list and only select the fault with a current detected time exactly equal to $K-1$. Note that the detections in DTC are also taken into account.

Dynamic Test Compression

We apply the PODEM-X algorithm for DTC. In order to improve runtime, the test pattern generation for secondary faults is simplified. We generate V2 first and then generate V1 based on V2. If V1 fails, no backtracking is applied on V2.

For the secondary fault selection, we choose a primary output (PO) with an unknown value and apply a breadth-first search (BFS) starting from the selected PO. Candidate faults are pushed into a queue and tried in order.

Static Test Compression

First we apply reverse order fault simulation to remove redundant test patterns. This greatly reduces the test length.

For further improvement, we apply several times of random order fault simulation. For each round, we do fault simulation for each pattern with random order and remove the redundant faults.

This procedure would be repeated until there is no further improvement for 5 consecutive rounds.

PODEM V1 + FAN V2

Implement FAN to generate V2 and PODEM to generate V1 for transition delay faults. This part is done by Wei-Shen Wang.

The circuit ckt-to-verilog conversion for FAN_ATPG is done by Hsin-Tzu Chang and Yu-Hung Pan.

The modified command line is done by Yu-Hung Pan.

In this part, we implement FAN ATPG utilizing the open source code FAN_ATPG. The original open source code only generate pattern set for stuck-at faults. Therefore, we need to implement transition-delay-fault LOS ATPG by adding some functions to the open source code. Following are some important codes and functions.

There are two flows we implemented.

The first and default flow is v1 first test pattern generation. It involves using the PODEM algorithm to generate V1 first, and then we shift the pattern and do logic simulation. With some gates' values specified, we use the values as a constraint and use the FAN algorithm to generate V2.

The second and backup flow is v2 first test pattern generation. It involves using the FAN algorithm to generate V2 first, and then we shift back the pattern and do logic simulation. With some gates' values specified, we use the values as a constraint and use the PODEM algorithm to generate V1.

As for the modified details, please scroll to the bottom of this readme or take a direct look at the source codes.

Original FAN algorithm source code : FAN ATPG.

Parallel Processing

Run two ATPG in parallel and choose better generated pattern. This part is done by Yu-Hung Pan.

Statistics for reference

Ndet : 8, Compression : ON

PODEM :

Circuit FC TL RT(s)
c432 11.62% 170 0.1
c499 94.69% 582 2.7
c880 50.38% 299 1.6
c1355 38.41% 463 1.0
c2670 94.06% 874 13.8
c3540 23.26% 486 16.3
c6288 97.63% 387 20.3
c7522 98.31% 1199 103.0
Average 63.55% 557.5 19.9

PODEM V1 + FAN V2 :

Circuit FC TL RT(s)
c432 11.62% 170 0.5
c499 95.56% 394 7.3
c880 50.33% 368 21.9
c1355 38.41% 469 7.3
c2670 94.13% 883 422.9
c3540 23.26% 495 182.7
c6288 97.66% 375 328.2
c7522 98.21% 1945 1139
Average 63.65% 637.4 263.7

Appendix

PODEM

Specification of modified parts from the source code of PODEM provided by the VLSI Testing course in National Taiwan University.

  • New file: tdfatpg.cpp
  • Modified files:
    • main.cpp
      • Add command line flags
    • atpg.cpp
      • Modify ATPG::test() for including TDF ATPG
      • Modify backtrack_limit
    • tdfsim.cpp
      • Modify functions (return bool instead of void):
        • bool tdfault_sim_a_vector(const string &, int &)
        • bool tdfault_sim_a_vector2(const string &, int &)
      • Modify void ATPG::generate_tdfault_list() for performing secondary fault selection in dynamic test compression
      • Add new functions:
        • calculate_scoap()
        • fault_reorder()
        • reverse_order_fault_sim()
        • random_order_fault_sim()
    • atpg.h
      • Add specification for new functions and variables

For more detail, please refer to the source code.

PODEM V1 + FAN V2

As shown in the following code, we try V1 first pattern generation as default(generateTDFV1_by_PODEM_first). If the default flow fail, we try V2 first pattern generation(generateSinglePatternOnTargetTDF).

SINGLE_PATTERN_GENERATION_STATUS result;
result = generateTDFV1_by_PODEM_first(fTDF, pattern);
if (result != PATTERN_FOUND)
{
	for (int i = 0; i < pattern.PI1_.size(); ++i)
	{
		pattern.PI1_[i] = X;
		pattern.PI2_[i] = X;
	}
	result = generateSinglePatternOnTargetTDF(fTDF, pattern, false);
}

The generateTDFV1_by_PODEM_first function is implemented as follow.

Atpg::SINGLE_PATTERN_GENERATION_STATUS Atpg::generateTDFV1_by_PODEM_first(Fault targetFault, Pattern &pattern)
{
	int numOfBacktrack = 0; // backtrack times
	SINGLE_PATTERN_GENERATION_STATUS genStatus = PATTERN_FOUND;
	std::vector<bool> gateID2changed(this->pCircuit_->numGate_, false);
	std::vector<bool> gateID2scheduled(this->pCircuit_->numGate_, false);
	std::vector<bool> gateID2backtracked(this->pCircuit_->numGate_, false);
	int faulty_GateID = (targetFault.faultyLine_ == 0) ? targetFault.gateID_ : pCircuit_->circuitGates_[targetFault.gateID_].faninVector_[targetFault.faultyLine_ - 1];
	Gate &atpgForV1_faulty_gate = this->pCircuit_->circuitGates_[faulty_GateID];
	Value faultActivationValue;
	std::stack<int> decisionTree_for_V1;
	if (targetFault.faultType_ == Fault::STR)
	{
		faultActivationValue = L;
	}
	else if (targetFault.faultType_ == Fault::STF)
	{
		faultActivationValue = H;
	}
	// assign PI to atpg
	for (int i = 0; i < this->pCircuit_->numGate_; ++i)
	{
		if (i < this->pCircuit_->numPI_)
		{
			switch (pattern.PI1_[i])
			{
				case X:
					if (this->pCircuit_->circuitGates_[i].atpgVal_ != X)
					{
						this->pCircuit_->circuitGates_[i].atpgVal_ = X;
						gateID2changed[i] = true;
					}
					break;
				case H:
					if (faulty_GateID == i)
					{
						if (faultActivationValue != H)
						{
							return FAULT_UNTESTABLE;
						}
					}
					if (this->pCircuit_->circuitGates_[i].atpgVal_ != H)
					{
						this->pCircuit_->circuitGates_[i].atpgVal_ = H;
						gateID2changed[i] = true;
					}
					break;
				case L:
					if (faulty_GateID == i)
					{
						if (faultActivationValue != L)
						{
							return FAULT_UNTESTABLE;
						}
					}
					if (this->pCircuit_->circuitGates_[i].atpgVal_ != L)
					{
						this->pCircuit_->circuitGates_[i].atpgVal_ = L;
						gateID2changed[i] = true;
					}
					break;
			}
		}
		else
		{
			this->pCircuit_->circuitGates_[i].atpgVal_ = X;
		}
	}

	do
	{
		for (int i = 0; i < this->pCircuit_->numPI_; ++i)
		{
			if (gateID2changed[i])
			{
				gateID2changed[i] = false;
				for (int j = 0; j < this->pCircuit_->circuitGates_[i].numFO_; ++j)
				{
					gateID2scheduled[this->pCircuit_->circuitGates_[i].fanoutVector_[j]] = true;
				}
			}
		}

		for (int i = this->pCircuit_->numPI_; i < this->pCircuit_->numGate_; ++i)
		{
			if (gateID2scheduled[i])
			{
				gateID2scheduled[i] = false;
				Value originalVal = this->pCircuit_->circuitGates_[i].atpgVal_;
				this->pCircuit_->circuitGates_[i].atpgVal_ = this->evaluateGoodVal(this->pCircuit_->circuitGates_[i]);
				if (originalVal != this->pCircuit_->circuitGates_[i].atpgVal_)
				{
					for (int j = 0; j < this->pCircuit_->circuitGates_[i].numFO_; ++j)
					{
						gateID2scheduled[this->pCircuit_->circuitGates_[i].fanoutVector_[j]] = true;
					}
				}
			}
		}

		// if fault activated, v1 found
		if (faultActivationValue == atpgForV1_faulty_gate.atpgVal_)
		{
			// gen pattern 2 here
			genStatus = this->generateTDFV2_by_FAN_second(targetFault, pattern, false);
			if (genStatus == TDF_V2_FAIL)
			{
				goto PODEM_BACKTRACKING;
			}
			else if (genStatus == TDF_V2_FOUND)
			{
				genStatus = PATTERN_FOUND;
			}
			else
			{
				std::cerr << "unexpected result in generateTDFV1_by_PODEM_first\n";
			}
			break;
		}
		else if (atpgForV1_faulty_gate.atpgVal_ == X)
		{
			// find pi assignment
			Value piValueToAssign = faultActivationValue;
			Gate *pObjectGate = &atpgForV1_faulty_gate;
			Gate *pNextObjectGate = NULL;

			while (pObjectGate->gateId_ >= pCircuit_->numPI_)
			{
				pNextObjectGate = NULL;
				// choose object gate index
				switch (pObjectGate->gateType_)
				{
					case Gate::OR2:
					case Gate::OR3:
					case Gate::OR4:
					case Gate::OR5:
						if (piValueToAssign == H)
						{
							for (int j = 0; j < pObjectGate->numFI_; ++j)
							{
								if (this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].atpgVal_ == X)
								{
									if (!pNextObjectGate)
									{
										pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
									}
									else
									{
										if (pNextObjectGate->cc1_ < this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].cc1_)
										{
											pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
										}
									}
								}
							}
						}
						else if (piValueToAssign == L)
						{
							for (int j = 0; j < pObjectGate->numFI_; ++j)
							{
								if (this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].atpgVal_ == X)
								{
									if (!pNextObjectGate)
									{
										pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
									}
									else
									{
										if (pNextObjectGate->cc0_ > this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].cc0_)
										{
											pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
										}
									}
								}
							}
						}
						break;
					case Gate::NAND2:
					case Gate::NAND3:
					case Gate::NAND4:
						if (piValueToAssign == H)
						{
							for (int j = 0; j < pObjectGate->numFI_; ++j)
							{
								if (this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].atpgVal_ == X)
								{
									if (!pNextObjectGate)
									{
										pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
									}
									else
									{
										if (pNextObjectGate->cc0_ < this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].cc0_)
										{
											pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
										}
									}
								}
							}
						}
						else if (piValueToAssign == L)
						{
							for (int j = 0; j < pObjectGate->numFI_; ++j)
							{
								if (this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].atpgVal_ == X)
								{
									if (!pNextObjectGate)
									{
										pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
									}
									else
									{
										if (pNextObjectGate->numLevel_ > this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].numLevel_)
										{
											pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
										}
									}
								}
							}
						}
						break;
					case Gate::NOR2:
					case Gate::NOR3:
					case Gate::NOR4:
						if (piValueToAssign == H)
						{
							for (int j = 0; j < pObjectGate->numFI_; ++j)
							{
								if (this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].atpgVal_ == X)
								{
									if (!pNextObjectGate)
									{
										pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
									}
									else
									{
										if (pNextObjectGate->cc0_ > this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].cc0_)
										{
											pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
										}
									}
								}
							}
						}
						else if (piValueToAssign == L)
						{
							for (int j = 0; j < pObjectGate->numFI_; ++j)
							{
								if (this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].atpgVal_ == X)
								{
									if (!pNextObjectGate)
									{
										pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
									}
									else
									{
										if (pNextObjectGate->cc1_ < this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].cc1_)
										{
											pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
										}
									}
								}
							}
						}
						break;
					case Gate::AND2:
					case Gate::AND3:
					case Gate::AND4:
					case Gate::AND5:
					case Gate::AND8:
					case Gate::AND9:
						if (piValueToAssign == H)
						{
							for (int j = 0; j < pObjectGate->numFI_; ++j)
							{
								if (this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].atpgVal_ == X)
								{
									if (!pNextObjectGate)
									{
										pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
									}
									else
									{
										if (pNextObjectGate->cc1_ > this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].cc1_)
										{
											pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
										}
									}
								}
							}
						}
						else if (piValueToAssign == L)
						{
							for (int j = 0; j < pObjectGate->numFI_; ++j)
							{
								if (this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].atpgVal_ == X)
								{
									if (!pNextObjectGate)
									{
										pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
									}
									else
									{
										if (pNextObjectGate->cc0_ < this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]].cc0_)
										{
											pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[j]]);
										}
									}
								}
							}
						}
						break;
					case Gate::INV:
					case Gate::BUF:
						if (this->pCircuit_->circuitGates_[pObjectGate->faninVector_[0]].atpgVal_ == X)
						{
							pNextObjectGate = &(this->pCircuit_->circuitGates_[pObjectGate->faninVector_[0]]);
						}
						break;
				}

				switch (pObjectGate->gateType_)
				{
					case Gate::INV:
					case Gate::NOR2:
					case Gate::NOR3:
					case Gate::NOR4:
					case Gate::NAND2:
					case Gate::NAND3:
					case Gate::NAND4:
						if (piValueToAssign == H)
						{
							piValueToAssign = L;
						}
						else if (piValueToAssign == L)
						{
							piValueToAssign = H;
						}
						break;
				}
				pObjectGate = pNextObjectGate;
				if (pNextObjectGate == NULL)
				{
					genStatus = FAULT_UNTESTABLE;
					break;
				}
				if (pObjectGate == NULL)
				{
					break;
				}
			}
			if (pObjectGate == NULL)
			{
				genStatus = FAULT_UNTESTABLE;
				break;
			}
			else
			{
				assert(pObjectGate->gateId_ < this->pCircuit_->numPI_);
				assert(piValueToAssign != X);
				pObjectGate->atpgVal_ = piValueToAssign;
				gateID2changed[pObjectGate->gateId_] = true;
				decisionTree_for_V1.push(pObjectGate->gateId_);
			}
		}
		// else backtrack assigned PI
		else
		{
		PODEM_BACKTRACKING:
			while (true)
			{
				if (numOfBacktrack > PODEM_LIMIT)
				{
					genStatus = ABORT;
					break;
				}
				if (decisionTree_for_V1.empty())
				{
					genStatus = FAULT_UNTESTABLE;
					break;
				}
				int backtrackCandidate = decisionTree_for_V1.top();
				if (gateID2backtracked[backtrackCandidate] == true)
				{
					gateID2backtracked[backtrackCandidate] = false;
					this->pCircuit_->circuitGates_[backtrackCandidate].atpgVal_ = X;
					gateID2changed[backtrackCandidate] = true;
					decisionTree_for_V1.pop();
				}
				else
				{
					if (this->pCircuit_->circuitGates_[backtrackCandidate].atpgVal_ == H)
					{
						this->pCircuit_->circuitGates_[backtrackCandidate].atpgVal_ = L;
					}
					else if (this->pCircuit_->circuitGates_[backtrackCandidate].atpgVal_ == L)
					{
						this->pCircuit_->circuitGates_[backtrackCandidate].atpgVal_ = H;
					}
					gateID2changed[backtrackCandidate] = true;
					gateID2backtracked[backtrackCandidate] = true;
					++numOfBacktrack;
					break;
				}
			}
		}
		if (genStatus == FAULT_UNTESTABLE || genStatus == ABORT)
		{
			break;
		}
	} while (true);

	// both v1 and v2 should be found and has no contradiction
	if (genStatus == PATTERN_FOUND)
	{

		pattern.PI1_[this->pCircuit_->numPI_] = this->pCircuit_->circuitGates_[this->pCircuit_->numPI_].atpgVal_;
		for (int i = 0; i < this->pCircuit_->numPI_ - 1; ++i)
		{
			pattern.PI1_[i] = pattern.PI2_[i + 1];
		}
	}
	return genStatus;
}

Notice that the generateTDFV2_by_FAN_second function is called in generateTDFV1_by_PODEM_first Basically we just make a copy of ATPG of the current ATPG and try to generated V2 with the FAN algorithm with it. The function is implemented in code as follow.

Atpg::SINGLE_PATTERN_GENERATION_STATUS Atpg::generateTDFV2_by_FAN_second(Fault targetFault, Pattern &pattern, bool isAtStageDTC)
{
	Circuit reinitializedCircuit = *(this->pCircuit_);
	Simulator reinitializedSimulator = Simulator(&reinitializedCircuit);
	Atpg reinitializedAtpg = Atpg(&reinitializedCircuit, &reinitializedSimulator);
	int backwardImplicationLevel = 0;
	int numOfBacktrack = 0;
	bool Finish = false;
	bool faultHasPropagatedToPO = false;
	Gate *pFaultyLine = NULL;
	Gate *pLastDFrontier = NULL;
	IMPLICATION_STATUS implicationStatus;
	BACKTRACE_STATUS backtraceFlag;
	SINGLE_PATTERN_GENERATION_STATUS genStatus = TDF_V2_FOUND;
	std::vector<bool> gateID2changed(reinitializedCircuit.numGate_, false);
	std::vector<bool> gateID2scheduled(reinitializedCircuit.numGate_, false);
	// try to use copy instead of setting up again
	reinitializedAtpg.gateID_to_lineType_ = this->gateID_to_lineType_;
	reinitializedAtpg.numOfheadLines_ = this->numOfheadLines_;
	reinitializedAtpg.headLineGateIDs_ = this->headLineGateIDs_;
	reinitializedAtpg.gateID_to_uniquePath_ = this->gateID_to_uniquePath_;

	for (int i = 0; i < reinitializedCircuit.numGate_; ++i)
	{
		reinitializedCircuit.circuitGates_[i].depthFromPo_ = this->pCircuit_->circuitGates_[i].depthFromPo_;
		if (0 < i && i < reinitializedCircuit.numPI_)
		{
			reinitializedCircuit.circuitGates_[i].atpgVal_ = this->pCircuit_->circuitGates_[i - 1].atpgVal_;
			if (reinitializedCircuit.circuitGates_[i].atpgVal_ != X)
			{
				gateID2changed[i] = true;
			}
		}
		else
		{
			reinitializedCircuit.circuitGates_[i].atpgVal_ = X;
		}
	}

	for (int i = 0; i < reinitializedCircuit.numPI_; ++i)
	{
		if (gateID2changed[i])
		{
			gateID2changed[i] = false;
			Value originalVal = reinitializedCircuit.circuitGates_[i].atpgVal_;
			if (originalVal != X)
			{
				gateID_to_valModified_[i] = 1;
				for (int j = 0; j < reinitializedCircuit.circuitGates_[i].numFO_; ++j)
				{
					gateID2scheduled[reinitializedCircuit.circuitGates_[i].fanoutVector_[j]] = true;
				}
			}
		}
	}

	for (int i = reinitializedCircuit.numPI_; i < reinitializedCircuit.numGate_; ++i)
	{
		if (gateID2scheduled[i])
		{
			gateID2scheduled[i] = false;
			Value originalVal = reinitializedCircuit.circuitGates_[i].atpgVal_;
			reinitializedCircuit.circuitGates_[i].atpgVal_ = reinitializedAtpg.evaluateGoodVal(reinitializedCircuit.circuitGates_[i]);
			if (reinitializedCircuit.circuitGates_[i].atpgVal_ != X)
			{
				gateID_to_valModified_[i] = 1;
				for (int j = 0; j < reinitializedCircuit.circuitGates_[i].numFO_; ++j)
				{
					gateID2scheduled[reinitializedCircuit.circuitGates_[i].fanoutVector_[j]] = true;
				}
			}
		}
	}

	Gate *pGateForActivation = reinitializedAtpg.getGateForFaultActivation(targetFault);
	if (((pGateForActivation->atpgVal_ == L) && (targetFault.faultType_ == Fault::STR)) ||
			((pGateForActivation->atpgVal_ == H) && (targetFault.faultType_ == Fault::STF)))
	{
		return TDF_V2_FAIL;
	}

	if (pGateForActivation->atpgVal_ != X)
	{
		reinitializedAtpg.setGateAtpgValAndRunImplication(*(pGateForActivation), X);
	}

	if (!reinitializedAtpg.xPathExists(pGateForActivation))
	{
		return TDF_V2_FAIL;
	}

	// set is at stage dtc to true because needs to initialized value from p1
	pFaultyLine = reinitializedAtpg.initializeForSinglePatternGeneration(targetFault, backwardImplicationLevel, implicationStatus, true);

	if (!pFaultyLine)
	{
		return TDF_V2_FAIL;
	}
	// set backtrace flag
	backtraceFlag = INITIAL;
	while (!Finish)
	{
		if (!reinitializedAtpg.doImplication(implicationStatus, backwardImplicationLevel))
		{
			if (reinitializedAtpg.backtrackDecisionTree_.lastNodeMarked())
			{
				++numOfBacktrack;
			}
			// abort if nnumofbacktrack hits the limit
			if (numOfBacktrack > BACKTRACK_LIMIT)
			{
				genStatus = TDF_V2_FAIL;
				Finish = true;
			}

			reinitializedAtpg.clearAllEvents();

			if (reinitializedAtpg.backtrack(backwardImplicationLevel))
			{
				backtraceFlag = INITIAL;
				implicationStatus = (backwardImplicationLevel > 0) ? BACKWARD : FORWARD;
				pLastDFrontier = NULL;
			}
			else
			{
				genStatus = TDF_V2_FAIL;
				Finish = true;
			}
			continue;
		}

		// is continuation meaningful?
		if (!reinitializedAtpg.continuationMeaningful(pLastDFrontier))
		{
			backtraceFlag = INITIAL;
		}

		// if fault signal propagated to a primary output?
		if (reinitializedAtpg.checkIfFaultHasPropagatedToPO(faultHasPropagatedToPO))
		{
			if (reinitializedAtpg.checkForUnjustifiedBoundLines())
			{
				reinitializedAtpg.findFinalObjective(backtraceFlag, faultHasPropagatedToPO, pLastDFrontier);
				reinitializedAtpg.assignAtpgValToFinalObjectiveGates();
				implicationStatus = FORWARD;
				continue;
			}
			else
			{
				reinitializedAtpg.justifyFreeLines(targetFault);
				genStatus = TDF_V2_FOUND;
				Finish = true;
			}
		}
		// not propagated to po
		else
		{
			int numGatesInDFrontier = reinitializedAtpg.countEffectiveDFrontiers(pFaultyLine);

			if (numGatesInDFrontier == 0)
			{
				if (reinitializedAtpg.backtrackDecisionTree_.lastNodeMarked())
				{
					++numOfBacktrack;
				}
				if (numOfBacktrack > BACKTRACK_LIMIT)
				{
					genStatus = TDF_V2_FAIL;
					Finish = true;
				}

				reinitializedAtpg.clearAllEvents();

				if (reinitializedAtpg.backtrack(backwardImplicationLevel))
				{
					backtraceFlag = INITIAL;
					implicationStatus = (backwardImplicationLevel > 0) ? BACKWARD : FORWARD;
					pLastDFrontier = NULL;
				}
				else
				{
					genStatus = TDF_V2_FAIL;
					Finish = true;
				}
			}
			else if (numGatesInDFrontier == 1)
			{
				// unique sensitization because there exist just on path to po
				backwardImplicationLevel = reinitializedAtpg.doUniquePathSensitization(reinitializedCircuit.circuitGates_[reinitializedAtpg.dFrontiers_[0]]);
				// unique sensitization fail
				if (backwardImplicationLevel == UNIQUE_PATH_SENSITIZE_FAIL)
				{
					continue;
				}

				if (backwardImplicationLevel > 0)
				{
					implicationStatus = BACKWARD;
					continue;
				}
				else if (backwardImplicationLevel == 0)
				{
					continue;
				}
				else
				{
					// backwardImplicationLevel < 0, find an objective and set backtraceFlag and pLastDFrontier
					reinitializedAtpg.findFinalObjective(backtraceFlag, faultHasPropagatedToPO, pLastDFrontier);
					reinitializedAtpg.assignAtpgValToFinalObjectiveGates();
					implicationStatus = FORWARD;
					continue;
				}
			}
			else
			{
				// more than one dfrontiers
				reinitializedAtpg.findFinalObjective(backtraceFlag, faultHasPropagatedToPO, pLastDFrontier);
				reinitializedAtpg.assignAtpgValToFinalObjectiveGates();
				implicationStatus = FORWARD;
				continue;
			}
		}
	}
	if (genStatus == TDF_V2_FOUND)
	{
		for (int i = 0; i < reinitializedCircuit.numPI_; ++i)
		{
			pattern.PI2_[i] = reinitializedCircuit.circuitGates_[i].atpgVal_;
			if (pattern.PI2_[i] == D)
			{
				pattern.PI2_[i] = H;
			}
			if (pattern.PI2_[i] == B)
			{
				pattern.PI2_[i] = L;
			}
		}
		for (int i = 0; i < this->pCircuit_->numGate_; ++i)
		{
			this->pCircuit_->circuitGates_[i].atpgVal_ = reinitializedCircuit.circuitGates_[i].atpgVal_;
		}
	}
	return genStatus;
}

As for V2 first test pattern generation(generateSinglePatternOnTargetTDF). We implemented in a similar way shown above. The detailed implementation can be seen in the source codes.