You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
87 lines
2.3 KiB
87 lines
2.3 KiB
const web3Utils = require('web3-utils')
|
|
|
|
/**
|
|
* Writes data from the VM step to the in-memory
|
|
* coverage map constructed by the Instrumenter.
|
|
*/
|
|
class DataCollector {
|
|
constructor(instrumentationData={}){
|
|
this.instrumentationData = instrumentationData;
|
|
|
|
this.validOpcodes = {
|
|
"PUSH1": true,
|
|
}
|
|
}
|
|
|
|
/**
|
|
* VM step event handler. Detects instrumentation hashes when they are pushed to the
|
|
* top of the stack. This runs millions of times - trying to keep it fast.
|
|
* @param {Object} info vm step info
|
|
*/
|
|
step(info){
|
|
try {
|
|
if (this.validOpcodes[info.opcode.name] && info.stack.length > 0){
|
|
const idx = info.stack.length - 1;
|
|
|
|
let hash;
|
|
if (typeof info.stack[idx] === 'bigint') {
|
|
hash = '0x' + info.stack[idx].toString(16);
|
|
} else {
|
|
hash = web3Utils.toHex(info.stack[idx]).toString();
|
|
}
|
|
this._registerHash(hash)
|
|
}
|
|
} catch (err) { /*Ignore*/ };
|
|
}
|
|
|
|
// Temporarily disabled because some relevant traces aren't available
|
|
/**
|
|
* Converts pushData value to string and registers in instrumentation map.
|
|
* @param {HardhatEVMTraceInstruction} instruction
|
|
*/
|
|
/*trackHardhatEVMInstruction(instruction){
|
|
if (instruction && instruction.pushData){
|
|
let hash = `0x` + instruction.pushData.toString('hex');
|
|
this._registerHash(hash)
|
|
}
|
|
}*/
|
|
|
|
/**
|
|
* Normalizes has string and marks hit.
|
|
* @param {String} hash bytes32 hash
|
|
*/
|
|
_registerHash(hash){
|
|
hash = this._normalizeHash(hash);
|
|
|
|
if(this.instrumentationData[hash]){
|
|
this.instrumentationData[hash].hits++;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Left-pads zero prefixed bytes 32 hashes to length 66. The '59' in the
|
|
* comparison below is arbitrary. It provides a margin for recurring zeros
|
|
* but prevents left-padding shorter irrelevant hashes (like fn sigs)
|
|
*
|
|
* @param {String} hash data hash from evm stack.
|
|
* @return {String} 0x prefixed hash of length 66.
|
|
*/
|
|
_normalizeHash(hash){
|
|
if (hash.length < 66 && hash.length > 59){
|
|
hash = hash.slice(2);
|
|
while(hash.length < 64) hash = '0' + hash;
|
|
hash = '0x' + hash
|
|
}
|
|
return hash;
|
|
}
|
|
|
|
/**
|
|
* Unit test helper
|
|
* @param {Object} data Instrumenter.instrumentationData
|
|
*/
|
|
_setInstrumentationData(data){
|
|
this.instrumentationData = data;
|
|
}
|
|
}
|
|
|
|
module.exports = DataCollector; |