/* -*- tab-width: 4 -*-
*
* Electric(tm) VLSI Design System
*
* File: NanosimJtag.java
* Written by Jonathan Gainsley, Sun Microsystems.
*
* Copyright (c) 2005, Oracle and/or its affiliates. All rights reserved.
*
* Electric(tm) 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.
*
* Electric(tm) 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 Electric(tm); see the file COPYING. If not, write to
* the Free Software Foundation, Inc., 59 Temple Place, Suite 330,
* Boston, Mass 02111-1307, USA.
*/
package com.sun.electric.tool.simulation.test;
import java.util.List;
import java.util.ArrayList;
/**
* Parent class for {@link NanosimJtagTester} and {@link NanosimJtagSubchainTester}.
* Contains common code shared between the child classes.
*/
public abstract class NanosimJtag extends JtagTester {
protected final NanosimModel nm;
protected float tapVolt; // ignored at this time
protected double delay; // delay in ns
private static final boolean DEBUG = false;
NanosimJtag(NanosimModel nm) {
this.nm = nm;
}
void configure(float tapVolt, long kiloHerz) {
this.tapVolt = tapVolt;
this.delay = (1.0/kiloHerz * 1e6) / 2;
}
void disconnect() {}
void setLogicOutput(int index, boolean newLevel) {
System.out.println("Nanosim JtagTester does not support 'setLogicOutput("+index+", "+newLevel+"). Use LogicSettable instead.");
}
public boolean isBypassScanning() {
return nm.isBypassScanning();
}
// ==========================================================
protected void doBypassScanning(ChainNode chain, boolean readEnable, boolean writeEnable) {
// this mode bypasses scanning, and reads/writes directly
// to the bits the scan chain controls, as long as those
// nets have been defined in the XML file.
if (DEBUG) System.out.println("Scanning in "+chain.getName()+" in bypass-scanning mode");
// initialize out bits last data scanned in
// this prevents problems with uninitialized bit vectors if the user does not read
if (chain.getOutBitsExpected().isInvalid())
chain.getOutBits().set(0, chain.getOutBitsExpected().getNumBits(), false);
else
chain.getOutBits().put(0, chain.getOutBitsExpected());
if (readEnable) {
chain.getOutBits().put(0, readDirect(chain));
}
if (writeEnable) {
writeDirect(chain);
checkDataNets(chain, 0);
checkDataNets(chain, 1);
}
}
/**
* Get a list of DataNets from the chain.
* @param chain the chain to read
* @param set which set to get. Currently only two supported, so only 0 or 1
* @return a list of DataNets of the (hierarchical) data out net names.
* May contain null for undefined nets
*/
protected static List getDataNets(SubchainNode chain, int set) {
// get the scan chain data nets node corresponding to the chain
MyTreeNode system = chain.getParent().getParent(); // this is the system node
MyTreeNode scanchainnets = MyTreeNode.getNode(system, XMLIO.SCAN_CHAIN_DATA_NETS);
if (scanchainnets == null) return getDataNetsOld(chain, set); // must be using old version
SubchainNode datachain = (SubchainNode)MyTreeNode.getNode(scanchainnets, chain.getName());
if (datachain == null) {
// if more than one jtag contoller, datanet chain name is prepended with chip name
MyTreeNode chip = chain.getParent();
datachain = (SubchainNode)MyTreeNode.getNode(scanchainnets, chip.getName()+"_"+chain.getName());
}
if (datachain == null) return getDataNetsOld(chain, set);
List datanets = new ArrayList();
for (int i=0; i<datachain.getChildCount(); i++) {
SubchainNode subnode = (SubchainNode)datachain.getChildAt(i);
if (set == 0) {
datanets.add(subnode.getDataNet());
} else {
datanets.add(subnode.getDataNet2());
}
}
return datanets;
}
/**
* Get a list of DataNets from the chain.
* @param chain the chain to read
* @param set which set to get. Currently only two supported, so only 0 or 1
* @return a list of DataNets of the (hierarchical) data out net names.
* May contain null for undefined nets
* @deprecated this was used when the xml file contains data nets specified hierarchically
* along with the scan chain bits. I have since split the data nets out into a separate,
* flat listing.
*/
protected static List getDataNetsOld(SubchainNode chain, int set) {
// get names of data out bits
if (chain.getChildCount() == 0) {
List list = new ArrayList();
SubchainNode.DataNet dataNet;
if (set == 0) dataNet = chain.getDataNet();
else dataNet = chain.getDataNet2();
if (dataNet == null || dataNet.getName().equals("")) {
// pad list with correct number of null entries
for (int i=0; i<chain.getLength(); i++)
list.add(null);
return list;
}
MyTreeNode [] hier = chain.getParent().getHierarchy();
StringBuffer newPath = new StringBuffer();
for (int j=0; j<hier.length; j++) {
// remove the chip name and the chain name (3), as they are
// not part of the spice hierarchy.
if (j<3) continue;
newPath.append("X"+hier[j].getName()+".");
}
//newPath.append("X"+chain.getName()+".");
// special case: if Fake Chain, do not use the name in the
// hierarchical path to the dataNet. This is because schematics
// that do not have scanChainElements must refer to data nets
// in the schematic, which are at a level above where a scanChain.dataNet
// net would be
boolean fakeChain = false;
if (chain.getParentChain().getOpcode().equals("fakeChain")) {
// this is a fake chain
fakeChain = true;
}
Name netName = Name.findName(dataNet.getName());
if (fakeChain) {
for (int j=0; j<netName.busWidth(); j++) {
String net = netName.subname(j).toString();
// create hierarchical spice net name
SubchainNode.DataNet singleNet = new SubchainNode.DataNet(newPath.toString() + net,
dataNet.isReadable(), dataNet.isWriteable(), dataNet.isInverted());
list.add(singleNet);
//System.out.println("DataNet added: "+singleNet);
}
} else {
// chain may be bussed, bussed name should match length of chain
Name dataName = Name.findName(chain.getName());
for (int i=0; i<dataName.busWidth(); i++) {
netName = Name.findName(dataNet.getName());
for (int j=0; j<netName.busWidth(); j++) {
String net = "x" + dataName.subname(i).toString() + "." + netName.subname(j).toString();
// create hierarchical spice net name
SubchainNode.DataNet singleNet = new SubchainNode.DataNet(newPath.toString() + net,
dataNet.isReadable(), dataNet.isWriteable(), dataNet.isInverted());
list.add(singleNet);
//System.out.println("DataNet added: "+singleNet);
}
}
}
if (list.size() != chain.getLength()) {
System.out.println("Error: data net list of size "+list.size()+" does not match length of chain "+chain.getName()+" of length "+chain.getLength());
list.clear();
for (int i=0; i<chain.getLength(); i++)
list.add(null);
return list;
}
return list;
} else {
// this just contains more sub nodes
List list = new ArrayList();
for (int i=0; i<chain.getChildCount(); i++) {
SubchainNode subnode = (SubchainNode)chain.getChildAt(i);
list.addAll(getDataNets(subnode, set));
}
return list;
}
}
/**
* Check that the bits in the chain have been applied to the dataNets for the
* scan chain. This only applies for scan bits that have their "dataNet" or
* "dataNet2" property in the XML file.
* @param chain the scan chain
* @param set which set of nets to check. Only 0 and 1 currently.
* @return true if a discrepancy found, false otherwise.
*/
protected boolean checkDataNets(ChainNode chain, int set) {
boolean foundDiscrepancy = false;
List dataNets = getDataNets(chain, set);
for (int i=0; i<dataNets.size(); i++) {
SubchainNode.DataNet dataNet = (SubchainNode.DataNet)dataNets.get(i);
//System.out.println(i+":\t"+netName);
if (dataNet == null) continue; // undefined net
if (dataNet.isWriteable()) {
// check that data in inbits was written
int simState = nm.getNodeState(dataNet.getName());
int setState = chain.getInBits().get(i) ? 1 : 0;
if (dataNet.isInverted())
setState = (setState==1 ? 0 : 1);
if (simState != setState) {
System.out.println("Error! Attempted to set bit '"+dataNet.getName()+"' to "+setState+" via the scan chain, but its state is "+simState);
foundDiscrepancy = true;
} else {
if (DEBUG) {
System.out.println("Checked "+dataNet.getName()+" read "+simState+": ok");
}
}
}
}
return foundDiscrepancy;
}
/**
* Read directly from the data bits the scan chain controls, rather than
* applying "read" and scanning out the data. This is much faster than
* scanning them out.
* @param chain the scan chain
* @return a BitVector of the bits read directly out
*/
protected BitVector readDirect(ChainNode chain) {
List dataNets = getDataNets(chain, 0);
List dataNets2 = getDataNets(chain, 1);
BitVector outBits = new BitVector(chain.getOutBits().getNumBits(), "outBits");
int bitsRead = 0;
for (int i=0; i<outBits.getNumBits(); i++) {
SubchainNode.DataNet dataNet = (SubchainNode.DataNet)dataNets.get(i);
SubchainNode.DataNet dataNet2 = (SubchainNode.DataNet)dataNets2.get(i);
int state = readDirect(dataNet);
int state2 = readDirect(dataNet2);
if (state >=0 && state2 >= 0) {
// both were read and are valid, check that they match
if (state != state2) {
System.out.println("Error! Inconsistency reading directly from scan chain data bit "+i+
" of chain '"+chain.getName()+"', "+dataNet.getName()+" is "+state+" and "+
dataNet2.getName()+" is "+state2);
}
}
if (state < 0) {
state = state2; // state holds any valid state
}
if (state < 0) {
// nothing valid read, set it to whatever was shifted in
state = chain.getInBits().get(i) ? 1 : 0;
if (state == -2) bitsRead++; // bit was read, just in undefined state
} else {
bitsRead++;
}
outBits.set(i, state==1 ? true : false);
}
if (printInfo)
System.out.println("Info: Read directly "+bitsRead+" bits from chain '"+chain.getName()+
"' of length "+chain.getOutBits().getNumBits()+
" bits (others unchanged).");
return outBits;
}
/**
* Read directly from a scan chain data output
* @param dataNet the data net to read from
* @return the value read. 0 or 1 valid values, -1 on error
*/
private int readDirect(SubchainNode.DataNet dataNet) {
if (dataNet == null) return -1;
if (!dataNet.isReadable()) return -1;
int state = nm.getNodeState(dataNet.getName());
if (DEBUG) System.out.println("Read directly "+(dataNet.isInverted()?"(inverted)":"")+" from net "+dataNet.getName()+": "+state);
if (state == -2) System.out.println("Warning, read intermediate (undefined) voltage state from net "+dataNet.getName());
if (state < 0) return state;
if (dataNet.isInverted()) state = state==1 ? 0 : 1;
return state;
}
/**
* Write scan chain data directly to the data bits, rather than
* scanning them in and then applying "write". This should be much
* faster than scanning them in.
* @param chain the scan chain
*/
protected void writeDirect(ChainNode chain) {
List dataNets = getDataNets(chain, 0);
List dataNets2 = getDataNets(chain, 1);
int dataNetsWritten = 0;
int dataNet2sWritten = 0;
for (int i=0; i<chain.getInBits().getNumBits(); i++) {
int state = chain.getInBits().get(i) ? 1 : 0;
SubchainNode.DataNet dataNet = (SubchainNode.DataNet)dataNets.get(i);
SubchainNode.DataNet dataNet2 = (SubchainNode.DataNet)dataNets2.get(i);
if (writeDirect(dataNet, state)) dataNetsWritten++;
if (writeDirect(dataNet2, state)) dataNet2sWritten++;
}
// apply write (above writes)
nm.waitNS(delay*4);
// release writes, unless it is a fake chain control
if (!chain.getOpcode().equals("fakeChain")) {
nm.releaseNodes(getNames(dataNets));
nm.releaseNodes(getNames(dataNets2));
}
if (printInfo)
System.out.println("Info: Wrote directly "+dataNetsWritten+" bits and "+
dataNet2sWritten+" secondary bits from scan chain '"+chain.getName()+ "' of length "+
chain.getInBits().getNumBits()+" bits.");
}
/**
* Write directly to a scan chain data output
* @param dataNet the data net to write to
* @param state the state to write (0 or 1)
* @return true if written, false if not
*/
private boolean writeDirect(SubchainNode.DataNet dataNet, int state) {
if (dataNet == null) return false;
if (!dataNet.isWriteable()) return false;
int setState = state;
if (dataNet.isInverted()) state = (state==1) ? 0 : 1;
nm.setNodeState(dataNet.getName(), setState);
if (DEBUG) System.out.println("Wrote directly "+state+" to net "+dataNet.getName());
return true;
}
private static List getNames(List dataNets) {
List names = new ArrayList();
for (int i=0; i<dataNets.size(); i++) {
SubchainNode.DataNet dataNet = (SubchainNode.DataNet)dataNets.get(i);
if (dataNet == null) continue;
names.add(dataNet.getName());
}
return names;
}
}