diff --git a/Script/validate_p2pkh.js b/Script/validate_p2pkh.js new file mode 100644 index 0000000..322fbe9 --- /dev/null +++ b/Script/validate_p2pkh.js @@ -0,0 +1,70 @@ +const { sha256 } = require("js-sha256"); +const RIPEMD160 = require('ripemd160') +const { Base58 } = require("../Utils/Base58") +const { verifySignature } = require("../Utils/verifySignature") + + +module.exports.validate_p2pkh = (prevTrnx, msgHash, transactionData, index) => { + try { + let msg = msgHash; + for (let i = 0; i < transactionData.vin.length; i++) { + if (i === index) { + const scriptSig = transactionData.vin[i].scriptsig + let len = transactionData.vin[i].scriptsig.length + len = (len / 2).toString(16) + const prevout = `${len}${scriptSig}`; + let len2 = prevTrnx.prevout.scriptpubkey.length + len2 = (len2 / 2).toString(16) + const prevout2 = `${len2}${prevTrnx.prevout.scriptpubkey}`; + if (msg.includes(prevout)) { + msg = msg.replace(prevout, prevout2) + } + continue; + } + else { + const scriptSig = transactionData.vin[i].scriptsig + let len = transactionData.vin[i].scriptsig.length + len = (len / 2).toString(16) + const prevout = `${len}${scriptSig}`; + if (msg.includes(prevout)) { + msg = msg.replace(prevout, "00") + } + } + } + const hashMessage = Buffer.from(sha256(Buffer.from(sha256(Buffer.from(msg, 'hex')), "hex")), "hex") + + //Extract Address + const address = prevTrnx.prevout.scriptpubkey_address + //Extract PKH + const [, , , PKH, ,] = prevTrnx.prevout.scriptpubkey_asm.split(" "); + //Validate Address + const checksum = sha256(Buffer.from(sha256(Buffer.from(`00${PKH}`, 'hex')), "hex")).slice(0, 8) + const hash = `00${PKH}${checksum}` + const generatedAddress = Base58(Buffer.from(hash, "hex")) + + if (generatedAddress === address) { + //Extract signature and publickey + const [, signature, , publickey] = prevTrnx.scriptsig_asm.split(" "); + + //Performing OP_DUP and OP_HASH160 + const SHA256_PKH_HASH = sha256(Buffer.from(publickey, 'hex')); + const Generated_PKH = new RIPEMD160().end(Buffer.from(SHA256_PKH_HASH, 'hex')).read().toString('hex'); + + //Performing OP_EQUALVERIFY + if (Generated_PKH === PKH) { + let verify = verifySignature(signature, publickey, hashMessage) + return verify + } + else { + return false; + } + } + else { + return false; + } + } + catch (e) { + return false; + } + +} \ No newline at end of file diff --git a/Script/validate_p2sh.js b/Script/validate_p2sh.js new file mode 100644 index 0000000..f36ca25 --- /dev/null +++ b/Script/validate_p2sh.js @@ -0,0 +1,138 @@ +const { sha256 } = require("js-sha256"); +const RIPEMD160 = require('ripemd160') +const { Base58 } = require("../Utils/Base58") +const { verifySignature } = require("../Utils/verifySignature") +const { encode_compact } = require("../Utils/encode_compact") + +module.exports.validate_p2sh = (prevTrnx, msgHash, transactionData, index) => { + try { + let msg; + const ver = msgHash.slice(0, 8); + const outLength = encode_compact(transactionData.vout.length) + let outpointforhashPrevouts = ""; + let outpoint = ""; + let sequenceforhashSequence = ""; + let sequence = ""; + let scriptCode; + for (let i = 0; i < transactionData.vin.length; i++) { + let txidforHash; + let output = intToLittleEndianBytes(transactionData.vin[i].vout); + let vIndex = output.map(b => b.toString(16).padStart(2, '0')).join(''); + txidforHash = Buffer.from(transactionData.vin[i].txid, "hex").reverse().toString("hex"); + outpointforhashPrevouts += `${txidforHash}${vIndex}` + sequenceforhashSequence += Buffer.from(transactionData.vin[i].sequence.toString(16), "hex").reverse().toString("hex"); + if (i === index) { + let txidforHash; + let output = intToLittleEndianBytes(transactionData.vin[i].vout); + let vIndex = output.map(b => b.toString(16).padStart(2, '0')).join(''); + txidforHash = Buffer.from(transactionData.vin[i].txid, "hex").reverse().toString("hex"); + outpoint += `${txidforHash}${vIndex}` + sequence = Buffer.from(prevTrnx.sequence.toString(16), "hex").reverse().toString("hex"); + const [, data] = prevTrnx.scriptsig_asm.split(" ") + let len = data.length + len = (len / 2).toString(16) + const toRemoveHash = `${len}${data}` + let len2 = toRemoveHash.length + len2 = (len2 / 2).toString(16) + const toRemove = `${len2}${toRemoveHash}` + msgHash = msgHash.replace(toRemove, "") + } + + } + let inAmt = Buffer.alloc(8); + let vout = prevTrnx.prevout.value.toString(16); + if (vout.length % 2 != 0) { + vout = "0" + vout; + } + let v = Buffer.from(vout, "hex").reverse().toString("hex"); + inAmt.write(v, "hex"); + const output = msgHash.split(sequenceforhashSequence.slice(-8) + outLength.toString("hex"))[1].slice(0, -16) + const hashPrevouts = sha256(Buffer.from(sha256(Buffer.from(outpointforhashPrevouts, 'hex')), "hex")); + const hashSequence = sha256(Buffer.from(sha256(Buffer.from(sequenceforhashSequence, 'hex')), "hex")); + const hashOutputs = sha256(Buffer.from(sha256(Buffer.from(output, 'hex')), "hex")); + const HashTypenLockTime = msgHash.slice(-16); + const address = prevTrnx.prevout.scriptpubkey_address; + const [, , PKH,] = prevTrnx.prevout.scriptpubkey_asm.split(" "); + const checksum = sha256(Buffer.from(sha256(Buffer.from(`05${PKH}`, 'hex')), "hex")).slice(0, 8) + const hash = `05${PKH}${checksum}` + const generatedAddress = Base58(Buffer.from(hash, "hex")) + if (generatedAddress === address) { + if (prevTrnx.scriptsig.length > 288) { + let scriptArray = prevTrnx.scriptsig_asm.split(" "); + let signatures = []; + let publicKeys; + for (let i = 0; i < scriptArray.length; i++) { + if (scriptArray[i] === "OP_PUSHBYTES_72" || scriptArray[i] === "OP_PUSHBYTES_71") { + if (scriptArray[i + 1].slice(0, 2) === "30") { + signatures.push(scriptArray[i + 1]); + } else { + let str = scriptArray[i + 1].slice(2) + publicKeys = str.match(/.{68}/g); + } + } + } + let len = scriptArray[scriptArray.length - 1].length + len = (len / 2).toString(16) + scriptCode = `${len}${scriptArray[scriptArray.length - 1]}` + msg = ver + hashPrevouts + hashSequence + outpoint + scriptCode + inAmt.toString("hex") + sequence + hashOutputs + HashTypenLockTime + const hashMessage = Buffer.from(sha256(Buffer.from(sha256(Buffer.from(msg, 'hex')), "hex")), "hex") + for (let i = 0; i < signatures.length; i++) { + for (let j = 0; j < publicKeys.length; j++) { + let publicKey = publicKeys[j].slice(2); + //code modification needed + let result = verifySignature(signatures[i], publicKey, hashMessage) + if (result) { + return result; + } + } + } + + } else { + const [, reedemscript] = prevTrnx.scriptsig_asm.split(" "); + const SHA256_PKH_HASH = sha256(Buffer.from(reedemscript, 'hex')); + const Generated_PKH = new RIPEMD160().end(Buffer.from(SHA256_PKH_HASH, 'hex')).read().toString('hex'); + if (prevTrnx.witness.length >= 3) { + let len = prevTrnx.witness[prevTrnx.witness.length - 1].length + len = (len / 2).toString(16) + scriptCode = `${len}${prevTrnx.witness[prevTrnx.witness.length - 1]}` + } else { + scriptCode = `1976a914${reedemscript.slice(4)}88ac` + } + if (Generated_PKH === PKH) { + //Performing OP_CHECKSIG + msg = ver + hashPrevouts + hashSequence + outpoint + scriptCode + inAmt.toString("hex") + sequence + hashOutputs + HashTypenLockTime + const hashMessage = Buffer.from(sha256(Buffer.from(sha256(Buffer.from(msg, 'hex')), "hex")), "hex") + if (prevTrnx.witness.length >= 3) { + let str = prevTrnx.witness[prevTrnx.witness.length - 1].slice(2) + let publicKeys = str.match(/.{68}/g); + for (let i = 1; i < prevTrnx.witness.length - 1; i++) { + let signature = prevTrnx.witness[j] + for (let j = 0; j < publicKeys.length; j++) { + let publicKey = publicKeys[i].slice(2); + let result = verifySignature(signature, publicKey, hashMessage) + if (result) { + return result; + } + } + } + } + else { + const signature = prevTrnx.witness[0] + const publicKey = prevTrnx.witness[1] + let result = verifySignature(signature, publicKey, hashMessage) + return result + } + + } + else { + return false + } + } + } + else { + return false + } + } catch (e) { + return false + } +} diff --git a/Script/validate_p2wpkh.js b/Script/validate_p2wpkh.js new file mode 100644 index 0000000..42e3d3f --- /dev/null +++ b/Script/validate_p2wpkh.js @@ -0,0 +1,69 @@ +const { verifySignature } = require("../Utils/verifySignature") +const { BitcoinAddress } = require('bech32-buffer'); +const { sha256 } = require("js-sha256"); +const RIPEMD160 = require('ripemd160') +const { intToLittleEndianBytes } = require("../Utils/intToLittleEndianBytes.js") +const { encode_compact } = require("../Utils/encode_compact") + +module.exports.validate_p2wpkh = (prevTrnx, msgHash, transactionData, index) => { + try { + let msg; + const ver = msgHash.slice(0, 8) + const outLength = encode_compact(transactionData.vout.length) + let outpointforhashPrevouts = ""; + let outpointTxid = Buffer.from(prevTrnx.txid, "hex").reverse().toString("hex"); + const out = intToLittleEndianBytes(prevTrnx.vout); //convert to 4byte little ending + const outIndex = out.map(b => b.toString(16).padStart(2, '0')).join(''); + let sequenceforhashSequence = ""; + let sequence = Buffer.from(prevTrnx.sequence.toString(16), "hex").reverse().toString("hex"); + let scriptCode; + for (let i = 0; i < transactionData.vin.length; i++) { + let txidforHash; + let indexforHash; + txidforHash = Buffer.from(transactionData.vin[i].txid, "hex").reverse().toString("hex"); + const output = intToLittleEndianBytes(transactionData.vin[i].vout); //convert to 4byte little ending + indexforHash = output.map(b => b.toString(16).padStart(2, '0')).join(''); + outpointforhashPrevouts += `${txidforHash}${indexforHash}` + sequenceforhashSequence += Buffer.from(transactionData.vin[i].sequence.toString(16), "hex").reverse().toString("hex"); + } + + const [, , script] = prevTrnx.prevout.scriptpubkey_asm.split(" ") + scriptCode = `1976a914${script}88ac` + let inAmt = Buffer.alloc(8); + let vout = prevTrnx.prevout.value.toString(16); + if (vout.length % 2 != 0) { + vout = "0" + vout; + } + let v = Buffer.from(vout, "hex").reverse().toString("hex"); + inAmt.write(v, "hex"); + const output = msgHash.split(sequenceforhashSequence.slice(-8) + outLength.toString("hex"))[1].slice(0, -16) + const hashPrevouts = sha256(Buffer.from(sha256(Buffer.from(outpointforhashPrevouts, 'hex')), "hex")); + const hashSequence = sha256(Buffer.from(sha256(Buffer.from(sequenceforhashSequence, 'hex')), "hex")); + const hashOutputs = sha256(Buffer.from(sha256(Buffer.from(output, 'hex')), "hex")); + const HashTypenLockTime = msgHash.slice(-16); + msg = ver + hashPrevouts + hashSequence + outpointTxid + outIndex + scriptCode + inAmt.toString("hex") + sequence + hashOutputs + HashTypenLockTime + const hashMessage = Buffer.from(sha256(Buffer.from(sha256(Buffer.from(msg, 'hex')), "hex")), "hex") + const address = prevTrnx.prevout.scriptpubkey_address; + const [, , PKH] = prevTrnx.prevout.scriptpubkey_asm.split(" "); + const genAddress = new BitcoinAddress('bc', 0, Buffer.from(PKH, 'hex')); + const encodedAddress = genAddress.encode(); + if (address === encodedAddress) { + const signature = prevTrnx.witness[0] + const publicKey = prevTrnx.witness[1] + const SHA256_PKH_HASH = sha256(Buffer.from(publicKey, 'hex')); + const Generated_PKH = new RIPEMD160().end(Buffer.from(SHA256_PKH_HASH, 'hex')).read().toString('hex'); + if (Generated_PKH === PKH) { + let verify = verifySignature(signature, publicKey, hashMessage) + //Returning TRUE if OP_CHECKSIG returns 1 and FALSE if OP_CHECKSIG returns 0 + return verify + } else { + return false + } + + } else { + return false + } + } catch (e) { + return false + } +} \ No newline at end of file diff --git a/Script/validate_p2wsh.js b/Script/validate_p2wsh.js new file mode 100644 index 0000000..bcc0152 --- /dev/null +++ b/Script/validate_p2wsh.js @@ -0,0 +1,78 @@ +const { verifySignature } = require("../Utils/verifySignature") +const { BitcoinAddress } = require('bech32-buffer'); +const { sha256 } = require("js-sha256"); +const { intToLittleEndianBytes } = require("../Utils/intToLittleEndianBytes.js") +const { encode_compact } = require("../Utils/encode_compact") + +module.exports.validate_p2wsh = (prevTrnx, msgHash, transactionData, index) => { + try { + let msg; + const ver = msgHash.slice(0, 8) + const outLength = encode_compact(transactionData.vout.length) + let outpointforhashPrevouts = ""; + let outpointTxid = Buffer.from(prevTrnx.txid, "hex").reverse().toString("hex"); + const out = intToLittleEndianBytes(prevTrnx.vout); //convert to 4byte little ending + const outIndex = out.map(b => b.toString(16).padStart(2, '0')).join(''); + let sequenceforhashSequence = ""; + let sequence = Buffer.from(prevTrnx.sequence.toString(16), "hex").reverse().toString("hex"); + let scriptCode; + for (let i = 0; i < transactionData.vin.length; i++) { + let txidforHash; + let indexforHash; + txidforHash = Buffer.from(transactionData.vin[i].txid, "hex").reverse().toString("hex"); + const output = intToLittleEndianBytes(transactionData.vin[i].vout); //convert to 4byte little ending + indexforHash = output.map(b => b.toString(16).padStart(2, '0')).join(''); + outpointforhashPrevouts += `${txidforHash}${indexforHash}` + sequenceforhashSequence += Buffer.from(transactionData.vin[i].sequence.toString(16), "hex").reverse().toString("hex"); + } + + let len = prevTrnx.witness[prevTrnx.witness.length - 1].length + len = (len / 2).toString(16) + scriptCode = `${len}${prevTrnx.witness[prevTrnx.witness.length - 1]}` + let inAmt = Buffer.alloc(8); + let vout = prevTrnx.prevout.value.toString(16); + if (vout.length % 2 != 0) { + vout = "0" + vout; + } + let v = Buffer.from(vout, "hex").reverse().toString("hex"); + inAmt.write(v, "hex"); + const output = msgHash.split(sequenceforhashSequence.slice(-8) + outLength.toString("hex"))[1].slice(0, -16) + const hashPrevouts = sha256(Buffer.from(sha256(Buffer.from(outpointforhashPrevouts, 'hex')), "hex")); + const hashSequence = sha256(Buffer.from(sha256(Buffer.from(sequenceforhashSequence, 'hex')), "hex")); + const hashOutputs = sha256(Buffer.from(sha256(Buffer.from(output, 'hex')), "hex")); + const HashTypenLockTime = msgHash.slice(-16); + + msg = ver + hashPrevouts + hashSequence + outpointTxid + outIndex.toString("hex") + scriptCode + inAmt.toString("hex") + sequence + hashOutputs + HashTypenLockTime + const hashMessage = Buffer.from(sha256(Buffer.from(sha256(Buffer.from(msg, 'hex')), "hex")), "hex") + const address = prevTrnx.prevout.scriptpubkey_address; + const [, , PKH] = prevTrnx.prevout.scriptpubkey_asm.split(" "); + const genAddress = new BitcoinAddress('bc', 0, Buffer.from(PKH, 'hex')); + const encodedAddress = genAddress.encode(); + if (address === encodedAddress) { + let strArr = prevTrnx.inner_witnessscript_asm.split(" ") + let publicKeys = []; + strArr.forEach((Op_code, index) => { + if (Op_code === 'OP_PUSHBYTES_33') { + publicKeys.push(strArr[index + 1]) + } + }) + let verify = false; + for (let i = 1; i < prevTrnx.witness.length - 1; i++) { + let signature = prevTrnx.witness[i] + for (let j = 0; j < publicKeys.length; j++) { + let publicKey = publicKeys[j] + let result = verifySignature(signature, publicKey, hashMessage) + if (result) { + return result; + } + } + } + //Returning TRUE if OP_CHECKSIG returns 1 and FALSE if OP_CHECKSIG returns 0 + return verify; + } else { + return false + } + } catch (e) { + return false + } +} \ No newline at end of file diff --git a/Utils/Base58.js b/Utils/Base58.js new file mode 100644 index 0000000..0474839 --- /dev/null +++ b/Utils/Base58.js @@ -0,0 +1,41 @@ +var ALPHABET = "123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz"; + +module.exports.Base58 = (buffer) => { + var carry, digits, j; + if (buffer.length === 0) { + return ""; + } + i = void 0; + j = void 0; + digits = [0]; + i = 0; + while (i < buffer.length) { + j = 0; + while (j < digits.length) { + digits[j] <<= 8; + j++; + } + digits[0] += buffer[i]; + carry = 0; + j = 0; + while (j < digits.length) { + digits[j] += carry; + carry = (digits[j] / 58) | 0; + digits[j] %= 58; + ++j; + } + while (carry) { + digits.push(carry % 58); + carry = (carry / 58) | 0; + } + i++; + } + i = 0; + while (buffer[i] === 0 && i < buffer.length - 1) { + digits.push(0); + i++; + } + return digits.reverse().map(function (digit) { + return ALPHABET[digit]; + }).join(""); +}; diff --git a/Utils/calculateMessageHash.js b/Utils/calculateMessageHash.js new file mode 100644 index 0000000..9cbc3b3 --- /dev/null +++ b/Utils/calculateMessageHash.js @@ -0,0 +1,96 @@ +const { encode_compact } = require("../Utils/encode_compact") +const { intToLittleEndianBytes } = require("../Utils/intToLittleEndianBytes") + + + +module.exports.calculateMessageHash = (trnx) => { + if (trnx) { + + const ver = intToLittleEndianBytes(trnx.version); + const version = ver.map(b => b.toString(16).padStart(2, '0')).join(''); + const ltime = intToLittleEndianBytes(trnx.locktime); + const locktime = ltime.map(b => b.toString(16).padStart(2, '0')).join(''); + + const inLength = encode_compact(trnx.vin.length) + const outLength = encode_compact(trnx.vout.length) + let sigHash = Buffer.alloc(4); + let msgHash = `${version}${inLength}`; + for (const prevTrnx of trnx.vin) { + + const rTxid = Buffer.from(prevTrnx.txid, 'hex').reverse().toString('hex'); + const output = intToLittleEndianBytes(prevTrnx.vout); //convert to 4byte little ending + const index = output.map(b => b.toString(16).padStart(2, '0')).join(''); + + //const sequence = Buffer.from(prevTrnx.sequence.toString(16), "hex").reverse().toString("hex"); //convert to hex + const seq = intToLittleEndianBytes(prevTrnx.sequence) + const sequence = seq.map(b => b.toString(16).padStart(2, '0')).join(''); + + if (prevTrnx.prevout.scriptpubkey_type === "p2pkh") { + const [, signature, ,] = prevTrnx.scriptsig_asm.split(" ") + sigHash.writeInt32LE(signature.substr(-2), 0); //convert to 4byte little ending + const scriptSig = prevTrnx.scriptsig + let len = prevTrnx.scriptsig.length + len = (len / 2).toString(16) + const prevout = `${len}${scriptSig}`; + msgHash += `${rTxid}${index}${prevout}${sequence}` + } + if (prevTrnx.prevout.scriptpubkey_type === "p2sh") { + let signature; + if (prevTrnx.witness) { + if (prevTrnx.witness.length > 2) { + signature = prevTrnx.witness[1]; + } + else { + signature = prevTrnx.witness[0]; + } + sigHash.writeInt32LE(signature.substr(-2), 0); + const Script = prevTrnx.scriptsig; + let len = prevTrnx.scriptsig.length + len = (len / 2).toString(16) + const redeemScript = `${len}${Script}`; + msgHash += `${rTxid}${index}${redeemScript}${sequence}` + } + if (prevTrnx.scriptsig.length > 288) { + const Script = prevTrnx.scriptsig; + let len = prevTrnx.scriptsig.length + len = (len / 2).toString(16) + const redeemScript = `${len}${Script}`; + msgHash += `${rTxid}${index}${redeemScript}${sequence}` + } + } + if (prevTrnx.prevout.scriptpubkey_type === "v0_p2wpkh") { + const signature = prevTrnx.witness[0] + sigHash.writeInt32LE(signature.substr(-2), 0); + msgHash += `${rTxid}${index}${`00`}${sequence}` + } + if (prevTrnx.prevout.scriptpubkey_type === "v0_p2wsh") { + const signature = prevTrnx.witness[1] + sigHash.writeInt32LE(signature.substr(-2), 0); + msgHash += `${rTxid}${index}00${sequence}` + } + if (prevTrnx.prevout.scriptpubkey_type === "v1_p2tr") { + msgHash += `${rTxid}${index}${`00`}${sequence}` + } + } + msgHash += `${outLength}`; + for (const vTrnx of trnx.vout) { + let value = Buffer.alloc(8); + let vout = vTrnx.value.toString(16); + if (vout.length % 2 != 0) { + vout = "0" + vout; + } + let v = Buffer.from(vout, "hex").reverse().toString("hex"); + value.write(v, "hex"); + let len = vTrnx.scriptpubkey.length + len = (len / 2).toString(16) + const scriptpubkey = `${len}${vTrnx.scriptpubkey}`; + msgHash += `${value.toString("hex")}${scriptpubkey}` + } + + msgHash += `${locktime.toString("hex")}${sigHash.toString('hex')}` + + //console.log(msgHash) + return msgHash; + + } +} \ No newline at end of file diff --git a/Utils/checkDoubleSpent.js b/Utils/checkDoubleSpent.js new file mode 100644 index 0000000..f818e52 --- /dev/null +++ b/Utils/checkDoubleSpent.js @@ -0,0 +1,20 @@ +let utxoSet = {}; + +export function addToUTXOSet(txid, index) { + if (utxoSet.hasOwnProperty(index)) { + utxoSet[index].push(txid); + } else { + utxoSet[index] = [txid]; + } +} + +export function isUnspent(txid, index) { + if (utxoSet.hasOwnProperty(index)) { + return utxoSet[index].includes(txid); + } + return false; +} + +addToUTXOSet("tx1", 0); + +console.log(isUnspent("tx1", 0)); diff --git a/Utils/coinBaseTransaction.js b/Utils/coinBaseTransaction.js new file mode 100644 index 0000000..07c8f20 --- /dev/null +++ b/Utils/coinBaseTransaction.js @@ -0,0 +1,43 @@ +const { sha256 } = require("js-sha256"); + +const hash256 = (input) => { + return sha256(Buffer.from(sha256(Buffer.from(input, 'hex')), "hex")); +} + + +const WITNESS_RESERVED_VALUE = Buffer.from( + '0000000000000000000000000000000000000000000000000000000000000000', + 'hex', +) + +module.exports.coinBaseTransaction = (amt, root) => { + const ver = "01000000" + const marker = "00" + const flag = "01" + const inCount = "01" + const txid = "0000000000000000000000000000000000000000000000000000000000000000" + const vout = "ffffffff" + const scriptpubkey = "1802d4ce0c486579206974277320406D697450726173616420" + const sequence = "ffffffff" + const outCount = "02" + let value = Buffer.alloc(8); + let val = amt.toString(16); + if (val.length % 2 != 0) { + val = "0" + val; + } + let v = Buffer.from(val, "hex").reverse().toString("hex"); + value.write(v, "hex"); + const voutScriptpubkey = "1976a914edf10a7fac6b32e24daa5305c723f3de58db1bc888ac" + const witnessamount = "0000000000000000"; + const witnessReserved = WITNESS_RESERVED_VALUE.toString('hex') + const witnessReservedValue = hash256(root + witnessReserved) + let witnessScriptpubkey = `6a24aa21a9ed${witnessReservedValue}` + let len = witnessScriptpubkey.length + len = (len / 2).toString(16) + let FinalwitnessScriptpubkey = `${len}${witnessScriptpubkey}` + const stackitems = "01" + const stackitemssize = "20" + const witnessItem = "0000000000000000000000000000000000000000000000000000000000000000" + const locktime = "00000000" + return ver + marker + flag + inCount + txid + vout + scriptpubkey + sequence + outCount + value.toString("hex") + voutScriptpubkey + witnessamount + FinalwitnessScriptpubkey + stackitems + stackitemssize + witnessItem + locktime +} \ No newline at end of file diff --git a/Utils/decodeSignature.js b/Utils/decodeSignature.js new file mode 100644 index 0000000..f2aed3e --- /dev/null +++ b/Utils/decodeSignature.js @@ -0,0 +1,33 @@ +module.exports.decode = (buffer) => { + if (buffer.length < 8) throw new Error('too short'); + if (buffer.length > 72) throw new Error('too long'); + if (buffer[0] !== 0x30) throw new Error('Expected DER sequence'); + if (buffer[1] !== buffer.length - 2) + throw new Error('DER length is invalid'); + if (buffer[2] !== 0x02) throw new Error('Expected DER integer'); + const lenR = buffer[3]; + if (lenR === 0) throw new Error('Invalid R cannot be zero'); + if (5 + lenR >= buffer.length) throw new Error('R length is too long'); + if (buffer[4 + lenR] !== 0x02) throw new Error('Expected DER integer (2)'); + const lenS = buffer[5 + lenR]; + if (lenS === 0) throw new Error('S length is zero'); + if (6 + lenR + lenS !== buffer.length) throw new Error('S length is invalid'); + if (buffer[4] & 0x80) throw new Error('R value is negative'); + if (lenR > 1 && buffer[4] === 0x00 && !(buffer[5] & 0x80)) + throw new Error('R value excessively padded'); + if (buffer[lenR + 6] & 0x80) throw new Error('S value is negative'); + if (lenS > 1 && buffer[lenR + 6] === 0x00 && !(buffer[lenR + 7] & 0x80)) + throw new Error('S value excessively padded'); + return { + r: buffer.slice(4, 4 + lenR), + s: buffer.slice(6 + lenR), + }; +} + +module.exports.fromDER = (x) => { + if (x[0] === 0x00) x = x.slice(1); + const buffer = Buffer.alloc(32, 0); + const bstart = Math.max(0, 32 - x.length); + x.copy(buffer, bstart); + return buffer; +} \ No newline at end of file diff --git a/Utils/encode_compact.js b/Utils/encode_compact.js new file mode 100644 index 0000000..7d9f642 --- /dev/null +++ b/Utils/encode_compact.js @@ -0,0 +1,26 @@ +module.exports.encode_compact = (i) => { + // Convert integer to a hex string with the correct prefix based on size + let compactSize; + if (i <= 252) { + compactSize = Buffer.from(i.toString(16).padStart(2, "0"), "hex") + .reverse() + .toString("hex"); // Pack as 1-byte hex string + } else if (i > 252 && i <= 65535) { + compactSize = + "fd" + + Buffer.from(i.toString(16).padStart(4, "0"), "hex") + .reverse() + .toString("hex"); // Pack as 2-byte hex string with 'fd' prefix + } else if (i > 65535 && i <= 4294967295) { + compactSize = + "fe" + + Buffer.from(i.toString(16).padStart(8, "0"), "hex") + .reverse() + .toString("hex"); // Pack as 4-byte hex string with 'fe' prefix + } else if (i > 4294967295 && i <= 18446744073709551615) { + compactSize = "ff" + Buffer.from(i.toString(16).padStart(16, "0"), "hex") + .reverse() + .toString("hex"); // Pack as 8-byte hex string with 'ff' prefix + } + return compactSize; +} \ No newline at end of file diff --git a/Utils/generateMerkleRoot.js b/Utils/generateMerkleRoot.js new file mode 100644 index 0000000..43b2ca3 --- /dev/null +++ b/Utils/generateMerkleRoot.js @@ -0,0 +1,31 @@ +const { sha256 } = require("js-sha256"); + +const hash256 = (input) => { + return sha256(Buffer.from(sha256(Buffer.from(input, 'hex')), "hex")); +} + +module.exports.generateMerkleRoot = (txids) => { + if (txids.length === 0) return null + + // reverse the txids + let allTrxn = txids.map((txid) => Buffer.from(txid, 'hex').reverse().toString('hex')) + + while (allTrxn.length > 1) { + const hashTrxn = [] + + for (let i = 0; i < allTrxn.length; i += 2) { + let pairTrxn + if (i + 1 === allTrxn.length) { + // In case of odd number of elements duplicate the last trxn + pairTrxn = hash256(allTrxn[i] + allTrxn[i]) + } else { + pairTrxn = hash256(allTrxn[i] + allTrxn[i + 1]) + } + hashTrxn.push(pairTrxn) + } + + allTrxn = hashTrxn + } + + return allTrxn[0] +} \ No newline at end of file diff --git a/Utils/intToLittleEndianBytes.js b/Utils/intToLittleEndianBytes.js new file mode 100644 index 0000000..e876b65 --- /dev/null +++ b/Utils/intToLittleEndianBytes.js @@ -0,0 +1,7 @@ +module.exports.intToLittleEndianBytes = (n) => { + const bytes = new Array(4).fill(0); + for (let i = 0; i < 4; i++) { + bytes[i] = (n >>> (i * 8)) & 0xFF; + } + return bytes; +} diff --git a/Utils/mineBlock.js b/Utils/mineBlock.js new file mode 100644 index 0000000..dad8c6a --- /dev/null +++ b/Utils/mineBlock.js @@ -0,0 +1,37 @@ +const { sha256 } = require("js-sha256"); +const { intToLittleEndianBytes } = require("./intToLittleEndianBytes") +module.exports.mineBlock = (merkleRoot) => { + let blockHeaderHex = ""; + const version = 0x00000004; + const prevBlock = "0000000000000000000000000000000000000000000000000000000000000000" + const time = Math.floor(new Date().getTime() / 1000); + const bits = "1f00ffff"; + let nonce = 1; + while (true) { + const blockHeader = { + version, + prevBlock, + merkleRoot, + time, + bits, + nonce, + }; + const blockHash = sha256(sha256(JSON.stringify(blockHeader))); + let valid = true; + for (let i = 0; i < bits.length; i++) { + if (parseInt(blockHash[i], 16) > parseInt(bits[i], 16)) { + valid = false; + break; + } else if (parseInt(blockHash[i], 16) < parseInt(bits[i], 16)) { + break; + } + } + if (valid) { + const out = intToLittleEndianBytes(nonce) + const finalNonce = out.map(b => b.toString(16).padStart(2, '0')).join(''); + blockHeaderHex = intToLittleEndianBytes(version).map(b => b.toString(16).padStart(2, '0')).join('')+prevBlock+merkleRoot+Buffer.from(time.toString(16), "hex").reverse().toString("hex")+Buffer.from(bits, "hex").reverse().toString("hex")+finalNonce + return blockHeaderHex; + } + nonce++; + } +} \ No newline at end of file diff --git a/Utils/readFiles.js b/Utils/readFiles.js new file mode 100644 index 0000000..50e1cd7 --- /dev/null +++ b/Utils/readFiles.js @@ -0,0 +1,6 @@ +const fs = require("fs"); + +module.exports.readMempool = () => { + return fs.readdirSync("./mempool"); +} + diff --git a/Utils/verifySignature.js b/Utils/verifySignature.js new file mode 100644 index 0000000..7794883 --- /dev/null +++ b/Utils/verifySignature.js @@ -0,0 +1,17 @@ +const { decode } = require("../Utils/decodeSignature") +const { fromDER } = require("../Utils/decodeSignature") +const secp256k1 = require('secp256k1') + + +module.exports.verifySignature = (signature, publicKey, hashMessage) => { + const signatureBuffer = Buffer.from(signature, 'hex').slice(0, -1); + const obj = decode(signatureBuffer) + const r = fromDER(obj.r); + const s = fromDER(obj.s); + const Fsignature = Buffer.concat([r, s], 64) + const signatureUint8Array = new Uint8Array(Fsignature); + const publickeyBuffer = Buffer.from(publicKey, 'hex'); + const publickeyUint8Array = new Uint8Array(publickeyBuffer); + const verify = secp256k1.ecdsaVerify(signatureUint8Array, hashMessage, publickeyUint8Array) + return verify +} diff --git a/index.js b/index.js new file mode 100644 index 0000000..0331a1d --- /dev/null +++ b/index.js @@ -0,0 +1,104 @@ +const fs = require("fs"); +const { sha256 } = require("js-sha256"); +const { readMempool } = require("./Utils/readFiles") +const { validate_p2pkh } = require("./Script/validate_p2pkh") +const { validate_p2sh } = require("./Script/validate_p2sh") +const { validate_p2wpkh } = require("./Script/validate_p2wpkh") +const { validate_p2wsh } = require("./Script/validate_p2wsh") +const { calculateMessageHash } = require("./Utils/calculateMessageHash") +const { mineBlock } = require("./Utils/mineBlock") +const { generateMerkleRoot } = require("./Utils/generateMerkleRoot") +const { coinBaseTransaction } = require("./Utils/coinBaseTransaction") + +function main() { + + const mempoolFiles = readMempool(); + const transactions = []; + let amt = 650000000; + let merkleRoot = ""; + mempoolFiles.forEach((file) => { + const transactionData = JSON.parse( + fs.readFileSync(`./mempool/${file}`, "utf-8"), + ); + let verify = true; + let input_value = 0; + let output_value = 0; + const maxLength = Math.max(transactionData.vin.length, transactionData.vout.length); + + for (let i = 0; i < maxLength; i++) { + if (i < transactionData.vin.length) { + input_value += transactionData.vin[i].prevout.value; + if (transactionData.vin[i].sequence < 4294967293) { + if (transactionData.vin[i].sequence === 0) { + verify = true; + continue; + } + else { + verify = false; + break; + } + } + } + if (i < transactionData.vout.length) { + output_value += transactionData.vout[i].value; + if (transactionData.vout[i].scriptpubkey_type === "op_return") { + verify = false; + break; + } + } + if (maxLength > 2) { + verify = false; + break; + } + } + if (verify && input_value > output_value) { + let isVerify = false; + const msgHash = calculateMessageHash(transactionData); + for (const [index, prevTrnx] of transactionData.vin.entries()) { + if (prevTrnx.prevout.scriptpubkey_type === "p2pkh") { + let res = validate_p2pkh(prevTrnx, msgHash, transactionData, index); + if (res) { isVerify = true; continue; } + else isVerify = false; break; + } + if (prevTrnx.prevout.scriptpubkey_type === "p2sh") { + let res = validate_p2sh(prevTrnx, msgHash, transactionData, index); + if (res) { isVerify = true; continue; } + else isVerify = false; break; + } + if (prevTrnx.prevout.scriptpubkey_type === "v0_p2wpkh") { + let res = validate_p2wpkh(prevTrnx, msgHash, transactionData, index); + if (res) { isVerify = true; continue; } + else isVerify = false; break; + } + if (prevTrnx.prevout.scriptpubkey_type === "v0_p2wsh") { + let res = validate_p2wsh(prevTrnx, msgHash, transactionData, index); + if (res) { isVerify = true; continue; } + else isVerify = false; break; + } + if (prevTrnx.prevout.scriptpubkey_type === "v1_p2tr") { + isVerify = true; + continue; + } + } + if (isVerify) { + const txid = Buffer.from(sha256(Buffer.from(sha256(Buffer.from(msgHash.slice(0, -8), 'hex')), 'hex')), 'hex').reverse().toString('hex'); + transactions.push(txid) //push the txid + amt += input_value - output_value; + } + } + }); + const transactionIDs = [...transactions.map(tx => tx)]; + /* const out = transactions.map(tx => `"${tx}"`).join(',\n'); + const outputString = `[${out}]`; + fs.writeFileSync('invalid.txt', outputString); */ + merkleRoot = generateMerkleRoot(transactionIDs) + /* console.log(merkleRoot) */ + const blockHeader = mineBlock(merkleRoot); + + const coinbaseTransaction = coinBaseTransaction(amt, merkleRoot); // Constructing coinbase transaction + + const output = `${blockHeader}\n${coinbaseTransaction}\n${transactionIDs.join('\n')}`; + fs.writeFileSync('output.txt', output); +} + +main() \ No newline at end of file diff --git a/invalid.txt b/invalid.txt new file mode 100644 index 0000000..9deb68e --- /dev/null +++ b/invalid.txt @@ -0,0 +1,5291 @@ +["23f6ae5049268f63e473c5314b58a1f2d4a0d4fafd8c8ccf8efe06974ad4e077", +"f41c60ea16b21f696611aab3a62552037fbe95012ac4a459a6c19c2c23cd3222", +"6494cdb062c4a1bc3e74760f9fd389c69c9cd0dcba246c0c58ab37688671edbd", +"ede7be5d4753a25139d6b6ff033cbdf0e43b729733cb09a6a3b44472ad27bdab", +"29ce2496811d9ec9b5c1bac7664a08e63505e29f1e7a1ab72968296117b20957", +"3c6a04c31a65fb69f92158f5a1d94d848130bf206cb2b2a4f98059d724b38c7b", +"9466e7a4264414d732858f94df6964af60e9300eb62e47ceed322dd1c7759fad", +"d3aecfcd7dbf9e68b86159ee3db909821d23170befc8405a9edf2af9933122e5", +"b27bcbeae0d3682814c3f308d66d77da0ce24c2a41c130f39a5768d30b64f5b6", +"44e4868052fc4693438774c202d9e05729f41201729590a25ffbffc81bf825c9", +"6af837a8b73ea1b5422680c5f5db6517460cfffa2ad4e81718654d5f3e875144", +"75633ad93688b75d05e60de48956a0385f67b8d0a276b16a121783b7284aa832", +"7c4e6413d905a5d3db5da4adabbd17e826ea98c3c7c907d14668bdc9698e28bc", +"3e281890426350ed7ba18d327586d9d1e9060105453862b31c62c6e7aee1e42d", +"94c3a0c2aa97fe8a9ec6d542f02dd31849c1bdcc0b65aee644c7a69d8cda87ce", +"113a664a4e8c737a2faec3a3c8d7ca0c000b775217c2145555f589f23b0701c5", +"b4c54f14ea1d6755ea40dbb2d9c6e1dfa151ce80c17d1350c74aa05bbd7d94e1", +"96ccf99eda56ded7a74e6a121ac28a5474efd5967681229f9f2275621cda99d5", +"fed864de018531a2a256ea3e70335887cdf910ea7e6adbf50d9e74546d900770", +"565fb24034a0a15cfd1d7ab1ddf6621f34701bc3cf6d5a2a58a3ac852dd61b92", +"ea57b6b06cf20acfe8dc444cb8a5684efb0a2699ce91f213fdf85784d8d9f9f8", +"01dfaef60604db4eedc66e4414d5c9da0014319a2b2b7442dbd2e2f3a6f40dd1", +"3d4308c35827e01e087d212fb538df11db0453a818ebe196c52ab15ee3cb4634", +"713cfb7e01dfccd2075553b3f786ad19e6b5d3d13ba47f0d800b290749a526c9", +"30848a4d556d513b19459c96244b26d947fc9e09d25c3719a3e40c2f22213aaa", +"054e776502b4aa4cb46bd049fda6776187ddd5222a9ef6ce30816af197e949b4", +"5f2551bc1d21afccc5ca0c2e901dd5915ddfb39e0a10a51267d087a9bb5126ac", +"66ec6b2bc37c2129740c2e611ed61ff144919ff89b34b976166b1fd9bf1f92a9", +"261bb9755e4299510e74b033ea9d2c4484dd200a49681e0fe12875cb9e0f654f", +"0f1d3efdd07f9feead7b2015b02d76ba6f0d61d66173507f76d986291b2543f6", +"e8010d7db50bd68acbdb5abf3d2ea02319d8f81498b5a2c899125513cc223171", +"cc6e51d15a89ef50623b7e8cd8503472b497468f86f146357c41b3dc731e85bd", +"32271a0e0ba74a715cef2511e927e9945ef1a2757d0655130e5114949c94b205", +"e51611d99fc256b3159287178e24fe79655f60592c2cf7a0bb867235f77a3f22", +"562020c589b13373527aa059a1f176bd54a0b2187d0aa7007b41d97f99facb96", +"607e8ac9f62647ee9d2bff659c1e08542aa99eec29a2bfc031215d31f37208f5", +"1aef558645f6c9de23619ed3bc55fa93238360e2ac3aae0f22937fdf0a07c576", +"2b381f25a8fb3e6f4be79cd7777f0aee1bf4f89787d8622375c7686e962971d2", +"f1e653cde750a5a6fcaee3543e97f2de221d3a66f51024fc7831afb3de635ead", +"2d80fdcc1105546c6abd4a1bf6ff9bbf7b6735a57c47026ac54ba576477ccdac", +"4928b35d4a4375407d4984352ac715787a026bd60133644e9a4a496e9e17a73e", +"d7109a6d7dc31c66645aab1f196f7f443c551ecf84e5d779bf7960013d2e001e", +"501d7d02be0a0e7d61c4a88f4fc592f4741ef764f248fba354bd5fa7c3715943", +"da419f20fbca14290ec1354efe22493fac7be8b804e73f66ac8eada3dcccfa73", +"fe976f5cc884a042ef595778d344eeb0480eb8519b712f12c169d992e7ef6b46", +"6a00d75b34368c65711afd5a979bbc669577ac49f8885156d1f07e2c6bfa116b", +"a06bcf91beee19d07ec3e3a96279924b58951f25827514b2e9bbebf807149891", +"4df509521a5f585245ac7b63f60be24ff91617c89bf6986111d447b563b34ff2", +"1e2219a02d4ab33bf2278b5109e007a06049a6bf9301a5efc9d6655b2c89fc5b", +"fe28fa2fe28f15e55b93311c2bda8843e0ac6dee81afbf2eafef57b51601edd0", +"4138d633105a5c05a683e44762de2b4c7228aa08fb44ea199b31a002816d7afa", +"c4da5fecb9ee797031869cfd18c85032de50c1927f03bc4e3a9effbb2c25627a", +"157ba593e460784fffab72c644eb11eddf851f7a1734c1f42258e2924f3cf5e6", +"e6c4f35ce3bdf932519de9c9056b196fa84d1aa9860824b177e89a9010b04ea0", +"d015e3f97284bc217eddee377e5ab47100a7fbd13a66fdd1fcb9acfc0454f35e", +"fd634363ae6eef2d18ac8b15d4e09e9cfe42d75bc484417a5910ad636cb36d0c", +"78699d0d5143ebd1d652501c5ccc2e92c1d839e827311c9c8843e33eabf5f499", +"d4c64b4503e737de8d7a84156a3f6811cd40bd80f1173a2691193ff333039aca", +"00c4a55f63c44d2e8916cf786988cb22b7cc5630ef25cd571c8d094acd0f5d97", +"46c0d97acc09e143d657af6e9b6f19ab32fc1d6b6c385a2e3c483710dbad0d17", +"b5d3c7844be6ba6e3743404617074e8f36022d1cad3984a19f3f86bfa5f5873b", +"331c77387f77ef84eefe9cfb1364986812950262a63753bc38b56788914447dc", +"c02edd9072f7a2638e66a6e39aeb3989e0e458a8af62de4ce19199adaac7f6b4", +"751daec3159100873c9ff23eb238b1b89b6ad090b87470be5c63a634a81326a1", +"c94490de2b84dd75b512de2009f44ae1deb901f35590feb085f6a8c5ee526ec5", +"1532f6a2029b5e8c21a0754e4040c3cfd3813057833345de20114743a50682c5", +"a1abf44f506bafa649f89bbe9e43a5ce3c8a180a7512a111c0644793efb435e3", +"3ac1100e7fd86cd4a642eb98501e46ec849fe33f6ad48d2e9e4f87ea66d99d0d", +"4f6478da3089556099aacc61f5e41cc7b73af9c15b25ae0ed797c1ec31f2412b", +"9c06b9eb982ea82bf7fcbb965f3d56a179e06de9eb4e3d009df21a0097bc369f", +"22ccdabcf558c260b2872c54aafceb274230d5dc3c60fa26413124807923b3ed", +"47a4ff75b8b3f006c2651d78d50cb28eb7086bc5ee042774873e595dcefdedb4", +"55d856074c126902d70e8328be059b72424684c956c947b21d35d86119636212", +"f8d9e70096b38b30dc608dd368bde84887f5ea03185d7fb97f03b903128c92e4", +"f3e83c448489faab14a6d5b58de0feb4ab398863113e46234b8397ab8428bd39", +"fd4187d4951850f60555688933f56f8e79e09fa8556707ffc604e7227cdd43fd", +"01e1bc3b0791e66f7b6f67fd0cb4572f33fbe9ba79dbc8261696d45976aa8698", +"c4d1712173287e5a91496e2f349fe4c4eab0294e409f4a908ff3b07dbd96ffaf", +"b41ae160dedb211e207908fd64471220303110da81e6022c0219765b849523e5", +"9d3ad5c70154474405171bf4f645225891fb70aaf0a01eb44ea1ee3c509da33f", +"1e4f0f5d85bd749542f3c355ee6bd0258a77f262680d8b0b6f415b3e410184e0", +"5e9dad9498d8e9921a87d8ad814dd4e060da7e275a11d81acf0e396ca02aee5d", +"ec90cbd0bbb52f34035e7f026612f188faac34da94edaf546a13b8fc6355d39c", +"f02486d56b084ff316697f4e148b32e640e60b47eb747b3adc29d8ce0939ef61", +"239bfa7a46ae6a52586498c3b9c587f4f91c1b152e36778bd3b3c4c1a0f88df6", +"c17f5cef441960890f409260a598eb352485018e9ce1111eabadfec889f7aa5b", +"a4a5ae1072104778400c31469fe3dfcc9520e6e232b70bead411dc50f1b74141", +"f1cf56620b7a13dc05b172695aefa999f520aad96512c49995aae18fd2b6e9d0", +"3051d6ba761294e23bece4f6a364d8b9fd91a4f0635a1b909f8fc49c2ac8f140", +"97d2d0596bfc23a3da80b241f3b9c9ccc5b6d0bbca760fd447c3ce286e260a56", +"e2dbcac985d31594ffa1193638e6fa07c948ba8199158b9bcdc4ea2a39dadc50", +"fb4a49b58b5fedf3dcf916c176775f64453061d319fba56f578ed6775b432e5a", +"92f8e1ef93ce843c37ad1cfa22b98b25f08b2139f5f127e1cc913d2051577bc3", +"56689baa126382cdb7dcbc09d28addaf0622199a2c500ab60bd8661956386714", +"0adb5b155f7420324d89de01c624f2e2d77fa826fb6ead3c3ab105f37c57759c", +"15e56695d9c7c64e636e6cbe2319a5d84b2d627d3982824e75d895eae30720e1", +"7591aaee8947dba90c28c8376cb83227fe948cbd5c944e4e2c0dc76aa1e92864", +"5e1c71de443e12f56b486c4f6962ca875c8378845cb80c5082314dc4319a25ba", +"e4631e59174556759bef2f3be57eb9c87a7bc6e994d44ab4e3cd94b6c2aecaaf", +"f830f31e0a2a69c205583041674cd291a26a135616b14c72d7885a333936f60e", +"4d8a075e69dece980a5e4c390d04fe5ae3f72883b2c0c4d2ecfc9800363d6cba", +"a6fe7f814a0af9cd0702e16a22aef8abbec5da283e4c077c6680a99e34ded045", +"d8b664012ac22039449f91327b7ceb671fe008b57858e35cf350d6ae1f5d4dc0", +"eec17afc7e1d454a29dfc9b1c33287fcceeaed8745df8947f3938031e357b95b", +"58824207c3c85453dd1f58a6fde5c9a8b97353c71cb328e61efd453e10ccfdc5", +"25b74a6c21df40b1e90bc990835a3ddbd23191d556ccfbda601d987ddf91fdd7", +"5756e6e7176df6d4c88be11e2ac21a3afc28541c21ac1b30a81f1ef9b8ad0ff7", +"e9628ded332b0ff209a7fef10147fe7145d81ec72330838acdb166d571e6d377", +"db3bb1122de8df3af6ae29991b6c3ebcea77cc0ba9029d9beb9fff3637b20238", +"a5d41ca3a3014586573d8a1a88be949bfc65da057485978331062a191ae74435", +"2e291753c898481c06bdce5d2b36ea41e1350091d960570fefee483cc00ce62e", +"9a28a69d96ae817e9fa054aac4432e4c4342b9b61a61ab3d091f2a831b4c5617", +"ae55da97bbca0b686d416e57782dd68b15f70eae6c163e6775bd0dbb96d51b04", +"4abe701d6bcc7d2d64e17767375a0e53204cec53a52bc1aaeab6d7049dd30b4e", +"71bd2d163bde2a4dfa155a3f2ffbaa4586650c0f18123d511897e98f5035d397", +"b8c89f699fec55eecd4d7b1369c03ae0a051e045bba4bfbe92bbd8e836de739e", +"668f0640f48b01464fd93c70dce16c707524f5d92f8b46538066281179ed96d7", +"354a82835d1c0264bdd4ae12c8ce33af04560d96dcce0472f412828841b8e1f7", +"652517493b4632b68a5e95f926106265d5629b763d7700b04a039c0fe7c4f001", +"36aedff968224e7218506864f75a342bbef3903ea09ba5168e546d62760a1e97", +"5541b25d6cef8eb1a651b8949f0ecd1ba8b47efabac2bc415795df676edc725d", +"b8d37029a2700ee2dd9351e8019041a9ca1274952cbe012125bf56da9efd4264", +"e16e335db96384270f7c7645f6a6ae17d6ce8114dd047edda9136f0fa508ecbd", +"f116ed2d43424fc738a964f84c7add650b695d47bca83d37069f4d26f18e6bcc", +"6792217da65d73af7d9e192714055221a9e7e27ba2387e7e3da554973d2747d5", +"de59f372a007f5c83e40e5ce731bf7bdca3bbe60cfae906a1f55321e3e617a42", +"947506106adc4c2e6fcb8b5f31d8251dcfa10626a0fff1d69c9b23d59118f6b8", +"ad3eef482455fe1735435301a567edecff4f9ce39fad533eb4c7b6334c2b3bc2", +"f0f93ce326eae78a22eaec613fb0f1c2db7e8128fd350df380187b3ee4909920", +"389441c88d003100dd2b18ae814870500bc359bdcff9f26bd341c57947be3331", +"d3af98f45a1a5f7cdc1d178e79b53c0caba4520ac60464a0d1f858034318c943", +"f46e148f7c85213773d22623e4263f8694c2ca193620080d26dcd94a81ee6864", +"c01e9437a83182bffd2b2f2fd7b66bba0b7410dc475987a3fffe0754cc217a26", +"0b38505cefb64a5220e2b840cf6c1ea1f3f6ca166919635958d1c5ad859fc5da", +"2db51fe88dd9fd0da2fcffdfa68efa49e9fbf9828bfb40f65b23f4b155bbadf0", +"df1f285382850e97b092101ed8ee5738420466171beb292f5c9cf728d37b8a3c", +"334b70e6b9a08046cabc7bae845a1abf759ef02ad0ae99fbcc17f92b5976b0c9", +"474a1c56c3bc40c6bc82711a2e412e077a38a3d7ab3ff637e4d89dbe6de271fc", +"0932f8811d9a8dcee568ac01a594e26c0e4021c01842c0cc23e0382e6a1a05f6", +"4acd515ca47d6ed0a3529f614334c26835a6d4e976c14ab910bcaaa7b36d0a23", +"3c31ffdb17aff785e3db32e8d766a546498af9a259f0cb3a0daba84dad5bf6ff", +"f26f3778288357fc96f5060781ed968e95daa6137df3327496d7498d22f77fd4", +"5883ca99ee4a01ffcb31c462fb1c72dc0bae3daecd46da50f8ec6402088fa7d8", +"e0715c04f235f7a349ac4dbd0c95ede273d3300994c2f6ba7a390bf6605f32fd", +"5f2bd18e58496c3ef620400fce307688500dfe57fde35b8876b601e35b2148ed", +"8e6172636c39fd528b628d91284f964990b7be16d6187881377dcc44af62aa9b", +"435ea260de9801f2ae2d969b4e5a52304e20d89d8a19d0f796134e501bdb09d6", +"00d2b22b6998a7cdb59b8a732ef128f42b0808f18cd59b79d71dd8ead279da8f", +"fb734ca210817bb37966ef62d61635f01e59b943838945708ce527906a28f7c8", +"e6fc148621c47f1de74d21b96ba4387b5238235b63d975837471518b092ac0a9", +"a4b6a8ae2ac22f06095649240152150f96fe7131d2b2805ef53b17c7c9eba0c6", +"72f6651e049c801d3d443623ba08659bb4e7fc849ac8c31ed0b8be06f3d25caf", +"1e4927d1e0f67bb9229888755af9e9482fb9947bedf7ab9bdb6b66d6703c14cc", +"b668c8770a6ea415619a57e0cef5b8e04b72645ee93439e677327472f45116d4", +"eb050bfe81c9b0b8c22134fb8baff5ee3430150c91bda2752ab58e4700e8fe8a", +"01e64a1fb61c2f34e25130fd07d57cd60d26608563dc8c0a88c36f1ed163708e", +"2ed7b335eeeb9a73283da91089f03c5179957b3630a0146b4675aaf1aa316f68", +"81fda39873c6ebb85ff1a277c9acf4b0651b1911d0fd5e7a960a28da58d88e24", +"18e590ce44db65e64311499b0dd262ccf31daf510fce71a5cfcabc0ea887e982", +"10c67b064dd30f411bc15ac3ddda5db2f91f697caa3dde48dfe853e4af660938", +"6fbc732832a93a76e87947f53095813b2081b0696c08a62e97dee5c76956d15f", +"a27cebdf5ea140d5a01e70713d8f483ec273482721f8ee7357a2b21aeb88b675", +"91733a34f18065dcd59641011b45890114a9c2387d52e1bb7f89e3a7fc5b258e", +"d54f8f64da628450848655afa5bf68b874909434064a19b5e534cd4790e57ef5", +"4fc00f3ada7ce14c0f5b8b147a8e31045409c0f1300b8a599038a5c8aa42da64", +"3f95cffe8d3e623a5e1ef500f16c331a1999354267e030e0c7eb9a6f3a4d6e08", +"f3d221c5086b88e12afffd8e1a3da0769afc6dbbdc4f376eefc485cee2a9ce34", +"dc35b63e2ef77c76a3895cf076afb74167a3d951fad43921f4cbdd7c1e92c06d", +"36373533c2dcbd7da81ee367c1ca6cc9d20530c5153c3b870ea543156bef8e8a", +"028e81955334667857e75d9e1c5c0ecdbe8ded4b509ba60c19e525e0f9324529", +"ef9db44d230aad1eb98e4c31e21e1c32c2cd2be0924c7713b433abbdd4ee726d", +"4408e4d4f52837e9504056c8e33b7e921a10c072b520ff810fc938ad1201b652", +"0fb5f6a27d1c71ad4c058b0feaf60946523a6092f18f658efd3ccbeef6e9067b", +"06b43ff291b2c5e2c26041592097991a95905abdfbffcfa03a29a68dbe183ad1", +"a329c9ca1ab7be7efc420347706a90658a6f8828ec5372fa7de0dfd0d9f40c9c", +"8336e695305d3d09e868595b7bb0cd2113cc4b5a8fa2a3fb82d0b773c29f9d7c", +"84df2c3d71d1dae9f957b06e3b14132eb8b02bf4fab6ff3b00d60295276c3d82", +"5796b04d9f19876a295c3dbb80118418e61c23c51e8ab0aa9070fa6c3c156e9a", +"2fa6608ec030a3a1e729312fe747228fd3ccab29ce1b34f112d861625c7cc519", +"6bd77a65a1d60164a3d27421e5d0c2b924b2e7a73fa7aca104f8090954094b79", +"2bfa9fac19d0840ea40d716f1189e0ed3f025f3a234d857bb7e1f47bed99016a", +"d950ac5d6bfecdb72f83c130414ed6d370b4a4fd84af267d3573f71e0b5fc6ca", +"84e29d085579fcb08a73c3efe9d6b81cd9aa1e12a714c42a602d426af6aa8dff", +"278fe7a833141ed5e770d3f204cf837137243c59b083fa703486785d9be95cb5", +"d9d3b508ad5c2bd3aaad9a141b7d3c7512b01b9f97a67109e8eae5e406535d9b", +"41fcda1f907a7b6faccd818e3a68ebf2d35cc13e3fbcd824c6562d6d3e20de70", +"2e09cd097b1d3ecbd45ede4b4019d62480d27222aab708d3574b03503375bc51", +"5a32fc52360d43ba7764ca33a5692945ddcb11b7f4e6b2092e439fdef23d7b15", +"f75ec76214779056154f67d410521217631bd713a28637774c9f2b1abd2d5294", +"1b3ed3f93d0291d0222c1bf9d7026882f9a122443dd9afa876c99a34d3aee79d", +"e6abc2e4606be093f7ec6ad756dd8b38a5e563b2b705f269fd562cce579a9a3c", +"e72d4ac01e830056a2e59decb441e983d99478ace0e976bcb6c4d129542b4b3e", +"a32865113460c88561d6f6a338ad1075c60a983107c8b00340df9c010b7a5561", +"491be1e6c8f1104be90aafd8f096f5534e9ed876b66d602ed5985aba0536674e", +"9be75b47c00999968dbf0d0cc64cb2e11a222e65bb480e82c38dab3212dd4935", +"bcd852857df67606afb7e0dded9a2dc7c2459a02378bbbb5067d93d827050711", +"73fbe719bf90ec5abdb9282f2bcc2d3898ca69e76bc52efafc9ec15c6ecf7e6f", +"2f26ef45203ba477bb9ee38be0049d4b794ac169805608bae95ea71a7b1f0d71", +"1d5e6d9f793d1121a075bab2c1b21a6f2e9b668a1047ca7a88d22eea803ebdc3", +"54d7a3afabc8f9e46a4e650e93b167b67b262492680a0053023d5aa91d0402af", +"f4d129d2f34ef150b71c967d92b3fb4f69bb5b8ac02fbc6d7c86c0c5828b3fac", +"8664c750ee7d618f862ed441ab6bb3df65fb98f855a4ddb9bb95e1a0dba626b2", +"aa5a705c3925310445303cbaaf57ad76ef3937cd871bf6e350603c4676f8fce6", +"88d0efbe681e09bfbbd6bae2698e1e8b697f3080a627ad653d91902d61c4fe70", +"79dceb83aa2eabbb13ff2481bddc5c454129605c5be4d1da181c71880821f1d0", +"5ae95728013d96066aa5f5769a04f8b23bd69320ecf3372cbb61187dea723783", +"1c8aa90e20d6cedde86efa10e4a8bfb57a9f9e5b3e63fa7b0fade6c5cdb3def4", +"49c723037d542808c093351ec40e5469f1644b2765c8b03e9537f72b839ab44e", +"3ec180732ad5bb6bd4b5ab413b2597932883fb7575c3d3f9148dcd52f0a19993", +"ec89952759c2db1bb7ad63f54e218cc20b5251f1d20784028c89f92cecbc7eee", +"8f5d087992f35b70a24caf747bdbc2e6469e0281533ce388742c5d2149a899f6", +"0e6d2f414fcf9b377ab0fc670074ab79f8bd51bb9d384870db2bba18e1d367e7", +"27ee36ebc959a3d5acfa705a204b22b0715b78f054a2c863711b9139066c94d7", +"7986704f56406f40de9da0f75f7fadaae830c9340d893da8e853d4ed632d3613", +"d33e3ce4399a6bbff64d50b0f3afd8eec0c2788de42d2899c10593937a2d1837", +"ccc6253c11d89af11bdcf20bdf513112ff3c0bc6fe526045542e1edf9ac6ecd6", +"9813c2bc0f34210dd68f79ef060d64b8d531190b3ba1072b56c61d2d0f4fc32a", +"fb00aa533156569b654fb3f8eaf9ff926c721ffd9bd78586dbad9323b6f50bb8", +"7caf72c076992de5bb24db15004c97da0c135910eb85986ea106de9b9a750b05", +"90b22ecd1aec05105687e856b863ff3fbf8720d8436c013ec0db3dcc478794b4", +"4f7d9562c25467a64f0403269de64a906aa4dd0a0f0091a7f0d0e683872b4885", +"11b26c7d7881bae8dfd0378c0a6144f6a9480f68eca67c9e9c1347a5baef105f", +"6941caa3f62f5462668fce06fbea53b2e078072da2e6fe832fc9fd17ac97d399", +"3b0697b9bd4a23f68878dea54f7ea3f3553832c9f7f9f479ea63f4f108e2b0d9", +"4eda2b12862c3aff56323d76a33f0739c655249305ad68a49d73afd8b4ee6a89", +"4f30834329f7edad2f63824893b0970e12707b6b0e0526532ad86bcf96695ecd", +"1125d8d58f35881c0e2a67c451b2533a171c3d6dac83e3f8b8a2822fd2885add", +"ff58846e06d32ccbc92dc8055499a5700dbac3278322de87584a5641e14dd84e", +"c5a042f7ebb78342c6c19d32e1d86009161a409e338399ddf4ebdf04a3cdb2c4", +"efd4e9aea95a1e3c3d909cf3ec1cc3fb3679b8c849eb0b5e30e86fbf85304b10", +"e1f3f97522d91c07396852274dba827e3c4b82bc7543448e9576fa40f1beaea8", +"e95ba3c7d65ba8f7f49f766161b390a4044e1c73b3e5aa3091b4b41829e8b9c0", +"43030528e8a7e94f14e241024b70def32ab9c8e4b2e6e21d67c8794de81ef09e", +"88caea9fa731401104fede6d8c55a0a209f5b47d9b2e1faf7b93b1e40100b48a", +"e329ae0d700c9c77b8a6854e49d875c81fb144668c5743147a9e759945917bca", +"d99d1ed0b029244c1ae60e2bbe925d3b9f0a3455627cc65825103708cb2e54d6", +"fd4da374c3fff7764b9e621fae33ca705a198487cf9e9286a555a5043af0202a", +"30f084abfa05053fa8669ebd1b88a75db76576408a8ec055416a481b92e5920e", +"80806828c4f751441dcb7e7092b5f234a3850da315414b4401b7e6cd9f491c13", +"3c46e1f7a29a8cd6df966c1acfcbfb07725908dda4ae7c67daa3b79c4562946c", +"fa4592ebe6bcff70d8f0bd2365af32a3713dbb06d7006993c789348ce1992ae1", +"2c59b81d1e9db328c613d0871257bf64bd0e4950d3fe0e4840d3ee8f86438707", +"b3ff4998e48ae2cabd84461faf7503eecd6284017b5bc5c5c05fc44f78c6c091", +"850d9841ec040916948e2fd554f5d2fa310efd0cea960679224797b640a2a31f", +"5d557a9146084fb040a3ba3821252516b177bc58cc92ac80e7453e2e48305d34", +"e295121aa3a1f036c197a456a8de1aa3581aa1dd3521ed03c0d0b10e4d19a30c", +"645def9858a5f318b1a0c6b2dfe362e03ead69aec62d92dec7c4f64a1e0bdf58", +"363b56dc509514c4c5a213a8662cce41e726275622d76797617281410ed56e19", +"75030761cb3f55f39727acc17ca9eaad7ccb1f30fe8aac0299d8f24995f38bb7", +"616a3c7621e815d686d7e10c4c09e34b1a6381de3698e8582abf7a1c8194bf24", +"17514744d6d0dba05cf4f715306e217649fb09cd6ed960a4e492c7563cfd1779", +"7ee39c538e97395d94b36396eba5f140911d1156f0ac03945cb920f299564261", +"1a5b04ec5053064aee792ba712d6da7a97227bf38043f985b276ae416bc1b6f4", +"c8a66f02bf82632ed4ad38854a15b99fe8d4b5ed5a6d877175f364cc43145546", +"5078ccef8445b6d9106d30c5a1eef4ed506cd5e5ffe7eea95b32c784bd0649bd", +"5b5f60c3d8a4047a4094995ec01f92a4d7b7404e89b20b8dae0e6aff4f2f72ab", +"cdf38142283471fbcd28e43f58e18fdeb8c5f3f1f8e911e99f881f4c892f47f8", +"7af08f934eb0c65f9b5eb703533b83728895c770bbc267bcc915a2de319074a7", +"4c78a63ccc28ca63a9827416cfb79c1f8f48bb2d3f7a15796e22f003af2fe3f8", +"992f5212409d827a7a642572c1c0d057361ec677e7f4fb0164f2ba3782322e66", +"85f61a4287d5c5f960731c98e052ff9160ca7e97be885812d9ac30928375fadf", +"e2c8385effec5772de2ebde71c8905735dac1619c25d9ab5d78357798cd723bb", +"ed2279eef7c135efae66e5c9c3c27c3408e028035876cd3ec98fd2569db14b7a", +"e45596793a3cbfe2854718298b2873d713838e313ea6f4cc779119cee6b0d6df", +"61502ca4e5f47e0af5cffe463fba0b2f30ba66df91b46b67f1528fc594abdc4e", +"ff4cbe61d1509efea59ec719e1e373381e8506ba45f30143194db878d7f007be", +"fe4327e8a50f82c4d9c5db51fa034a3481f9081c70c5e044c6c3aa5a90629021", +"d17c2beda41c5e4d75813937ba1562d7c3092855711b60a476ced4bea19daa1b", +"88877248b629da0df5a8e31fd0d1fbecafe48a3d1fc65eaff141ddd9bdb3361a", +"32c59754db276d52aa1614fcd9da10db3b94b4567abd9d2ac8013841c66dc857", +"e883f54431fe2d695db4ddb8525d0259849c04fd2147894de4b19161fe6240c3", +"6f50621749d7e3f0e39a4e12609ca276372582a0ddf26ad20a9db154b4a11b88", +"2d747cce829789de7a7d63e28e2ca010f08ac41460fbd7cf7c2f401acd03a3d2", +"6977bf58e30728ee7e728c5d6e1a54f77e961ecdb7f7878d9c6cb66ae3cd8185", +"6d4e879ed12d13e28928117f68128fd5fdc34f7d83d8335a06199d0b3c116f03", +"3a94eee9363ddfd07d62397f79fcea8f67eae42066f157e3391c330cfb2c190d", +"97a08bd37176e19d096aea29b4e8056f37a1044cbd0c360952eff0d2f1e66e14", +"6ce439a73c43933b7fa5e1c4827289701331ba64c57c8c8144925a26414a0f20", +"59e6e8185a2ec697cc254fe507d4381badfd130081ec27e382437fde1b96e7b2", +"dcc38fad28369f8cc7f0dd965d8af1e254f58adf1aca79ac7ec3c752d35095bb", +"309e6e0a8077d89384c6f0d5acda4003b0099240c13218f4aae87bfb1a2e6549", +"aabf894a815ca072db22e11b6b296980c73efc1ff097e7a33a873b340e1e8cfe", +"d5c8db08359a29eae0a8832314eb860eab1ec9813264b708f228d0294eb46ce2", +"bfdaf4f2f9a1f5f3ca04b5ec2eada866a0cd2eca7d596367c449fed1baa0df0f", +"826028c8ceb00cc8a72262b28ab4697c9ed5100076cc27dc76381c8491c78c10", +"c915d5eaf3f70f36f1e095ec72266acf9c2e363e8a1dea27d6048818b625e0ba", +"4abc8c037943869a505a13b793fe4e09273528ad7d142627641a8bd931e7dcc8", +"4acfc5c06029499275ac0c9f4a4126251911730032e54ed02a7b67652a88ca34", +"66b926a712476c9e407169fcc89f7623898e836a115a7c2dc47322db62e048b9", +"9b2ea67df5bb1a64b6f1a78681765758f4d88a776803428ef36fd5de0877aed9", +"17d0b4baad6ac2f92e034501c9e240e84354a051edcbc736a9e747e4aa0f54dd", +"95a61bca0187840a674ee36689856c45e6a2ca151396e0b58608aa90008127d2", +"e8fc54f536126d2ff772bb97456894abf57af592186a706130dca4bd733100e0", +"a2c0d75488be52e1d8fb855e1669d18323ec7a2abc11357fdb76d652250a21cb", +"b73e65d50a91efc5f7c9c42145569ac42ff083cf5bf1ac68682a62fab5667338", +"0a49d038b029e22177869a428fb54078592ae26714a044e7cd8bd396a0d3a0cc", +"3ce2cd3a62b9e4a55ba93f1ae35c8a1719fee86167e555bbcdb61846801cf5a6", +"6a3aa44c335c90912fe8e00ca4c856342bedb5dc63433554db6afa989a0102c5", +"f081702768fbfac76f1aabee5287f3e42bac3e6f45fba8fff894c2711e400e2a", +"5cf5e45f9a31cc25b43b7c090a16f9e2c6775fadbe1c7c8602fc35766bbdd694", +"8ebf6fb79e91b4b5336dcc829225d877eb09052d011fb5f9c0f8a1394c9e4b14", +"a6604e4d3e37b514e61158b50f7e72014fc3a16902e7dde50aa810205a601c46", +"7cb92ecdb39820505d2d5fac724fa131437a4e94744f962c5b232e9701347cae", +"1fe9bf4d6be3f75394e0e9243ef224fb750caec44ad73e866328683301536e9b", +"8b64d85ae3550441fcd09a3965c80540750f8ad41619b9ec7c0cc3e5b2eeb6bd", +"a3b7fa922316a76ecb70c628db1af413f149e734f6e1784446b9bfed574c945a", +"a4c7349ee82054614b3f572ebc6407b404a141ac40e39f014d433c6e4546a54a", +"25c4f8f8cf15c2476d11508794f3399d685afa08302a4281e3659fb0059e056a", +"75014b0b913c56ddcccac98821d223a6e259d5284e05ce7c19cbf4ac7913f84a", +"495898ce409e3dc275697134cdb3e2384cb75d97762b4a2535f7e197c7b8407c", +"2cefda610288c15f4587fdc44e05b71fbd71e9e5eaf8a2380e3d96e62a75f6d2", +"2b04dca720c5f72999836cc47ae147bb163f9aaf548b0c7b501ce3bccee1894f", +"cb0f3a524f1e8ec32ffc429c56c85823e1203538ddcc66f0a7c16dca82f193f7", +"7bfdb3aa0787da35b2ec4cd4887da0c2ed737c8028bf763cb735faffec2ecab6", +"df4926d13709bb88e23192eeca032937599ab4b8ef93984aa618eccf1f4cb06a", +"c5120cb73c4a74740d5d91597bf4ef88eff7af63332e38fff096dc6221676ffc", +"b0ba266cf29a520f0466ff7c12a85660f37f9716492b52084266147e26a85dbe", +"945f624a68af13879d803fb82fd3be8e3993e23c4c638909d2fa11c7aebffa67", +"c343eaa9447bc1c2e9a67ba0d54c84eb8c3fa2f6b92e3712381edc9d4ee29934", +"51a24a134668cffc7c27d9ced39689cc4e11d98438e624ac74447bb11c5aaadc", +"19fce64044d9a2506755184188b15a7a486eb95d4c569c88b130695a0de64add", +"42a50f8504dd68bb04c121a3db55eb2fd26db0bbff514cbc4e6f01fc3ffd6a69", +"e62129c818eac960cb7629680dc576d6896b01a4ea407609e9b7254a2abadda4", +"2e1d2933de238ffb0690fd7a0124c6964a28e082c786b9fd6c072844fe8a10b7", +"0d5bfe9dc1d6db18e245e8e576aa0f9c3ce7429f0f2d75249c25dee7f08ab090", +"2f2f35b41f1f8824144dffafa8dcfea47eb627c82cd8b4a425d29e3e93468336", +"89d88aa4701fc8afe67b6871c3cae0070c79900a4c0ca2eeb0aa11d6b453dd0f", +"6ccbdb2d47dfeb2b644d180ce58c9f450ac4fab5ffe007daca501b859b4b51b6", +"2b49ce93d8c37d389416ec96fe8cbfaa5f39fb99d77d8c51b6f795a29a14b3d4", +"06a4a70caddaf2ee025abb13d41dadafe419da511f4c86f0d0f2b63fc439466d", +"980530a47282c1929419aa05e8838fc981be073c1a1e6d26815dec542e8586b1", +"c593af3879b48528e2cfeffa6c84ded459cb0bce5710cda5295881a1da285cc6", +"8c84a56cc66296a3599f325dbbdd268669b07a29ee29cecf1b7351be6cb050a9", +"4d225e2584fc6763f02e46ec15fdb6cdc44be16df5a0c9e87836288b822d5155", +"a02920445705cbc5bf6c6aebc4eebb029cee19b84a0443d3fd3bc34d3da35340", +"6e3591997e165ec7d5bcff87ab35d5df2fb09ec6c5f902524564ad16d81fd1a7", +"e6a8e262ddca89616873e66bcb0b7d8357975f613bc30f580b9bbd59eb203748", +"4e66e1201785bf8a51572163778b5811a1d2fae8edd03dc7201c6e645078554a", +"65026ddc0389973e0c1a0972b27af7847fd26356464c13fb1107c0643b1ac57f", +"f9d628737b27e0df9c982486c3d404fa76d46348545825ae42b2b34a2bb88239", +"84201fc071fe4e6cb27e0b23e03592206caa6d7ed04d92110d33b25ecbdb2135", +"8ae8e6a0fa895f0bd1c8f131ea1b4f37c9cb7bb2f4a8851b83ddfdc5bb056af4", +"8971146d485d0f4d74624e51b16aa8adf0ae3d1a5ab999e76b8fa1e573d6de8e", +"ee1d255237524e3cfe94ebfc3c231528c9a015a422c2fe1a5758ecc4b63a6adf", +"e054880c270db5c535f00dbe8c19a7a3303e1a5e4f2e16a00f90bb46b6d5c80a", +"c8965070a1b123043ad0ec9b91a6f8973965354933ca5552d503c37c463ea75f", +"25280625fb975446ee1bccf778933b3036e146ef59991fbe79d9613b0c9d5839", +"3dc4a2e8b72094538eb0c76004b52fd0932326ce0adfa20685a213788b5af607", +"4187f49b0c11730fa866349fe81ff468460f786f4493c43b42a64e02799f45a5", +"65057a08d2fe86abeb4f708358307e242197ac864c0502bff550d2b8724869de", +"cf35e6b0f34356d9865c01a1bc17e7b8e0d74ee7025053ee96a226e3f3a75708", +"57d7b09ddf00c95b78ebeb7d5ec1242180379e225da9848835a2df213901b148", +"ff8209519db4c18f015cdd52200e7dbe27a0d8091df4c6e711b6192549f14e66", +"77e325d836081143a5915a55b92897c44f220ab07fe72cffd46b0c07a2d1629a", +"567ee35327374472acaa8b4223ec39d19d5b97e4bd168cc6ba25951c7b84af8b", +"769381ab213bcbe6778f4a54e9fe7cb7f713ee9e79c7df9d413f89d1db0a45e4", +"4ae07b4c93e54b18b220885e7e17de9759ff7dffe684c739d2e6969403baff7d", +"22592e2d1463064769e39ae6d68f5f23d6327c91aab1fde108b7f4a3e381b51d", +"77154f3f83739e811e7e30f017fe046a3b448a37a06563dfbd438583a6f92467", +"ef3385ffde53e9e7c510e24623cbc39037e96b6c29d2db0b4eba836411cd7562", +"ae4283a8eac969535e1beaee23a1ffef309b08df04c3eac5f24f3385958a67c8", +"7e898c156c1553b17ce61bcccfed53064f5b4d25eaa55bacb1880fe18e3856d3", +"2f9d0ef3524be3779edede8fa49490facf3063714c7fdd37d9d35e9bf152f225", +"22ea615c9f50f226d81a21ce7655083124a20de4df222fb2335884637e616692", +"fa928cdadefddf0aeb7027e05d97be567352fb973a357e6054bc06454a88c917", +"084a6f432e44691e53abc81c93d9076c621f7a0a897662c623cfa8ced383512c", +"0e10d74b77aca5d329c73ac35e86d3df933f178851b7b9f7885ae4a703f0d99f", +"046e76facdd7058a348c93743db1b893a8fa5335975ec840c6f1d8112d387368", +"76e3c979f1f7d6551bcca4bb79ea4e4c7d5d61da58cba3ba8d47fd7b10e35806", +"aeaa1c3944e465fbc1ee7018797af9fdf568f2a16446d8b9fb6cd6dab0510f1e", +"4cd076a59bdf4fd1b24f04f17710f8777ccc93c0f5bb4d5076a68697bfb506d6", +"a4c788c0ba40d168e1d7769d56a59f4bdbb4fdaa02cac655c764be9f037e0dd9", +"d6f20b6b07d64bec6b7c681da2f74160246a9552b3aa9852e48ca2153880a7a4", +"a301e110c14e1f6a4e721f9b65259b429230fdfd24218f447107d0a26980af6c", +"148d7c0ae548a910ab82eacc32043d423cfc4e63fed29f613fb3a76001b83a45", +"8a26a0852b5240947925b864d91db59995cbf1158ad0189e108452c5e96bb477", +"4a50886835d6d59d07f28d08d650b3a691ecd74fba087c182011bb969e01d01c", +"308a547d7401df528e54ab2b8e26ebabc47299170931fd94577ba746a4113458", +"d87cd14baeebd2e7939c0a019fa3ee49122422b900447e5f1246c77d9803e4c9", +"bef93e4811c3f7133825033ace72adb52d34c3e1447ee6c589a813f4c1fd0d4b", +"4243eafaa10312ab96393bbb430b524cfebbc09c25fe0b6e0012d532d7191aff", +"bcd3a68b65802f798620403e0fe914ffe055ce7bb9c429aeb6790042479043bc", +"ea9daad58c1604d34363e41da7b8a54ae3d7f174c3e94fe07486212a23111d8f", +"a92f169cc1222944fcd05277b80b51cb3d940657d7e7fb91244fcc920601d5da", +"ff172907aaae2527caaa32041be1b2362dd3794f364285655441265109aa2a7e", +"50d96c43c78ee35c96954ecb5e72be8d0142f6810c928a3c57e7590458f5b16e", +"e7c6c5e694911d7ca9ceb5fd0787b66dc276e085cbacc781a04bf2dc638eec2b", +"23cbb4c14577555a88da040cd36c94d7b15896227d79f605fefca77bfb39bb9e", +"8ffa91f2fe706360ef61524a7070376f76336ecedd9cdda18757db84fd09c3cd", +"9af161f38a18c09a8c50cf61b3c9099c52242b4c4e0aec881bb64a198df24571", +"63b0ed06b2d3a6eb78e176591abc3548a027a0e7f261d1398516925f5944807d", +"77b639918e7ff45cef74e7ccddde43a1dd1ca35e2aafbba0f7c673dbd5e77359", +"1cc42019721917c8a767a7a067c1ba526e892ab20446b86d1d85502a05f113d2", +"f8a07aed132481f0938ad9712373c71134fe8f334e26ad08f4546b0560624258", +"173a85542a577f34d1731310d94a58e0ecbb7538c84ba021c18aa780d651c7f0", +"3069bf4ce43f0c00c049bc0ff2e33a17a4825c1a2c16b770e5b0819d89afe260", +"c9eb1a33a769b0886fab721218380c0600dfc4931cbd15f993767e19382b755c", +"1b416c76516b3e7d4fce69d49992a298596d9c9c8b33925272a3c643b5b7cdff", +"3a54dd2f150e2851172b93672c7129ef911a7d0c5290bac06e45e22b1dcd99aa", +"994fbb34a1f726e620fc60ee59118b9f4ea6bd0d726d3ebb1c65315d2ee84d02", +"1197360c30877fe7eac48939bf12b7df8d5a4b2a630c335b498ba6f5881909eb", +"985268f865d3d9794459eac85a5007eee037e0a9c0f105429b01c836cc2a9bfd", +"ec1bf337ed22eeb9fba1abdad6468b2b07895d5d22d4072955f0d2d197b98803", +"a7520e9925bce7700b42c7773aed3dfae56f8b6de5bdf63ae03408234499a54b", +"424ddf9a4ecf90863639ea953b32a44fed84b68d92bf1beaa46e3c3564e09c1c", +"776595c53a1a39c135c9078b917b4824b7c4627ffbd54aefc2102c5d27db5b30", +"78e4320513ccb7ce87d96a37bcada9c52fe402e96165afdbbe5c40ab2517b246", +"1b6a7b3129a41d483a97023d40aab110c4c15e5adef7a7e63dd8d7bff12b3f0d", +"7452570f63ee32f760fffc766d2594a6421bc3ca12c021fef9a18f01360c9de0", +"7d817d4066a14b7bcfbf6d826c30ef103dc9af9a8c07b79fed38e47cd258411d", +"537c5fb3a61cf78d62951194a1c039402c146eb34eeee1a08771bf178fc8bff5", +"37fa08f9934ddfcb4cb6995ced98cb4c6862986b673bd9c031947a9655e2d6de", +"0039a6f53f95f41589a354498db454a3f9db24ba9dbe6132556c2de830a40772", +"f0a9b97a66cca15db4f67b3fc02cc9a30d346adc0182a1a25cebb6a60d04a7c0", +"a643772e10c7370a9e2c20d5416908b6c281fb4e83e15af6068db64db2892197", +"70367ddcf0dc8453cdf3e94ee436a6bcbb85460d1281ecc1da3e41276c139082", +"6b9beb2916c456bdba6c4d260197ca497838e40c83551e87215793214e9e9461", +"550d1c7eac671fb95ce42cf31e42b9fbcf2cdfadc3556096511f2e06e946cff7", +"885b538b492494e20bdb1cf146226229d29aae3c94942faaa8fe7fe84295fd30", +"bf5c65cee6fa18358d378acbd848197b59caace9fac024a19b5bffd61b82d022", +"96c4dafb543012b8d4a1cbd9d5cce2ca609ccb914fe8c3182f932e066c977595", +"4c640a933f9a715b795b2b252e9a1504164057207a4ddb6874b020a7b04d64c6", +"a249cb965c75d2ef7bea59f563efbbea2e04236fb153f35c35d00185ecf3ed90", +"557c7a173653c6f9d029a9e3ed3ae35e7693582da45d2156f429a41f717267b9", +"a4efc4a341c5935037a660df5048ae26dabde87b91bfb2f4aeff2bd95ca1d3ee", +"4fac49f0ef846f5f3365b6e8e7aaad05e0d62d92d29229c751a7789703a559e5", +"4a4d567a38a88f7b25b52debab462888b42527b9f2c780bd73b9caf9971b3648", +"293e6998f29681dd306c7226d837bee850c75c12eed2b0a95416433b3e4eaace", +"f23d4a1e8a337443d921b51c34c9de6fe12c20c7b064f3ac11cbef74e9534d73", +"674a0b96abee633e9de9e0115f1684d9d29193f97070f8cb495f5eb3efa0cd5a", +"89e1a2dc4491c787b59afc4fcce6683038bea98fb0a7e1db598c5eb775c20085", +"0ae92afdd505bf969d05a70acab21640e31983c6d33bb0ca34e2831d7d5163a9", +"995873a15b6533d7dbd913f7594e5e07a70b0b3e048cad24d37594982468af77", +"a9f07d8a8559503751c051bb8ccd8f9a8c73637b6d01a18eb71dc0e997cd1f19", +"024b79a0c3de513732c0735048333440cde978628fd6e7eb55f0beecea1b3db1", +"13daac68642862161fe904098001c323a4029aec7a560c808b1b37951ab4f891", +"18c52d7d20358f70be2f4261e541cb645a3071bd069201ee7122f12fa42baf95", +"8deefe00c6109d149f44e14ec9bf524d0f56e39e7555f486dbffb711962f5a88", +"78e0c64fcacf9f33f0c72ebaf333b71a0c8dbe4bc6e956f8b58db190d33c387e", +"483122a32eac4d7db5e051792243ef0baf140b960d863bdad66a0b13697a3583", +"a4ff53c76b2299cd6b45f6e83e76b720ed688054936568774480b11ac3de77bf", +"2ff342a5e7367c65aa623357649905e04cb48aa40aadc450cde961ce8b99065e", +"94fbfef25f116b423f810f60b8d4c6f494aba8d8b7c90ea150346cc88818e9fa", +"b36a8ae301bfa2709ffa100035206c2b67dea71640bfd734bd4be52f3875999d", +"3abb182d9732a3e46c5786ad8ebe9a744f32787c8b2cfdc1158e017b245aded3", +"27a4da1cbcfd3f3b61e4cd464170362c36ed48ac6bb687268c347a3e0558f929", +"043d26be9a82e267d723037d50a799b98bce4ea638c4282b7ba31dcb3b1da376", +"87205f407511c09aa210c7a394129ba1213a78fa99ab66e9a0c79de3664b96a3", +"96038a4907310d8f239d2746cab6b1c2f90e9d8f76c1d708d9a2f4c48bdc1b96", +"8c0f50c0302a6ccd153d7dc1dc3d258a6b998d459e21147029b96a65fab4d733", +"4c46dcf5b152eaff56b3056aa6df53470330ce4f051bcbba72f1be0452d537b6", +"d24aacf92286652b3fb412fe4b4269fde537d732fb8ee6dd5149a67876ecb85d", +"54d1be5d4f90f64ed1834801b11f09847ac7a4cc70d8b8ff14efb81d722aef13", +"006cee82da34170cdc93bd8de6fb40598b0bf3915afd84cd1f5a7563ec9f9e0b", +"4ebcd032c8b58c130daa931c7cf27bd8c39be6ebad22092d0b51e8fc861e3bb4", +"09c39fb7c4d8751329c8b30c17f7f0e22864ec401e68b37656757b1f4ada0414", +"1329f77bd96898ef4ebbaa0031f94a8f79d856bba8d09b42443b31ca3721ea7a", +"a1f5d621cf61d8900bff0aa7111eaa1f55a3718e6a5e9673babff08565ce5eaf", +"59e6c30bd72a3104d733fb2d197dc911f020b6e096d461b0da9af0a32c0f33f1", +"42118bb7af6b7eb753bc2f7ca3374728247a5f628075e8dece2509022015b6f0", +"4f7326149cece1d12653988308f4e6d57f499e3da26e005d203b755b7366dfbb", +"39b85e23bbe56f50a0e18baed104ecae91ec0b52b83571f930b050d3fc096fea", +"d553059b2d5b758df206c298562b6f635914f0e56e023fef23036638908aae93", +"0b63afcbca7fbfae9c7684de73e10266e929e50883a1017429400b6c54694b3c", +"b70a6f0bb2144e125fd304f8438712b9c899e78a66f54c5061c8c373294ca128", +"9ff8df743e393c3a8d629aace03e49396b04d811f6a645f08b8c216a8a6231f1", +"25c50c4a4938207dedfb73c603539c602c1391581fc7c4554bfd7cb186fd3b25", +"e4cc146d80927fb85e975684236915db7c7e71d5ad0e05b595d03527909f8b15", +"0435c719db432496667c9aa78a0ad6b05a95f31557cc9977799b73c4a4f486e9", +"c2968c8a276978bcac607701f1415f85b526d0bea59d85e6c875f33bfbfa478f", +"e5bbf27d6e7b58931fbd0b218d23784a499b7aa7fa4511f958ee9dede4016457", +"db6fbfaee3dd87a4e8b7e363c3a9f8d0f96e956af9f26178168da19577f246f2", +"dba4c1ac290b4e99f72abdaaa626b79af1060ae47d3437730d15b24dd949cd0d", +"892f24e55ef317cd538150fb8ae35db296f4a7547159022f239ebaf5b94111fe", +"5f93716704a9b26815597f15ec07886b31219dc669140875e366b03eafb2204f", +"779dba5cbe7df2a30aa2c5fb6ea01e4d138ae5f0daaba524566bd3591b9503d7", +"f67c58c8bc3a61f8d86f023850d99185af84c2b42e8d30ade472e10893ec09ac", +"b84be88981bdbdca09c94d8b037be0aaa59861965a097e2ecde9c4e23ac154c9", +"b610ac251c75fc94046058b1ee287bfa2476585f06a285b7c5b5e21041be8d2e", +"ab366ed7b3ce603f5e72887afe64146b949d22c6c94d4239214a79ca1e40bad0", +"243fb0da5503ec568349a05b2d531f27a90479a5fd2028552750758f7194be8f", +"73f3b8544acb210f385835c189e63d9c71f9c79623341aa12320f23a88c40013", +"da0d7b63bea7141f6bda10bf9392df5656633c422868db496f26edf35d3af889", +"10bb3342035a4fb5f253460b243bae73af1829d071faa9f03310ff31d06f2586", +"0925c6e0b5656de44458df7dbe81592d39a99fe46c15f4b218bf2473d536f890", +"5c77277f4d6ad20b03a69d2ab042ecc9d56da536ff00ea5a9d49829b4a817a89", +"ee80a3b899c5019df6aec80214bca901edf7d6baa83e4e74c7d254bde9f41297", +"e5489b5f9c859fb0173eda877a3f539a33b0fdecb066bbf4b5c7df9cf6622ad8", +"18d4ac91dd3ce5830735cb15ce770e3de84d2de3a5412dd094338bddefb85f48", +"90577ee4cd8e4e94b2c0e63d8006e2fe343bb6ad78e6f071f9e90f9e54c7406a", +"51841ebe84bbcd25ac7b7c9e42a60834f1af0be57294e26d52a5daec7232b9b3", +"103f5a7e6832f4f50637116936a0e93157835929ab284d31011a62494139d33b", +"70bae46b64c9e7a79ae27da721b58938b729475c5da25203a0f89861faca0326", +"07494dbec82039c765bdb51243cdf279247136711c34762c5bfb938919cd66c2", +"b3f42fb4e188f1d4cc59338a21760f538cea557c3be1f0535589d451a1db49f6", +"422d2ad4bfecc31958abf7bc443f67d4d9cea0cc98a2caa9e1a008011ac8d0ad", +"904d039ce2f3825ae1fee72b75c39c5cd26247bf887a3f5378f57a49c13d8875", +"c0e786b6e06e8f9efd9ca36398a96d5a11e972dd3015a418f6042a2977f8ccea", +"2e9a28d72f9cb208297d12289ca3596063329e9e1090394db4240f189862c330", +"fbbc71204ca35eadb4212ecdc4c880adaebf3de31e489612483431d436bec3ff", +"d384e942183a61829d0c99d84c87475eefcdf6277b72989655007493e4bf021c", +"f4e2406276e8efb0f3040909b6d77214783576050ae3ff5dc0bfea80151c9b50", +"0148a6661ca58b79ecac28596049ac006b7f915f28f317fc7ceb1414b932f66d", +"eaef93245209bc0bbbb806d6131fa32007e4f027c71ad6299ad0130e595883ff", +"81b3b214a11d78d72f44384710db6c0669b0389b8a3ff57483c9ae0bfb48a0cd", +"f8667dc1ca86a5306b3decda77deba1cce1197235f46be15876b8c34f954de6f", +"8b67051186fc1bb6d3a879f9696621d0352af7abfa6278b1d2c70b6c5f042d3d", +"4484e62b918faab6e22f75d3bbb623da98e9179fb38c25b7a2bdcdcef80bef5b", +"7ac9a1a7a96e92108c1c3f755f089211b504c56b25ed59340cf57827f397b9e2", +"765bba9b3f1c0475c296307849ecac8c2b82a6c84c1f8571d167845b453bf7cb", +"b6a5d62a4ba3c02a520e3ebd7ab715dd111b3da917894bd807f3ed134f9f5e30", +"039735c85575d6eb6eec70c1ad85966e7eef8319d17e6e8e3b1043d62ca59e47", +"8ffde53f5b878f9d8d6da369013adbdc2a290ad237b5fe6d2870f27b033622ef", +"88aa4846f29a1034655fb2aba40de2d9471248755ddce97cb3cf90c937488295", +"6356aa747802299c61f0674fb2deea7674303669df3fd051ca2e2cd36ef977cd", +"1ae1d54c53aa2010e7e172cd736b28cfba30346c605e74bbfc58114eb1f9f1aa", +"d030eb85299af43118e703a1d61a6915432289734d164f56d7c08d44351b05b2", +"392334c51eb432b855b42312f22809650738d052817c386b2d977c3dff73349c", +"ed555082565d8a21d2f50371dd58abac5f0ad5c234a86166e58a23e8fb5b2e11", +"fe1e001716914fec7c17357fab233efb81cc67772a559dfad42174174522af25", +"4c5092439b4c117a91a7c7ca5d6dd79ce365f664f4b61060b99315947da64484", +"6a2cb03353b8b417fe11f7fae8d01fc51d456ae99383ce76980981b6ee8a7f60", +"7dc8fe0ef386d661da6b0d23859c2499cd48fe272e6a278d855d1cd10dccf1fe", +"3ca8440bb6377ebc8b0951930accd582ce631a72dc10f7a83e1b0ad6e2bef840", +"5734b051fd53812970098d5a73ea301409d287d40d7100414e59a1e0aea9fade", +"bd5aa955fae94eb5dab26422f9c1d4faff1d4511b8814bd3b1ec4fd64797964b", +"86e67d2d97850bce7595ed9f6468623152e8af46953208ecdff8bc0296a87acb", +"e94fe8611b35caa09bc22bd45a9c11b60639ae91098d58df3c4fe637846f94c9", +"fd5dbb0cc1793fde82bf2549c8c71b4977957e1d42aa03b9f4f61f487d8ee790", +"899e497185cc1803ffd86cb066253c1a02a2f2cb1d1638bf11bda2931719fcf9", +"3444e08bc2d09b93b262075de09ff0c8777251e2d4526c979e205bb039a8a047", +"70c203260f3eaa2e318f8567d127a538f1a0e30a1f36cf0f37446b8f0f666d71", +"299ffb8495dbd031e723d71e7a9e41e3aa64ca16ad470740efefb023c32525f5", +"782091cc73972341edfdd20da4bb9935148a790149022768397066167efd0e81", +"c006120f52952ec6857cd2605b1f0eb06f2b5f2c8d70ed2792f2995a00ea03d4", +"c9890df20d3227e6f78dc9fe43b440a1111fd2ab46e11cb935f2e9a634614cdb", +"20269504b5b1295b68a04bbb152356c2ab5232e9c9f6f89f7a0b979ff7a692c0", +"d647901bbe0cfcd4cef8fc87b6925ae88e9f7e501441077a47c7006a93bae1a7", +"4a86991d9eb0d8533a92cf5e135b59f0d64bbe4e66ae7382a16c74714d3ee33b", +"e8e6d2499c8afba322ed633e63f475e5c9a1ab53fdf55e7ed008c8a1bf769114", +"4a576c2352cd2283e44e2ff8946f60c47974d3f7116908b2f3af89328d0b0497", +"3ceaf3c79dec614ea2d5268980f8512e14e2f9fa6b641b2b6cabb1b483cc3a3f", +"86c5aac0592d3316df8ac7d37652a2ff9e95ca3cf670bfde9f26ca47d9b17498", +"52cc6b33e70cdba0cd2868d8f90e1d8e15c203f5a536945b9e13997af7ed0d21", +"13b95d31a11056e9cab30f3dfdad49c12762a8c095c94b59bb688a79041a28b1", +"a6a0d82d9cb5d8e8954090692cdaf9a225bf3000155fcf645c94af63a5969cf1", +"927c98c0b02b2fce276b058cff33de28ea737685992458fd9329259d5c410df8", +"409bed8556bcc8916ca855412fb85f302955a96b2d6400fc0498e1878faccec9", +"bc67ef729905f32b6ccae01625a43757b88e5ccca2c6a9d02a553808977aa165", +"9da69e47cdc1615199ec10ef9057c1debabc678ad446228268af721ab7ca6b6b", +"0be531f90b7f03eacfde082f494050039b1e81a651fa114b3cea053b3ce7b3aa", +"276a064030948a10086df5354a0dccf4c731a60ba801872512b25dc58667019b", +"a5f39ae2623d4ca3c8c2625b755315d35570bbc8d9a49bc0331d3bfc0a4487f8", +"004947e806c5afa74ea4b64de0bfe63bb7488c2c3e4e5d4d5d6c8403d16de46a", +"d4d22264ff277176440adeef08313a40ccbef0a6bd63e7e7f13e78ac21a8d21b", +"95026ba2ff9ce39bb5267bdc0a1e19cfaacdae54c0375199448b843c6701209e", +"5741da1da00b6932d0a2639459b3e1ddceb69578036bfc9eae9866f30ff956d1", +"2ebec5ad12c46e226578c88154097977cdd7384df4b47c98ae1b5a73009b8d0c", +"792257445f2f59fcb33e459e1c83d970601272deee376401ef21b21edbe5d4a0", +"05488e148ac6bae282b40e219778317eeb59e30e320f730e6dd3f65d60402d58", +"eba5693b48342a6fcd7be9393c343d2ae2841ca22d0bf131fac48eb44aa2e32d", +"a29ce83ebb869439cca17e8ab0a2c2c46ed4ddef68af71b09f2da49d45ce34f7", +"5cbceece9337efe76e3636b3559fa676662b5c6ccde4c9467b32f6856725b35d", +"382f52c2606d0b2c9889aab51983513faaa3028f905b09f144cf8d4cabd922d5", +"b6c72dacafe9d577985e4a39acb93ed53cbc3d9292d772628856a2c26e7c8fed", +"502bb6f07a4f9c10c4bd7d6a17336feb8f0274c0672240925e64b76830ef88ed", +"e56188a1ac89937ae99571ddefcdaf0bd0f05fe5d83c6c44da735a1e8520de25", +"9a127620b355b8175170213042a6f9b4df3aa1bef4046f6be73e81f6f2694006", +"44b7955116fe11a335928193273b9477568cddc61bbb8ceb2bed56a26df5465c", +"c05181923a92a8f493a58d3bfdcc3a7dc5fb3f4a3d743ed8aa71506c4e771ca9", +"57dd36f69b16e1f54b77755753c486ec5423497412b2f65a09865a4bb356430b", +"2c3249d410841a41599d42d5bb9016b4d9dc94fec46e2c2398f9e13df8cc2eb2", +"f39163564331cec9c05688788e25096cd95ab495038c6bb203244a2175dbe08a", +"1f25052d80d02257a4bf3a41627f392603f978112ae090247477d053d2538a60", +"11276d9cff6e97e4dc70ed5e45f8361ebb240d5f208a8aad9d807da3094a0d65", +"e0d9757d4d88ae3770d9670b0122aabf58dfbbeb2efee48f4e952056a8281604", +"f65d26516ce7873375160d2a022f43b8f4f9c5e743ae742193175837b1dd05c8", +"ce5182f11629e431c827bf6a72ba0974d6510ed12cde60901ce079f4f666ca27", +"c22aeff2e48460b609c1d0623af2e8b484de0fb8beaba8ad374e235ccc70c8af", +"9daa68967cf3ecf5dd41c4218131afa3af4a46b3e40ac3088b98623d6dc31dac", +"5be9539feb6c45a0d5ae93da7090bf56ecfdc0e5a9073b52f8c2a3d0d4c63b6b", +"08bfd9b83964375d2b206144f38ea9a5e2965ca00425630ab795c9196b3b758e", +"b249d5c3930ad24a6c65cb9b2c6674ba6dd98dc78325e582cbded46de9211dba", +"9b5ede6660da2f6ea9e1aaf5887ecf6b5b0aed9b7852531d2f83f07b5f914842", +"9e9769201a31a3b2f4b6c856cd49c34a97b61185d487fe64c81bb08a384b82d4", +"5b4a436b40643465f6ac328f9b3f3e750b4b3907dc86f63298bd0062091e65a5", +"9312a0568fe96de2762c9222832c91870b45fd07f0399281355fcd32e3cd562d", +"65109c0b4e4e843d8b0e27b429a78d1277ac999af71ef7c31952e45b0f747b2d", +"9652168767a9183578b8ef33391e87a3a73f5a711eaae60e4892501c7a9604f7", +"5400a84b9c5b7fefbba9d651c9447e4517024a6d724befea78c37d775bfaef2b", +"b72b35e22ef707071ac43ae4e7eafcf027eef76dd351bcf84ee0f749e8bc0bad", +"85b968dadb55c3130efb9ae5da3a0e4561af5e888b18acce732587546477ff80", +"0312433c75e12e46f408b1d2cfc99f044491520546b323876637195204b89fad", +"90ff83f08b27f364e5e046ca664c562525b0a2f2cade2992a3c8d1430280d248", +"fe1ce018c06b424b7941c02c4740ec944c049c239d9c2418d626931d692af0bb", +"176a88ba2a00fc943be41ca69c1d1346cc780f550bdab1af72f740a1ad7eb676", +"372e95302e54cdbbc6221aa066d3f7bfc08cde388b6e4cf4a448ee180322518b", +"f932a6f921ea2c1e7d920be5667f963c1dda94b976f1dc0ccaaad5af56750330", +"3af4e655bbaabcbdff639791935101eb7c6ed0ce3e0141d09896d01ac9a440ed", +"3e451bff44a2bd3aa3e583daf6da2f8047563741ce48128178c3ca989453a025", +"80526d0d743bf15b5e34ff6cd0e53a8c2579a726a576b0c24033e2408bdd9a61", +"778568f790486a29c250a45d601120f30a83302e596e402e991c2fd97f196e60", +"fe804ade979ba0e485efd3e3b965b8d9bb283bbefa6be9acd0f779a23e605fb1", +"6eb31cfe7278617d86f08905bbe4b2d59bcb078776d59c71cae092d9e5334b66", +"9fe84c0675db00fb56ec271b1756ffe1e28295aee61b69f85ef51606241a43a1", +"3de6c1e1fed587f97bbb1f9800091b6b18c130a0dc2a50eba638ee4e27dd4df9", +"55da5aeda4081085ebff1c0a100afc845234cd25691458601af4ebfed0ef6d10", +"b17f07860b0c14e2bc6d34e195627d0dc4adece75d134042e84603cd96939578", +"87a018d99d9c8f83fdb33c408a5d68c532eb9c550878994942cd11a885f6b636", +"b8ee77979c6256e943f9587b883791acbb0f58b61f8b538b84332c1508cb4cf3", +"1c9c35b8e8fe7027321497991a3922626ffe6b452c86c7e49af00aa59718acb8", +"cd208b1e5569693c3491bc25684eda7a799d22511ffadbeb8bb493fbd9ced118", +"bc5aa0be6822604bdc51681f79f79979c96707959745cc7656c0755315220a31", +"b9d8869bea54606df91b216179ee99888556691e0874effa3858fb408c7d1b9a", +"1207676a97827ab6bd33503c3488a5478fa3b12263c1289c0718abc6fadb6353", +"eeb2f90835be6a78b974e5604c1bbbdfafac2975ccf860dc6e29096a92a49629", +"95405acbea414ddc6a105054c7278074fb1adec8be296b81a3ff8c7130c49153", +"7cdf9102b615c7af4e32446d39d7ce00fa28a7be7223c801a9d13ca426d98451", +"3a5c864f998db222264b6341022f8241ac719abc3ab2ce6b55d82e6ed7bdd6d1", +"7af1720162fb4836acb1b664ee3d1e5a2e62dc4f176ae0787bad6bf10b4ad177", +"a6e80cb0a5d7df4f4a92459a75c69e9f728d722d5263c8ed786565475064610b", +"0a16f8fd00064bf6987be8bf352ba6e7f7dd9fafc5cb3211e98895d637fefff9", +"6c1cbf22ee54a5561da18b7602a2f48581bcf231f23b674a73c4b705f3d4d6bf", +"ed178aebd26b682900e990bf47e155bba9d9da9c7b8102748a5251b85113cf8a", +"41a2ee43af09ca218404987ee75ee5af9078fdbf25727b153c6b2f3f5dff99f6", +"10c5dd0d943008f61df52ba020e877c3d50edc5a13249ba3cdef75cefa7b8099", +"6cc0f3a4e4f769742030dcf6c863fb2055058778504b4a0162e0f56fca95dcbe", +"54049d94ca55f9629a1f77684d4223ccf573694d996abac505f3ce8f30a99a2f", +"835d922ab34992eb523ae79468800af340042e5ca31e3c374397f9a9f2dd78db", +"3eacddd6205d57c2d2ab08091cc2400d531cb3a68db09c288bf04dabfd242c98", +"32e2cb839595bef0aefd3b878ebf3932e7aab36d7240551ec3337de3ec02f92b", +"3e30527e6999ac26cdc19d03e89804653f1b9223c96a5298885b17e396f9725f", +"141655b0e4239480afbdc12874e642ffeb94da72536ee4b8bace750820b26a06", +"90950c341b189be5715a1b6865dc1d345c3df78f5a6f4d170c3ddf687bd50ada", +"2f0ada73e7579b6430f45bb5d709225d7693ed037777eadb615845bf42dbc487", +"d656c247d39629aabb160f0f7ef27137c9a38deadd6fa546b4138280ec5a2a6a", +"104536329a17c0fc991d49ca92f70a3a783594759e644fc53d68b3e963439ddf", +"34323ee50be733dbc8b6c941e309146e9eb80ef19f150ba691d79f43596f45df", +"58bc766d3c2b53b2ba809ea324a118f3462fc9b31d6b0c84b59a0ffa57b95229", +"9b987bef1945cda4966dad88b436d6c2018128362221aafc8db5b68ee41f053d", +"3839bb72ce2c1a0541749bf66432f8caf06163b2b1323b95dc11260c29075da4", +"5c28d6c7f9473bdd97b15fb6338af8ab4563e96006f8986162945d26101e27bc", +"05c6c3956d29471641a64c742bc20b0df2f0433967a93551b3b44be45bb1940d", +"81dc454d2a424048b4c525a04c22f2eaca189e1ac8babc9aa4c016657f3736ad", +"30be3e9123203d31cca49bcf7be527507afea03be6b5405357d0be5e6d0218bc", +"d1e1bce3634ccf7d4cdf1253338818a413904af68d0d118e24f691bb3aea912d", +"88259a5412270ef7d7ad5e193bd1fb80116876572be64e5ed4ee0655a90663ea", +"da9f11369022a13bd75605a6565894a2797e2add93e4946996abbafb69173ba2", +"f7ae3a49a02a6ed20ac1285727df775c89cb5a173a97101707659f8ec4b862b9", +"5acc8d0d01a087bccf76d8b00e2e718cb89a851b9751334a4570a0e7a78c43a1", +"cc7dc394c5f7690fc13d6d8108d3267d1f8307be6647aa013b7c5ddc79bf3bf6", +"1e3853af76de2f01c704981d8471d960ec786e09e1dd4450af152a618d30bfcb", +"620cbe4ef4208fef238358325d26f8402f4ce0258bf8e8aeffa59dd1a97d7739", +"9e6312c96f4a73fd2ffbe01c1396afc6f03ea79a1f72f6e84816d2b260311c7e", +"307147f0bea68f85bc5e75feb9a44364b29d9e3bda55f2c7d64cd691d2d957d7", +"b017b7eb681c42459118baba989f0ec85658a5a1d849d7b906005632eb436b8e", +"0dc58ee231b90c98191e4eb12d0b713100393239c4e1423f56e5a9fbe9a47a46", +"f67a518f790b12caea7eb0f80d85798c9032fe177396289c8f5ee068b40df854", +"4bb76255eeb125d07cb4e0bacb58cd892ac500a35b7ccbfba464445c906e6b88", +"bffc60c5e5833f5296c97e64ecd889c8246fdcd9e3febb09ae2642ae7e582e3e", +"6d1642a12cffc513a6c1354f841e7476430d45cdc238b554411ebf5068baadb2", +"c1ec95bde4686e2705aab12440f1bbb594ed37e7fb3467f280ab6d4851009334", +"1997cacb8111f42e2759639249a6cc5cdbf8be60446156f1075f4a3c07ba318b", +"7563c40550811f0c61f1a4d3efc4d6bbaba8336b0c32866bcbde4356187972e9", +"7d2ec1dae96bab0f1d9e7c1f757c7e99b2ae358623a5dd38d1942f48ecef6c1a", +"a40bff04419e0198de71be67293d6098119f76bdc9111f56f45fd7774239a8c7", +"d884a04e05aa9d2c80274e32876234b7a512b9aa19d8cc38d31d5b8914b4294c", +"0c7694679a8a9c9e01797fe6d50b205f78bfa08c103a6a071f254f315202a472", +"fb010de922a325f102668f4857a8122af2f1d76d6017b242e6bade87919f897b", +"b6160f7709397675345b68a1086e69fed6101b5e88e1cfbf8dda98f9e5f47c2b", +"eccc616bccda53868b317cc2574238e06f132e1d2e8b7545ea36fc7fb6968506", +"0021ac86c40196fc165cd50a666c97c533e6597abb8fc58028df700a9f1b77c3", +"935c8162fb14a2e82ab4e3e2ba0d136506da3dabf939f6003fecbc34376945e7", +"b876f3b2fef63386a8e1dc0e303fb1af91d471fa80ec379cd065b27d2d79b691", +"d1e78f0babf79268778d6cf15aa5d366487bf62287ff001003a4ca8efb704d87", +"aae418af49fd87b79697c821b360f5468775f31e57f77f7d9ef65f3f734d1196", +"97677e4f0f3a121c7d76ca49534b4dfb4f5312da4509b1e4a0a569381c25d17c", +"85c71ef80a4a32009062a8448513560d9371308895026f04c1c2dd041277385b", +"80ca6ea4c6d68d687bfcdec1842981e3c40dd39f9a3586b80c6ac59af1949b69", +"90ce1468ac3a3da1147287f5e5b4b50af79c8c91d82c95d4e513b69c37ed3b50", +"5e92e851331bed72c2f071622f935b13bc546052ab6bf1d93c4bedb1874e4ef2", +"06df0c9ccb11887a2407f093d81f9d37ebb1dc147d982c99173e318c4c5b9733", +"aff9722a9867bda2a1820bdcbd0a4ba406024881df9c688d536924d4aad821d9", +"d876e87acc413203c999db930024566b868fea8d543410edfbfcdaf0a4752ec2", +"0121370b10dc8703b4e37714652e19613433d9ff559d2f8af0f12177ebb6e9e3", +"b732d6cc5e24f272d5a0edc54e3b631c09305156bee4fbf22d93985a93c4e9a9", +"d80ec2e12215c4de6bbee244173391c7af4a91ec9d2ef352ce9cfe3a565f9be5", +"a35f7e003d937932767c798c71602c5178ce5d0151fcfec2043787ab43ead2d9", +"06414dd9c015a9e9e6cd2cc81c7067b2ba81617cdd559e8b5b72697736e7693e", +"bbc3dba8d1aa63c9da777b6bfe06444bbc8df6411c60424fc30245439a432524", +"99563e2c59348f1cda5070499bee85ab8a72635aefcdda13c8454b83ebe010cf", +"c7c0cc6d879fac3a5774bca00cac0e8a56c52c930e51d16c8bb45ac3f8fba3e8", +"f3ba337c544ab0113acf18280f611f798883fdf11456104f648f364b74baa09e", +"56b9d1938f7155691e95d7af717e2d313b57c5f06382726f3690dc5e26508945", +"6350ef6541697b2e097886431cec8c2613dbc28b2536301edf7c734ab26bbe8a", +"4ea0b5fca7a36e4ed43aca9028da1e83b5cfa2fefee99590134972f51891e6db", +"76d69630d85ea660c6ef19c7e38e60b3d0478ec116f53a3f985746bdf054a931", +"d0251a830a057750356e4e64ece9fdd57e02ea8f059b4843798da841883c3767", +"9e8f6ebb1be6b4b917f9cb735c9c15b7afa94731d557cea26f5069bb612edab2", +"9fbdf4acc5e5d03551a3a959313a10f2a2c6bce51f890344138db1e9a654fea7", +"42b7511e313cdd893200a92d9a49a91a41321705737a6f0c2ec69e945244890c", +"b6f569eedce326d450d2c33045aa38049b62b21d6643cb1947d8bfa4a4e74d2d", +"d31ced8fc1e88c2bb412a77b38a5f9a7e9a373c3c951e72f8c50b0ce0f5fd9f2", +"fb2338e8d3f4199d6a7c9e09120cdfe3ef0d63d3d61e9efd019f59b224f176b0", +"be65e899c13d0f49ef5d4e6d5de7213078a5dcfb28b096bca4844047bf8f2b56", +"0d50a2c4ea199f506b763db1722e23e3a87d865be9200b8c1ba204c765a1bff5", +"a37221191dc8591155fed00abd44b436b3a666c7054314a11d16cea32650eecf", +"03108ea503040bfdaaa0a78e79ad8dc090013f1297a1731ac1095c0793166c66", +"1e93897df26f965d16de3666a7e5cea7c5d6a826e4a4465bec67c934ef0df98c", +"efe132b854f1deca46e3358d44aa81087c7f7d3a0e19790a5fe514b445e7951c", +"d2f2b08faa39f33ac1a17fcedbbd8ab38a11f9cf9669b3739381bfc1513f75f9", +"a8ac062730a37a6794828720d19ab603a81e10f3cbbc74ec2d224f0f84c86cce", +"3c1030220b6705236a917ebe0d4cac1e517d65f35f6a98a852b435475593b6d2", +"3e70de0b0cab2d104a2c2d5a68b277f03f6b94e1362aab6f3d9575d3361142b3", +"68d9901c3e1b60e588b5a162449195b9307a153766f2258f3cfa189ca4b5d618", +"2f4ef9708215278a83b4afc481739d72fb4a32e0ab1e1de798218940864811ca", +"bb2b7e13abe856b2738ae9fe86d4209ae723cbcacc3d1cab86a65e8badcfd59b", +"d1156e5752876584e17b45f887a826b8c3b9a72fd33e70805c9d8f3f03bf81bd", +"dfe2e943a84d5464481343783a02923e7d6da31ac7a4d6fb26ca5c3fafa55e57", +"27bb364c98a7970af301b85bfc1d924b71797d293c82aea7b85300b6d0a90fb6", +"bbe7ecadc4b7e7d30742a567a4ede550851055ac68f85df78ce665bb90c058d0", +"a57b44f9ab66a9cafd4c33c1a3a5419202e8eff3d357eae002039871e225b57a", +"91d10da010124b30d3e1994aa0bfa8fe270d8d1d127c3cdc7e33965e4b6667c4", +"952254b6117063f7d1621155751d0a28e459b73d8151ff3718f74cb4fb4ebfb0", +"da3e3409b821829d8f7ea7a3732129d3e25c0ce7285cebab3daf4d18b2e376ae", +"e442b167157b5a028dee9b2fb427411f6d5fb185e956d16644f90a64f42d14a5", +"46af7ed7dada9334b223d9d70e3eb6290979bc85f78e8617411e9503716b6a7b", +"e70e7a182eb3d7ca9fc62a2bad33611f93c221720a04d432821dd0b3da5469e0", +"5ae5be014ba43d0054e9e5a8028cef55ef765733fc42c58d91985c4a1a95c980", +"06531bf411cb31b669e4b6ce0354cfa2c2bff30d55c33553c6bf8a4da946e20d", +"91bba3a17be20c5687347a4614318c60880e9604b95cc37566dd1e05523e60e6", +"c97f8c6cf9105f50e403e596f45a15bff7101935345e9f77e9619607cf6a9d8f", +"b2a2e3361a1bcdb30cd08d9f9a3c735c613ab0213240c535356136f00e314cb8", +"61ac4a854c3f42511a623b93c460545229637798ec259ff4aa4e4cf339234282", +"3845d1bb27a66a678f219c001276cf8a34421dcdcc04329affffb1f487bdd6b6", +"0eea2721c97b3a58caa2487ba8b09fe90ab3ab33c1bcae0bf6bc6a7da76da830", +"daf4112917442679e670fb713dd7ce4447ce3a68d5b9494e40d274e03486e3ab", +"90c32b1af80079f25601acfa9357e8f5a4593f85b2b13da93b7bf4b64ec98bbc", +"5470967c00b7873c37abb69e8c634a97067287465fda9efff73f56150a1eb81a", +"e18f48c949baf9e2c257586a75e001b3e26e5e166146b87081310fa425e41bf7", +"b66b4e672529cf7011463cf77fb536dd379cc72ee6eb89884f9981cfd1ad87c9", +"4caa9ff4f81dd8ccf860e7cc2918e537160512bde802930d1380db5cc1d75da3", +"989ff46f81ac81a3b97b05f91c92806ab0fbc0cb6558e6ce69b38d6863d5fd12", +"44f395080a8268041b384496130cbce63e701d09c23e81e1aba24c45a5a5252c", +"6be7a5e610ad76a763eb69fd4eeecc546b1a5bdb539226a73d857a2469d5f0b9", +"6eef97c80d9e975c678a26044da564a51e5850b88630fbafb73204a5f7e34449", +"87a36b486397703f39846e521f2e1a54879a8b13d7cd70d4f8b416da6da2e500", +"b121ddd4c5dc762bc6c56935919db200c0624f8ba9ca6522a2ed14e9109cd5de", +"30969f10e6206a789f1ec6b7a122b534eb85e23087d29273ee1d9370ebb0a5a5", +"36474fde7d01b0014bd2de348e00862701254557e02b6b5bbaaf3b16f9b6208a", +"6d20e7c653c63cce9f26ff5c0b53a94da8cd427448a623d3d5302c830c49f165", +"76dc1a816583f375612b2c018f923b5d305f88ddeab7901f1861e65b934e7114", +"c26aacfab1d28b5285be0ed573ce035ea8f93a8f90d00d24b5271e8499354ed5", +"a9c865c893bdeb8c1c5968e0e475157aba5cfeb1814c020c70eb7846cb00f8ec", +"fdaf8b0286c66577d0f8636e37f341e166eb53cab6909f844d2921e75b1f1102", +"641b5ce7a7989b175c0bbae12d446a66bf955fe43c159a882fc041af9c88b8d0", +"46ca3110109a70f29e6cc81a11a9a9df55340fd99c22cd2073a745b1f8951e81", +"e00e397f1f27d1cf78a68252e1c4c8e491cfcb9c6d31dc40ab15a0aacaa7b12b", +"861227c9ba2fdd40974af9aa559d538e108defef7d8613cecdca65b10b6827af", +"46ae20730ec4b6064a25f5f23a35a230f73edbeb359e82f5c01a3eda8db11083", +"2aae23125a349be545cf1b1220b3f04ad5ac648de82b9bc62ca678a1fec7d5cb", +"a5254f0e7f9171b5bc979917ca87f6bc8419f71fdc3fb512f37965df3b9420a9", +"76dc5c0790e7273d3862741ab570682e42cfec9b101636b7d5df76b504bdabea", +"3160fe1c2f780ba53f2fa2b44a09de609936eed52435946b0139621bce5a19b2", +"461570b1d4894e0eaef5db795ec75726a0305808104be67e3022ac55eac6159c", +"e1fa9b3aeea63b028d47072444875f1efab1d845cb3e1b9c605c7ec430c0208e", +"72fe6219b3799d8ea6ebe5f35b171e189e750c27d17f722503c7840b71eea69f", +"d74ca0d0ec7978746d50c18364e958d7fae3c43f007f1ffbb7cf94d56036548e", +"005be5e0e9dbb7f0f5d792fa412ca86efe6c7d6c6e014b94d8ab8840db096178", +"dd54ab1ac66e9559f1a0bac4e763d37586f4f8d7c14aa0cb22486b6a76a1ad79", +"67542bd9593de095f47c9ce1a81614bf972b50de08b190b9f65f32c2433a16cc", +"b73474d1689027e4e65dd1bd0c6b0ac279101fe5f8a1f03820e11081c0bf2728", +"1491212fb92deb287f3e1804cec650828da91d5c86133d8ca7539cdc761ae0e9", +"5328b5afc54ecbc36d720317b4d27bf5ccd383f35074592ab3493c8d76ccda62", +"836b1602ab6e6c798cbba19e82bc22ea5887c64e01a317607f809e9c37ebf984", +"f2584170829e025e4682e9225cd63ae8b81674dd234f0dfed03f709f0dbad9b8", +"4ec75bc934da27bd34f45b6ba301404ba0d686381e37f46e37b305c37a10be35", +"38c79d5d70a78e7076a6a99c4650ec04dd72df1666ebde0f81b377bc2ae5bf3e", +"0ebd5b7a3cbce0720ebfb86c41832c91e947549aec4fe3e9bcffcde2d83f63a3", +"dd86ceffdcca7a745e868592cdc6c11266564f319ddc28d3614fdaee8e9550c5", +"e353c3a1cf90d8302d94052e3d902592a105928e4299c6b940d707020600f5fb", +"8aa951de6e9f5c1b616c392308e5d933c1c08f9c72c7fbe3f5dca11bce472154", +"ec033859d8eefd57e5cde2422d604c50bd33b07973d899f20a228ecfa6f62725", +"3b3743cbfbfd3a7fe8e8360c3be2c2ac30a0caa6f19ab6f4ad7488749bcaf3d7", +"098c2489bdbb5d728fd74327edf0303f0a2185bf5871b636ab9322cdfe7e73d5", +"2f01956bfdde1fcba2fd926ab53f910fd7c53690000ef871dd333c6aa38d4c01", +"5e1c18001fe4193833f8ed3cc40c82eda65b6d601252ae78150d559f806e39f4", +"32ea519c8a31c99b41507e8830a1007b477fb277aeac7c0e6414d4446d550bc0", +"23bae6ce16593f84a6822eb4204f7e7628d506d147daf7379b6ae122286f4dce", +"8b3873b02afbeb5df225e1fbc5efe2c41b38013ebc4225ea3f32f50ed36cbb2d", +"290079b62dbdfcfd0ae51082dc6fc7cfe4c22603dabc72083e9fa8b9ca54845f", +"212924687a58942fd23c30e2e42a802cb00837dc043956bfa64b7ba456235cd9", +"6313c4ea814c91ec72c3e2930dd9370726a5fa41615f8f12aebd34897dcdd4a6", +"08d65b4ff1c2336f80ba3780a5578e129e6805379a340489aba9fdff0c4a6e13", +"492c8a34b81f3d21ccabad349befc1f7c81dc54b68e3c42e60830d29f1d069e1", +"6d28b966b27d9e93e6239a9765d980d1be5d56d7443a4bae10e132fe8e66b7f8", +"378952e80a8b59a6defd4f38d9269c0cf89833c886ad0945f715839a1633d27f", +"2f33026021780535d8ddded4d31cbdce2b986d88fb11585e39bbbf0f563aac75", +"f948a03b285b8fecc125c42ee230e5ac22228263189583c2d0a572e35496fca9", +"bc3881b6b812f725e4a08c48b075a7a7487eb8c6ac04d345125467e695455553", +"ff3fafcb58d546a36fed36931daf9b88784770d5aed6a507d78bf08d545eabd2", +"a7978b6d09ffca907a1f69dd3eab1511e8363a2d3b3dc9c5da64c57d3c72da74", +"bd6c0982cdf3673d593e51429b6e00aed22caaf5d542663b47c168d7b95cbf8d", +"0ce80f60e096ccaefcb787ef0ee3cf7bfbf7164649a3c7e63c4705684d373045", +"2bcdd7448e349009433ba4a6501b9feffabb8db16c4ec9329aa6f6ac45cc57fe", +"b0f323ffd5563099f1d9b08ac7a8a18817fc4cface7d7493ed040b37e2d96eda", +"ab539e7a9f45cc673ec8639ae1a8394f002cd126ac7a51151a7a5ce6e8ac36a5", +"d9920d28aa499f164621b167ffbe20687fed0f29e36ea1fbe254067913dc1ca9", +"54b3cf54dc48f939bf797609a3bb9aaabf8ce2cbfa5e1247f5b74d24521803e5", +"6da0c04a0f446599a0c8979dcd6285eb7f9ccc0ea3f9fed5892a14371a31793b", +"ad3ab415a409ea6ebf4ffac39fc079ffcb68168064d39953054c62b631cc75f2", +"600ebd36e6c63f625603a24b986038550352581f0d23f5b259f167d7af0fb95b", +"34c7ccfa4e8e20644563b2651eccb64d7bc31d3e8a681bda769e7c17b9bdc881", +"24a1e103bc83dd7bf1ca8d9515553717896972804638445b3ca9e41a671f001c", +"7c8d5b4f0d75bb55babf4cd2259210cd05a62e9a839ddd78c9d6745686ae53c3", +"7494ca987f52215a8c4b7fa34c5b7f055ec0977d95753abf105a4a89e36398ad", +"b0593790d7ea4ed74ffff34905913da78345a4237a449839c6f8eced37fab135", +"a4010714cb1ef94d5a42d0800c5ad314f937617c5a7c384399b875c650246db4", +"8759a2ec09c960f7cef83d72ccdcff7cc33007dbb0b694f50cdbeb611c04b3e0", +"6953f23701a306b0893a2d0761544407759511c80e057c113b2dd260046c7e91", +"03aed9b68c1b79deb9672fe6682e6ce3e12041cdf480235dc923c6fcf86fe03a", +"e580c56a0d3a5b2596037a667c7d17aa6ffbffdf80c26a9fa30433217bc01487", +"67d250ac876136ca23990b87f37ccee6390ff7ddc73cfb8dea8133b0d4d9783f", +"0031da6252125a1199f740e3c90c45359d27c695c6849e537e4173dccbc9e1e4", +"485cd6b0eb17c59ea62f4a9feb82e0c9ceb4d395e711056b4814c4ff5a99e176", +"fd8a839ebe70c444c15274a6c6baa10acdc71712e47d99f1e7b0264b100ccb28", +"9e48f83e5ce4cd704858f5185bf4eb98ab9966ae0f5fdb777362cfd8c1cccedc", +"f868ab94dc48f4dac8ef3651ccf09e0405e6efd7c7f804d726d675875db6c9d8", +"750a27adc83d719ce863303f4c8d7f74a4ea83c95fa092ed4c2faf26a56f858c", +"d469180799a6cbbdf54001fc041b1b2fdc4e24bcf1f19d1a60bd20a87b9537f3", +"1904b89f195d8d617f8a416b80dd7abc2c1ac500be432170bc509b237c5d1283", +"65775d2f215148ed213fcc5587d35e09a9561563e7de65a08867e7da49245229", +"63aafbdc6470b5288f70b3ac6063a59acb0a88492f2560023a4112c2a77f9795", +"a62a1dd31befa1df1d86b3b5a4d99f5d175367fb4a3784af64bccd2ac3937e3e", +"04a491f51552d679e9821226548aeedacf049c36257137cc14c66412fe755c61", +"e5576849d5981f76d3b1d19184110c8991132115dbf1e780e7df0a7241df5a8d", +"d026b3a4457ff5a99ad28ba834db8952c35e8509efb0bba5cb45eee116df70ef", +"a73182747492d62c4adfbc9ead9b8ce7a03d4882094fb227633909c389781fd8", +"ee0c6b15a655f3f62dab4ac16498bd0fae4974cc4cf7422337c2d0e02b10e275", +"fc3f5e03c024eec3b0963d31af92572dbfc5d0ded79314dee78717b37d4f8cfb", +"e125398fafabe2e5da03048b910ce12de3671f532055a59e2ac45bc1481b1153", +"99f10c8fe5f1aba938d343e4388c5d3cd0aa06aae35d1b3784f43537466ddb0c", +"73d35230bf345028709c15a49977813b34bf6a299a1f8d4bdbfce7a69fd4de85", +"8ad482e9fabf469deaddcb124ad27ba2a41df82fc811c03a1f23bc11df9df1b7", +"d0fd211e1582829c4199038a3b80f81bb312a5592ae58ee9e815dc608c9bb3d1", +"e4132f4fbe1a42d0aca1c5979f4326b4112003b818bc6965e8e4f01c6b6de5da", +"07b26af043c8ff5803af941fccb8380826f0b131c77ad3ebe13b7f1cc7227a4f", +"98f38fee065b764b0d626f9051e958efa95cccd2ea77707c2fa2f50df7c8db75", +"bf0a8a673a9963cfc7e02a016c30f4e642de3e2ff775b15c7e02cb07d0a96ee4", +"73416c4fb267f1ed00bc3d699a0227239ad690d12ba6c3f62e31a6b0ef7625c8", +"ec8850d6175fd6abe7b3f06f32ecae3295d5d54b9276c0162b2e81f54cccefc3", +"5c8007abd4add002d727d02cec6231adfbde7541eba9f4a3c2307fefcc02297d", +"0f74c58e39cef7493b87318a2e6edbc8cf34ab3a0e83db5f43f40d7cc7af0d26", +"a4ea8fed43ca5d0d4f09daf5898b70e435ac5aa0469ad3ccbdd6c2e6a2f61bd0", +"e8df671e19136a1f1b82ebb8980589e3c0fe4b97d13623ad45faf06f3169ce0a", +"9e7f81e22653a88204f7336615893eff2ee519b9c26f29090ad1acf271fab8ce", +"c6d9518519ff651a87388616ae16d528bec73775b92c8749a77110ff8a93ce87", +"c1eb971113b9fac237f9f1d9736ce083dc203bf35274c1b8b7493b66612a39f1", +"d7c6f9fb13f626b2bcf626537cbd0087d52bff5f299bebc525773c6ca3f23b0b", +"ec89c40bf72cc34ecdc6f387b431f9ed0a8f1a5e9b831c908dd7be9e93254733", +"aedc3cc5b83ad3e2f6ff5729e7ccd3917f74917e85ec4f01a07a5082b2da5982", +"f2bb39ef4c826a60505ec938470b9af73d0836ae5128ef1f7f086d226cb81238", +"3ee3271720cabb318cc8b2eebcee9baa5863854bdca5aefa3d0f9b0f90b702ab", +"c8df8671f9668ac130fc7765ce316ebbf578b00a1c4d7bcd26f63cd8332ea6e5", +"b8114415290445dde4ca10bbdbb8aa73aabc15119b30fc70ebd176c563bf3326", +"a7370d812757a2e8e637cbdf33647cc46204baba770627116bca31a1271c6bcc", +"8eb1eaf6e5b17328b174bb6c2597f5e2ec3d1a9b24e68c467cd7f74f5b83b37d", +"6d591ee06e1d68699f42802480d586fbd988ab6abd7b911cff06b307a0aa7e69", +"d3d6e988886bbf08e81e25ecaa89e1e93554973cc1a5a3f72de6b6b6b9fa5aab", +"e7e3c6160851e8b7bbac99ca699cea7858093ad4643a33d75ae8faf91f835bd7", +"e87429659187d346236bf1e487866e6a1a0765195aba3ba6e9d9b5fe4fd7ecdb", +"3a104d3ade01ab3b7f3e29915c8e032354aa811d0476f4a16322aafcccea4e56", +"9de08224dd9c48619a6573fcdc089303e1dbcb1c51e5f48aa2b83a7710b5b172", +"39959f0279207ca7414d864aca5233d998d4ebc950f15a4556f4dfd3b1932b92", +"60fa4a4157f7731241fb8710edb8af7291abb5d8ae5654905e93e6345da5e8cb", +"2c472f7d38dfeaf8bb8fad365daefe593f144f30b6913bb18fc4e0cef8feec46", +"0697ec1fc33104b8fc13fcfb19e9392fd18f49ea8566c6dfbcf8180b2a26f911", +"ae55af382d8a9dcb958f1414582092347166bb4b7bb1a754396e0bccd56014bd", +"20aa244555a00f1cf0f73d4eef596d9118db4aff90e348959795917617647dbf", +"e5145332793533c93b0d20d5eec0e9003c78458c25eaab02bcd3fdc66ab92ec9", +"edf6637fa0a2d95568c8e67527e4cf15be07080ecda9bf51cfb9beb759c74f13", +"71531db280f61dab2c5d70bd100850b5b348a2fbaae8d488cec04339d1262b45", +"147652be2b51a411e060a0bf72a4bee43d4ae1ba1aad56ab0304c6f85a637372", +"a80f68d30b4ed14f6f24461cf9ef99799d651e8af9c9bb218f343449e555ca74", +"89a53ae8789a005d9e17c3206fedc25582a87977600f55da02f842e0cc1f4980", +"ac4fadfbba8ea96d4371ac8acb30d3559c8f7c8de1abd372ec0bb64419cbb3f7", +"6548415e0f89a6f7975be174d890df4bae8ff4e8d552d6bac3659227402dd033", +"cb6956e264a5fd11c99a08b537f9513a13a2e3412b031dec3c01c45d1f7ed667", +"7622523495aaf2e35c50bedfb81e0f246c72d5d82452d3bbd8b70fa7bffec7da", +"e49c3bd519d0b13d795f511b63b2ce0e4b57f5a9b14f3872e40dcd8dcb65e638", +"c7c4244bea1bd64f84c8aa9acd7f3483afab2b3b58bbbde549b9d5dec124d3f4", +"eb20dd7c5433acebd6e1cd3d8193feae63e19312c0b08558151f20e6cb1868f7", +"8dda430bad837cf356ebfe7271a00f92d4802ba85a2aa4f28f5fda02eff98338", +"5eabe6b4d9ecabbeb050b2e646cbdac4e265e6b40fec3f35eb23834d92505bae", +"3bef08d2589d3f05b995800795f75e3b81d3690d36fd2a381573d6c5d4f13bb3", +"e7afaa3577d559b05eaf5db73ab7925f9ec68b72efc59d4a85456d006d2727ac", +"3e403252010e3f1bac62e325a318919253cc3d47d0c408aa4fd2ee7d81bdac5b", +"4f5078f08730d54999066d3991efe2ae575e5451d98ed0856091cfc1a6fc6da9", +"c6fcaf5c357c28461cacf9da67723aa6704136a19b0615642fb7409c5c17e38d", +"620f69a5df706798affc8735c789242c5c10b5d0f44ae3de687f9178ef8d69e9", +"6f7288ced692f20b9eea4c638c5368846a279ddf857992b41e70ec9c1357f8da", +"f12fcc9d8b0f8544e8e45094100c678d32d842fe91c29a9f0f6e187ef7192a1f", +"f690cded71dc52db6f9a6540a6883c75cba2da5de472861beb8cacccfd6d6090", +"2cc9123974ff8558289f0d5e4b50fc8e1ebc3e5f33e836fcc074834bf84b45f3", +"e0c9233c257a3dc6acfc04118472f9c55623cba384acf1d19362946a040ded8c", +"dea52eb1777e41b812657b91b2e078b8b2cc39b3948b0366bca3a45821902481", +"0a403d28dff954a0ff41f056b1c1f95315e7cf337caef05a9e2e7f18e62026b9", +"ee967636de2477ea0410a16c7526509f2902d52ed1cce379f64256f0f3549d50", +"ae2f3352a6262c1dde90c3a67b7401c724c00b4e25fb8fe6407508d879ff8fa5", +"0fdfdce7a90db27147ee168103b36985f61ad43065fe6664951c5861d361c5aa", +"2b5e9926cf8abaac9192a405131406b0abcaaf27c9ea8a91770c29d92ec5e43e", +"7077c693d8b4af463781a7b7df92ecb3ae8ac9d1cf016244327d1c75e4d60123", +"0edf9bed57bb4d663c690e17eb20692f0339f337f3c8f881d9834650a38c4b62", +"6327f69509bb03c4e065cf7776ecc6ab025aad72e34a6c3ea96dfe66e0250028", +"e09e03d2f8c380440752d1241742ceb50d1987b8fd07722787307c5087f59ee8", +"b99dfdf05957d075cb84953d0211da38c1e8901cff792f39e25f63962ed45392", +"33a7db843f094fd2dbd59c60df2d592091d4d0fb8ce152eac0c9b3c42bef46b5", +"da0be3db59c76e336e9957bf859b76cf86fa9cefff4963f3703d82b55661a161", +"e4db2d84306a2ca488ebe1d8099bc0f5d355571e0643c02eb7030b4afa2aa971", +"4a560e22c006f2a9155795ad0f04cfc271405c0656050b73b07d4dc5aa7cd1de", +"6965423c14c817a42affbf88f9ae2464a678aee1a9194327b49368e5b5bca8c0", +"d10c32a5d1137338367c32454990e4b98f6f1a09d82340327d6910eab6b75e85", +"6fd1af0ad43bbcf728184c6bcb6a48f1eedc2bae0066a9bc42fa9cddb1de028d", +"4e0b8bf2fc44fd3fff3662d5544d73cb23358ffc981e675d04eaa7a31e9ade9a", +"e29374f5cffb03c1b8eb54e183fc932226ff0037870fa562cb804832a1fbeb57", +"f561dd555e9726ed474a1f5cb1dbcaceb70496264f485d77372771d9504c2e77", +"81958fe6ee48c9989416c1b433ef4f66640567d1a2ebbe04180724b603065dc5", +"d5df196a1ed41e50d0c8ec443d102d1f38ea5b39f057e9c43af6081fea8ea482", +"6bfc92ab1ae8323b9d3e8b7527264a9222a03364ca2595c0007c95c5754ebf83", +"6df284f006191e823c1a496076a7b45e8562f840d53e152b4c4d335acbb9b42b", +"1afa88e19b848d95de486606f833ce6a45bc9ac684f2f8b762e136e96f2699aa", +"180ac14dcdbf3759430aafcac0a4b28201914a6b2ed15bb4b470cbc9b7244997", +"a7df823f41af9f4b16d14076776680f0a0b4992672e5dd48b55751ceb91687e1", +"4912b4b39370659b75386d6a974b67e4c7e5434f1604912c6b742b76c5f80401", +"99b139a1bfa3b50973c882be74108261f201cbc20696f1aca4dda3547fb83628", +"139f31c4fed4bc91c3f2ddb59ef81af7feaba999693771a91d2d88578c7dfa4b", +"87c9c83a2974fb47023abe3827da17428a4028192e38b1c89bd65c1e196d3e42", +"3cb3e87475c541029686f32ebffbf7a2e512e86738a3cba79f7ca56c811dc06d", +"9acbca60437a52735c17a2f6642bbab45a5fc2248d19dcb7e3a61ad061e246ef", +"46c24abd65f4aa55862c73106794efe647641219d217018ed66df8b5f718a971", +"216e83ee1fe7abc8d89c1b0816ad2447ee9f76acece04aeb004f59b4b21468d1", +"ef415ae3b56e816fe2ea5dfa42ad7d4f0a5ea4518ae256aba3ce3dfbf61a54d8", +"818d3ac49eb25ee39f830dc820ac905920ed58657f0bef5f19defe7b6ce62925", +"34fbc123ef45b39e393919cafe4d75138a117d8eb96bba48506f4f0fe50592c6", +"d263c02f82edbe2d14b9044902eb3b9abf5a2c0fe52b7e7abe5d83efac857733", +"23c1f8ca022dff9e586aaf43743916b5cb5b45d505ee995b55fa14560bd22516", +"e3e3efe45babf86d2a23dc93592c1e707963c604615c3eb5b5e9cb7b6f13cca9", +"15509b01e5ee29fcf75399ecacf7ec7036b6178d64dec1b5c930512a116a692b", +"7ef71ada196a4fa0287d09f60852b7b108ac663cc92085f6768989f1be64bca6", +"aed28953a220d0320bc33e45b1c7d89b26670f6fc630f00eb86f579cf03802ca", +"172224957b79af8a907b2a50b69a55d4b3dc77ed44ab7ac91506e45579a5d0d7", +"9f381711c7687d63b033bb5923520b32d71d08c2acb7e6cf8744511b0802af50", +"7f24e59d0e7ce3697888ace50aafcb92f5f55e8cb63ad9301a090408fcc4f6be", +"e6c629cbbf8a73e8603b914d5b2fb62c74890f8fbea75d20e08c3dfaae937ad3", +"9a7b6bfde4985e82f671e75262aa0e20512e083ddcaa54e1cdffe7b49519a3be", +"fe19763d5b5e2eb1b38088afbb6fe6bbd55bc613823f37311c9034fbe2ced354", +"1e05deb3e9b1a05c5a29b164b1593ba4a0672b186772bc6524462a81328cbb0e", +"4783725c77e2d0e90d4ae4656c69c1b54a264685a8d0180d8b412397962f2aed", +"8e3d980fd5f51c7c473ba6a41fadea679691fc87956940a74037a34511869fe9", +"99c4347dc561ed023c1d48e7826e3f6a0d903c95001e6e2e1466a3a280591786", +"aa7638781f16b52f40d2e691550bf4682ad10f323584ea56566b007d1580444a", +"2164dcfb6fb3fac50f08cb48e33d1aecaef232e0706e7986ded1ef3bb12bfadf", +"958e82458f1384fe0fccabf06ea3399e8522be4b5ef95c882ecf932f33b06b68", +"e9a4487734fb9e8c064504cf71e0ffe53020d97a3cf8f19b9678e8a3e0cebae6", +"97143a98ed954698e0294fe1bc4a72887e66440e273c48e860e50c6bcb1bef90", +"1f68a97c6267a26c608e64d37fdb3be6844c75ba8d94c860beac5011fed3227b", +"dca16444e7da02aacf3931ebf9a4ebc11a3ba8949cb89f8649db794863a18053", +"c4fca7284be57bdf4caa683c351f214654c137bacaf7bd0a6cfa966f605a7cee", +"3c0a15188d149c3856cf0cfbe5f8f64e52db9608c4b3f66b8d421688390bd9cf", +"852cf9b19c5b25d74374b5d3ee968a899c30d3ac33be1673cff532f252009399", +"dda05acd2c6261bec3a864570b65b485283c16ca76e81ea309c445d64dee529f", +"e6d6d088eb9a5ff76f5a0a6f3e6193608c3e59a69ab9b734e3554428cddd0eae", +"3744de57025ea32d934798ea141316675594a7f8e96ec2ef62e5b531efe9f249", +"8d8d1a54b84c50b98f2b7cc7629711a9e8545e31325a6a1cb88321d3faa25b02", +"42a310ba16765c51c20de19e1fce4709bff69ac5bd51dc3a03011fa21136d333", +"585c16b4debabcc57f6a55ae79b6a366e3b609b50775e2fa7aaa29fb308b5c8c", +"2367f2a6655220c19cb85753029cecb06c4bc284bacd88dff4d8e91ecbe0907e", +"c2432559e7a6d98f5973ada70913faddd976f702da2c051284bd9f7b5deb595a", +"c1942e58a1e33e8c339fe8192e475de3edbde70ad7d615c0c77c39af8960a6cb", +"7cae6e6ea152db0e106164f91883b2f181c636286e6aab59b6ab5e46ff692820", +"a1501250daf41bab535c9fe97146137d4fbef4d5e19cf9716c6a12087c37647a", +"83921e142baf96bdcc921e2c91ebcf3e6b673e0bb95ebdd625ecdc469adb6920", +"a1eddf5ff3f1c5a760ac1dab80b3220b8048d4009a07275dee306a3350e905a6", +"4147526b3e1020ce01bff062c83eee439e4926aed7b3e9889d17484f67edce5a", +"d9a514de486ddc2e049224b8a121aa8d54009c6468b45ccebb94846aec6c6b40", +"c2ca2ac4023ebf6b1949654423c03b7b7f75f8dd7d08e080d9be60a5f7852d7d", +"2a86d46e4403c4c63ae49cdee6e9e28b0eacebf2f1f3b6008d076e12b40fdfde", +"5b2b3f1ed7857040d575b55a58d32b0322a1741e67e6d2c6072ce178f26c8f46", +"fcb26c7d17dd0a981209893db9283fe19f2d4ce3ea35d6eccefd8db3c67000a2", +"2e5dd132f996c9eeca24d4663f937e5e5935b896b1bb696930977cb0ed722e84", +"1401025700fd9a9e477e39b78282b1e9929920395d9f4cb77afb671af0f740a4", +"9d43a58596639ed577a208d86eb3ff599972c0db25bd8f37701a5fba55105100", +"7a41df7255c7d308e663b53c8df7ee8da507079f20c599274142e59721ea21d6", +"825217c06a33d7e84618c185fe02ab5d03330098d3e3374503f5e5be4a9408e4", +"c82ae7b8c2f03f7928111c36447b75900acd3874a26cc7503370902dd986064b", +"b58d6ce4b4d10e1cb7dd03d0b305c9e02f723760d916505d4182a1362813a1d1", +"7e06d11c611a0cb3fe9934165cd12b254637ba624ddbfc25439e714cf6bafe0f", +"242f8d05ed142a2050ffb131e55ab9f85eab3460a6647176804334a69cfa48d7", +"8364022cbc702107c5f9f980961757187a20eed12e7ff1969ecf13d5035a239e", +"a6f80ce4c354a4b16b33e35ab36e6362f6f419d018ab16dd17b5964cfebfcbc3", +"bc9be08b0b1dcb0720615fcbca3326e14fbec3bc280f80dff17797d98c51b761", +"4745a9c3ae1c1b4a2ed9ae0c2e9d476d65565e6a44454f58a33c82485c99b18b", +"626a3586c13085feb31c4ef0d19049f34e79f34e811aa66691642f40d98c7a1a", +"a41a28f7e42bdaf72848ad408e8c6d2555275019d3a6391f24f54335b202c350", +"9320514680fa08c400c18ffe3de8a7378add2ac7c5a282fb8e8a5c0d2605fce1", +"bfa0ca24afb007e9a169c627c8c15aaf28379c091bae5a5adae7504d7d405cfa", +"b446a4e6f04aa5782dada0957b6367a6222960364b8b627df23cd60e8f74f3ff", +"05b0ab9e186a891133458510c8dddf7912bbbaf930637f0467247c0229808fa0", +"de26b797a9361ca0bdfa68844dfb5834696a566719d1b648a2c3c54e4cb507db", +"3859b8c2730121b0c104826755e5d8be17a45f0df54068a946b6fd2767a0a4ff", +"ed8dddbae3333dcf0faf8bf676f9d65ab3573231b18d28922e089003618da116", +"f46414f0f299f022eba28d7a4317563a12b220532a419b80b99ee1ed28f6d998", +"a8a98b1813fb488d4561f0e0d6261cf613b26b676df3b58bec889a65f7f1fb3a", +"0683ac705a28e52cf290a07d9572bcb75be656e8a83f45666d2dea9b7241ab76", +"b6109caa4fe3961c82f2f1ce0209e7ff15fe272f240bc7348314232d91e4d447", +"5d59da2f8dfe6e6bcf5bea27674891a1f70d1f37bedf5f87dbdd6e87b88728f3", +"dd6bf5c6dd325941405718734adc3bb5a2563233a6dba612f56d9146691f73ff", +"afa9a2de1ca70b25c26b63a8a6db8fe3a8b6cea76367af4797ad9dc27365382c", +"af539611fc84be3fb44acf5852f01f1410ca1f599ec29b01ad5236e39280d277", +"4f35f9988802e1907758014c1acb76b332fff721b547056a9aef9ec74c968557", +"c98fbd90c7ee9de9231503bcbd9724519b5aa20fd5539e13d83199d89d86eac8", +"0a735872c8a99e84003ed853e8819233ae7c91ed6bd23942d379e2918be246ab", +"8d644475a04da3daeb324a5d3c6a3ba2c4c1ebb36a7b0985581bc249bf480089", +"ad90a93589b4afe272455de68769433643249e5fd6fdd0142c00fc5baae53953", +"dc7b72e68da13906f0ed524aab9f068b035fa74019d7153753f63e44225195ea", +"0c86c33183010597b4d25870b8407f139982987d6f5a93862d81f904aabd6ddf", +"c7388e1b8f441cbb44676413900602fbd58ac3771c3734e819f42aba2b31937b", +"fa966edcd3ca6c3fecd2e1e520a2e1480ac7b6f41b27a195a128846f3116760f", +"7a37ed6919d5bc567b82cf68eefa9d0103dab2702a60878f7f9fd108fbedce58", +"0e366ae2120a3cfb879e8ca1c9b1e698a52138148b185bb0ffe5e3af2651a7bb", +"7d416d43783c830dc6d3766168f03633a482b7ae3c0c8a64c5e130f7f260ec54", +"e28e9ac150fe846011deb5a39ee2d71afb08aa79cd5970b37a282bfe129ff0ff", +"eb9c004efa23db44c0e5bb2f82e72473d1d61bac19490d16a94d2cbd5cffad99", +"357225d6fc5a5a8aad8916d35c58c05b4c00af05e3f5e9069a6905ca7f820d65", +"b354cb67cab5fa86dcaf2b175de39e500b9150dbebc6b65fe97e836722448701", +"b4baac3dd349a1b2adcae0e9c1250ae43476d3405b9834172879c6aa7101e8b0", +"18aa3fd906ad29ddd2154bc2e09781ee50078d2b905214805e4d48aaf9e366b7", +"85954b8070762e3a89333300bf650f3be61c3e4fc636ebd7d0741e1046b5c9c1", +"55473324a576e966c7b1249ef44b7b53323cdecffceeb6f7790ae51549d59da5", +"925c9e11ff464190f5b5b8199785683fb613f6432bddd20f435536762f97cb44", +"319ab99f5daa2548652043db0a2e53198ebc29b80aae4b0b88e1ebbdf4cdab7c", +"4ea7ee648c9d55fbbd37b8ecdf6368ca095e3b8d23439fd0de17b23029ccc470", +"7d73dbd506c024176c76d51bec6e89ae2d504f628412b53952df4298fb115081", +"c13175c938e3f403e540909a5e11b4d78388a98b8d7394fc098b466d8cd41b41", +"287a71724b0a834ffba9c00ea8ba62155b00629a1e0407290d51e7b6f5eea0f9", +"656e9d8f71e4bce077aeb33889cb6bad99a99e8296d97e81ea18cc2c7cf7cd00", +"997c11c9d0e3a5f6f3ff9c65b2d4457458862692215a343833120699271befb4", +"aba137d9fa97658b3c2857b146e05927af961acfffa2a9394940f78df2ad2956", +"b10fcf88e9ea770c7a170ff150188850041e14d2c5f30cb80320d2fc13152cfe", +"a61dad5e7db4e0524c9f6fb351296d1af9c1118fd1e6bdc257b3dc7c48fec71f", +"4e1bc2a7fc422ee2ee6a686cc58503f397cf898df143d1759df70ee5c5a8d411", +"243f0f0459853a3a54bc75bd3d3ef4d775e72397a49aa8760b3086f6e7a72184", +"495dbc4d8fe0dc22d8e811d041117b8ca2797f8f11c2c198ad11b54f42bcdc2c", +"f8c8f8db9c402362dd6b5332e81debdd0e4ce83ae4224238496d9fdab730b245", +"1033cb6409b4da991a641fb2474977f2f24496d84483c73c259a6e292b01a1d0", +"22f1bcb84b2607883f23508a4b014ffe61aba9b5b44396899840207047c630b0", +"04f329b2c220448826b3b58d03192014a2bd6f8c987e5a55dd58609b258d3432", +"d5d476b26e47c3037ea25a558d36e54cdf8740d868bc06da5befb28868c9ca52", +"e13e223dd787f78182a6259b8408e59062a5dedcdea20028d600d111de5b6390", +"8564b8d6a3a16b00c0e986c0fa030b5d14876d5d4a4ea696885e524b5bae5bbc", +"da8966bc61cdadb5845c8dc43b288e92c0bc742755969b349fd721d6a38af544", +"68c7dbf8ce6a1dc995d91a9849bf61541d9eea392eb426f758368145e8b84c3c", +"bd61373a85e59b45da0eb15f413bea3c8bba8e903abd72fc0ca10f6adcf29429", +"13c644c4dbacee307d30d6719879ffa546756aaaa3b29d1125382ecb46474b1a", +"613c14fc3ac3504318b01f9dce309190e989fff19e5f10658bdf7364515d0e95", +"71c5e0578f18c9a558627b38e660f590c3eb3f6655e144e7508e0a79ef1db3f1", +"ca57407f3a2cfc4fb8def48fb2bf92dc9963f890971d196d1221e0cc67650fa6", +"05a5366a6fd57e3a3508eb6938aa4633dca934c79444adf8dbc599feb8213e8e", +"35f36a6bfeef16c447c212b94a9fd039b1d1ff638597e9aa17330c373387cef7", +"42fb4252f03aef5d460299d694eaa45095482fb59282600c4095a9fe3aebd2b8", +"ab980ed336134e2b8e130e44b59433421e5ad9a0af0287b58f71ccc5d450aef2", +"98f4a1691f83beb110f3bc2b726e3b4542a3eba1116d57820de95393f93cbb21", +"6499ff42493740187b1098a7ce2f732c150fb65918542c3c2e3989f074f72d33", +"01269a7ee8dccd5fb374251a7a299a23c227ed142564503dd5c10942ce0065af", +"aa437628638cb98163f14fc91b2029b24aa19303fde63e9c33a509c1a9a7b62a", +"15750e413196f28d589f4ba75c0d21a15c465d2b68a1ab068423b378015196ca", +"a03904ea8531b30af884144712bfd7c7ec1d7ffd092aa81efe653fd9feebd99a", +"2fd700bef22948f7754792ee9418c582444135a685e3b9f7d721c633035228ef", +"c587bf02524cd4a5dd529f855ee41192501255fa80f1073e044d678ec8865ed4", +"6be90bf1e61600d90e8eabaf79ede8c079c751b4c63601fa53d4c937253bd722", +"26e8b71a8c7c043ca18cf83956ae00f293d397059c3a028e18d49821ef93a10b", +"67fa4b87d1031e4f68dc757dccb5ad5b9a9faa1e6025b428b49cc63df813ec41", +"2383ce5d0e87d094815e0b232550347873cbf7ccb679581206dbff8f98c57d7e", +"14cc9d14aeaa57110bfcdfdd298c91b03ecd52e6d4c20f55cd2c6b393a3a8a97", +"54ad84f9eafea39b70af34512d1efbc4cc44121a66ea2f916912ae226006c0b9", +"d8a9360a4ce4c01ec2bd1f7e14f26d7ac70eb423e54acfa1b00d8116c0e6e974", +"6da086fb73167e54767d266fe80d9860b10a80065e970dd85873395e013bfcda", +"8de5f153fe213f18c4cb691414a970928ae26e9ee682106234b62915e1db1025", +"cbf62c4d4c652c932ef0ece58c8fbdd616efe3c138bbc768871e62ceceb40e11", +"a3d3ac697f33ada49392469e66a14204de4464b23ddef2e6b4cdc212dc43bcab", +"8f946ecbd72c33a04d5c0a0796bb2878c90d3d93018e13422b8e37fd6f63f0d7", +"13a28d790145f1c4b9d5fb7d17add3e2487b2296efe2d12b528186ebf71b873e", +"9d34c9b4cf094d63b2f332725b20376d9a2f36b1da15e9ed094102d5d7fbb267", +"d484a803815e472112341e6f11633800609fee9bdb42f52ece0d59ed89ea0bba", +"c1f4c4e98cc4c1a1b5fd6e17953ff36525807731a03bcb3af3ef5e6c2a0c35b6", +"167feaf0674f7d6e99943920438688edfbb2c7f031e0aee4b03f2843c369222a", +"3415272952cd2403d3472d2796ceede3a698257173ebb32c88e7a8cbf07044cd", +"6e0aa90945fbbfde4c43bfef3caadfda10cffcccf715e1158a15c6761857f938", +"bd988dca987a913a0264e481bc2e8961668a3bb29f51c42cdef4b74c99b561dd", +"5c088b8837d4b3414a4604358ee7ad7ea9c26abf50e4c733de1777a18c1f744d", +"301c86e6b4295861d2e93071d7d7955fa68110befdb4cd626d68cd1bcc2d7b9f", +"981208ef73bce2a891efaeefd3dc97c8b374db0714bd0b0fe2348b5fa507865a", +"b4a11bd6b1642cdf20391b1a0a23ba5aa438198f9d00406f4f0df0a2cfb4f53c", +"14dca7836407782411d2f9d7f9b62d770e0dcbc43d30d417dde6745ac00de412", +"855995aa6cfad55562207d5fc8d8c644f80e61b4699d9810cddb8025ef40b0fe", +"4c6d13fcf37ad1246690ca798f7f9ed574f3b33ca634bbe2489929a14a41fd06", +"e19ec8ef252fb24a7f74466d93d7544cc5bd017ba084a7c6671a14f174f8ce66", +"de275dc5a9ad045a295ae98b8b7b5f113e79e778590a8a1925b597bf0407409e", +"e603f52844e2096a8ba02de86ff4e868756079887c1de037b5ffc7eb04c8a1c1", +"dd074456bacaec2242dbe390063ac3a60db054b39ed92c479eb8123cb2b0e256", +"1c0226de2a17981aeb5345fea922e10feb5432f04b9a9533524b25d2567213c2", +"4f1df95bcdbb6e0e161608439d10e34b4e5ed9863d2eb2ea8c07fc6137555cc9", +"ac1a4a40be1a14543d2579c2eb0cde8c447a41c7f42c63743d7847a8fd6a5361", +"150b2939f4707c283545fd4021fe3cb933742236b58c63ce3ca9aea7aaf38bce", +"237da897b97bd740690f192c8b644db259d12edab83fcbb4acfcbe02444ed831", +"33a43c58b5a21847e28e57eadb2778468b03888431c90c9bd12219244cf347d4", +"2e642f5b8ffd0aefc455ed55ec83e8956cf660de757e040ccf9a31039807beb3", +"5aa9b34078c144d472bfc662b9697de939f0eeb6525c78c5f16bb8b1963e9eaa", +"ced856f750aabeefbab75dbc00f528490732bbec8bdb0f7ee4c429428c489d03", +"41c96df8b72b3c9693ded032102eef5c7d354772d577e949d061e9b631f9a2c3", +"647d82a668e9c2bbef88a7395bc0f28932b1d6e488093110219b948716191dcb", +"b1e6d8d88676c87b6f06877c12ee1a1463e14c5c9839611ba7de87d752f080f5", +"996e281a807314354c977a4fda773babcfbe9f39ae515dd0491e1f988c84e1bf", +"889bb6a7d582e0a3845a6c0062e5012cb3506e70987cf359398a9d3a08ff8c22", +"3fdda48a52e50717c1ba37d4063481b3cffb83979a306452a658203f67c128ed", +"2a30b6547075d68259ccab0933b6aa0cdcb38e3a5a6fdc98a4ef43720be99137", +"9b224ad62ab1c70f1660fc3d051083fae13108e907a385db265d22aecfabd65a", +"b6dc30e17f48612e805bf459c1163b93c8c797160dd719f351153bccfde4d033", +"4bb5795219999b07321fec1f12a65b2ae2c4ef5b3aab44573d7aa766c8adab27", +"0d11b55375880cf6a0b77037ceeb53f81c56baae3de5cacaf4223f4b1a248ecf", +"bfef5590ab7c6294e078de66521406eaf3d885da1fd8b62057dd0d0b4a4bc885", +"9280293c579316ab2bcbde6d71e54ae58a682804d999fda218953dbc8b4c5f62", +"284f8efc379a6ab3d58afea7f693aa845032b06248bbc4db4cb375deb921d9b1", +"a1cf0b2ed6d800e22a226956e01abed8f8934d3d5fdf659fdf029a08a690164e", +"c748f8cd6098337267b98f4207419a6c5603b37624698d9f2ebb78981e3d3d91", +"d44f7d749d4687c8082f49eb148a114e7f5443615c43353c74c459b1673dd2c3", +"af283b8c1fe2e2a43dcd6992664eb1f1272aba683c9f75aece89bebf5a6a8ba2", +"e58eadcd82d81dbc545fade9104d6925a89f599a4366f9403b773115d88cca07", +"a032cb84e79b6b4705edeee77a1ff57fe4e42327025447987c1236ab26c9bcc1", +"e135d13dd7e9e195c0d1a9d396e49576a6f832a1a93c176ee4c0d1572258a828", +"55a9caa385e76be158c22b617d6ed92a789d16c3736c8340668f4c9413896f5e", +"d564e61b494621148344a091deeccd997e43793d3f4613b7e06fa660a7be4902", +"9ee1ded0a1f8906efe0702ea6fb4af7d3f027516758cd2734e7b734c13297940", +"9a73611a2ead901874359d3110e525af13b59fcf2cba9256bdb6c1080b2ee34c", +"ed1a896d53c1f83c97cf26993d4342bfa7c692aaadeaa0b181c6edbcb0dc9d7c", +"6a53100cfeed1647a6b5c3534fa199c67c52b71606382b50d4f386525a77dc88", +"df13e0972e9e5446ea13d886b20e17a404eaf6f5e4ed42d838ecb067cae8971d", +"a11fa065f1f61f868706a799eba2407d19e2c0a9143327150df96c76fbfd2a7a", +"087742207b0e7f0d171f0ce4999bb1003ec9e7da29dfcb67d2b2b3590b462679", +"8bdbe4439df7f63259203666efaebe82944e614a4de380d1ae2ab79631774ce9", +"61a6ba7e8760fbbc770c799794e7489910da75e8d569f52cb8bcfa3910e107d0", +"b34ae492d8c8a85679ab659a08e5becf213ee8ecb576bb787450bf0b1803d300", +"a57adc7478e44365ac3d9c9d4241c77d55678c5fb579a1cce4e202e44a588113", +"e09c0dba5b94c9c4325e622879410e0dc0696944bbcba052a1639c548f4ef96c", +"88cfb56f223d4a7c46abdca119d528fb2acba9d5409fcf263ce4e347234698f1", +"c3bdce9d2a6a9a5bec080feeb96babd80644d1a53e39e0340e695d50bae395fa", +"779c3da046d1366ec072b2094427e7f3a7809a16a5f3ceddc5acc2527931d2b9", +"9feeb4fd5faf78b95d767460d8efb78188f512090eaec0771ec9e65f88d57b4d", +"2cf09409b56681b093a73e79d3569717e2d2500655f6644fc098ffd3fe229b87", +"be5c1e12a978f3bfea562fa6e4708ebbb7c45bc112b23c33887a937cd51627a6", +"e96afee7b926ba56e1be566c6225945f88a0a2cb968b6700e79763385fb89f10", +"7eac82262c8f68c068c882b747e02bf1be690fd499dfc52e83ff74e95fd90fca", +"2c17890130218b6495366a50f67c8d77c00e9732a11196ef9ec89b66beea9771", +"1da8788d59a43aec654d688970b044151453076ec26d940f2545a3ceded9e2ef", +"69217171c28c4bd2a677622822c4c73458b6004656e5eadaec56bf2b6a5a3d42", +"9ac193c94cedabb91b2c12f8fc078846f527822acd162a3603e04d8d76a9e3e4", +"084ad8c8f5e3ac702326bb1705d31be52bc45e42166abd456e37de350ec87f9d", +"271ea297da421ab506d26597c863598fd7b897d26b7514d7904ee828137400a9", +"08cd857277ca8b5a1ad1470ba26bcf716c31146bc3a73636728a1ca9ba1666a6", +"042103b178a9217bbe7dc81f8d24dbece68e72fd7e201e89af835f27928bffcf", +"63100fb102b1c720155ab64c96583316f4efe2c38b19ee7df75ced9004f4d1fc", +"d1953a8a32722b02ffa34a2be7f6a3cbf6f8993c30f25b04d27cc69710910084", +"d11e998cb62ed02941783a49681ff6873677b8e1d1642bcb7214bbee80f1f4fa", +"4f07a9fe7e51928ba285b8d007988598e0e4bddf16bedb4e030aa1006a9d0c40", +"92459b3f14f695553abb5d42713a348e02efe9034e236e135c181984d9307459", +"6bc4a0f5bc77e1ebb244b53445b219d3bad378683bc645000bc8a760e5c93550", +"0fa17676ce16a01098aaf5e36acb05fc1e5c79e2d5cd6b2f5f1bd7bc5e119834", +"60a68f2b83632dcc4117fb186a2186cda35b5121fb2ca110a7474369528ba8da", +"30ecd2af479a70391c8f5af4006217b2c5de10b71805b6270d71780535c852f0", +"5ee990e6350e6f9eb03c019000e263956f991925034883009d859fb1cca5be50", +"df1bb89e84a1c981d67b3b174d38120c790e8429150a9f9f536d0a12b9bf2637", +"03ebac2aaf222d7332789dbb1423e3869edb30bc3760f6f3669ccfd3cd787677", +"67599d7ef119a94ca9d8e04874a7d9dddcfe22abbdec174535471b685c1d1170", +"ff1b464da89547767fffcbf91899faf761a711b0f06f6c6833a0f516c7a2923d", +"5dc2a2e28431c6d64e1522bd945269bd9f839f98825e8affe6671b7388399d7a", +"516f6fb6c6ac4d044b901c164dbc933a2b28ccd54ad2aed8e707e34c001095de", +"240edac60a22ef8b8fca52340bffc12ee15bfa81b74fb386e62878253ae30c75", +"f21102e542fd69d236eeef717817e9b3adc914bee13004e9772077d25b28ade5", +"f0fdbfc57e4bbba5b108025953edae824830e0ae489328d103c2096f0165561a", +"e902c0f9a45ea0a219ecce5cbd02a21eb8bacb23e28cc178061105bb46349b5c", +"02410a1210c73736556ea279f0cb06f9096bcfe681124592de4434709356cc78", +"8d8f9ce2baec7a5db57aa559f20d80e2bcc18647d85086c7a702b5117045ee0e", +"99a941793e680f6b2c5979ca640c1c207b4b65eaf03c6b49878b3598de5ded97", +"c06486c0da11c75248a6a946aeb2f60af3240c0d3012167b42e9fae014f5dc81", +"a92898793b636e3ab616742ffa1e48215e19b1104a9c3794a0dffc54e43d3121", +"98d5db4b5cf8a6586e1d0ccc9609c93a456a0acf37f2f2d790134383de93af99", +"9a5ced0d22af618cc0d9c7caf039f4f0d3b70c9f414bd30e90222f65b32ff0b0", +"2c5e732a3265796cfee9d92ab1e728f401186cf967194a2784a3982dea570678", +"e6bed07a1073eb3a92062c8233e83810f54ad0c54a1dc924b12322fb6eadb57b", +"a0b84f1dab142ded79326cde049b80aecdf1d0041f2ac442874549917b4cae35", +"bd5d76d22b2eabb02f7cd0d726c31bb1694b37d6a54bcb55c11610bc1b004642", +"d52043fd2914f0790500c162539b43a86543b4cdba0178b15926da07bb0ffb62", +"f0be0be33490fdcb1cf9087acdb7ff9f7054741cab5c49c67637a5541142d9d4", +"e7901fea9d74c2754e870f75ea0f387a0d4715065f09ca404cd5c0cbaf7308bf", +"71a4b2970df0d368cd3a74acf967e743f6c6a2ef22b1032c4290bbd04021b6cb", +"0f1b4b0f3c6a68b2b86c98808f5ad6534337edda484f3289bcc303754ec0b6d6", +"b2ba410dff075a4f91aee02773677a8b4c833c934ccde6e74d34ace4b7148580", +"a99b3015a73390ddefac4d3af2d2208b011277f2008cbe40f962a551e01b13e3", +"bd56d8914979e7ef7b000663c977d74323ef51f110a7720f94aa8bd4bab901a3", +"b6844a6f003d20f3748ab1243f7a8fe10aa2638da7b690c705a9a44eb716ac93", +"b138fd81fa6c0123dbcd13e1aee0fbdea7ba5bea4d9e4edff84a64a79efac5c0", +"e96255df0710ff341f22447357f154fa7c133c6166c02a93a583e8655ed71f59", +"2520d13affa7d61399397b2eab77d23d1299874797b820d845bd01a17f380ea2", +"0457094775ea1a3ea4ebbb39ab8625646475a5c0ff3445bd16a381cac854de41", +"9aeb1dc93cec556f94fe470eedd2c5ee9216c39ddfcd8a8834bd37d207c9e9b4", +"ad4281028ff2d41d0b422540ff3b671d70787a6696172d282d8c5590e689c564", +"c306b7e72f151427d82ebfef22e8b96e74a3881e5a216b859d061e07013f6330", +"0ce90994acbe4ce246bc371815f50c06ce5b1cebe6a42f165073ad64736905d8", +"073539c868485c35b80370c656348bd86047e6c6c93d8bed9024d51204797e83", +"38c528d4b1bd8880a26d015f7be69d4fbd200611f2c914916c08e1d26e0dce66", +"7a1845f7e5b71035754a273d448641e521a69fdf38b50a90ca69c65cdef4afa9", +"7a2f51ab16dac88b311af9fb5a40bf43cd003e2a65b8187d97dc89d9b1a830b5", +"c85c0199ac0089d5c701b32e0924a34fc2b9c1bd07ea4d905b9219aaa7adff61", +"5c37b68c8f8023cca53be56c331f26e8e3a15c1f71b03e90ce8a10b5e3868d7b", +"89d16d94844468705d365d8b252aa7c02267eb321a24f57ee2ab1f9c0b693d9b", +"22a37a0a183b02dec4d15a04dcaa6547a7bad9435380b01b0108fb842e704996", +"7a7aab65e0b372b210c6828e5a7294b48bc77930c7aade90662900b599562f0f", +"22ecce78d190c103f9179d917f7a6a2f347fdac0371a955402d9c5e98bedbd37", +"b480bcbf8c02857ea59a95e6d7529bc65de9de53167f5cbf185ca0b3299154e8", +"82fffefd42468d52cd400ac2361894b4224478e9ca351f2186e82e96196e0988", +"8b6847d7cc36e6369432813aa3e5b5e162e0f84d5dc0d5e29ab7fe37f350b77d", +"780d298936dd67f594bbb737af993fcc87384be16b0240a4f591aa02d87d660c", +"a6e77b508d7c226be1f06dadaaf8539e8955b9b9cddfb698ef499b235eac0730", +"264dda46242a930297449d5fb74da1c6f7ce7e0a4592a5e30a357fe04eb9622d", +"72232b27f144f59f2a99174c12a678b833c9630c338992c94b987b2c8b8cb019", +"0fccee4e52882c1745b99f85ad916ff61c37180f0b583a32e857245778cc6f56", +"85c91c2edb6aea350889523c736ab0d3cd5ff4ae6a94e9516784c9968a1e653b", +"2400f8e114afafdf863cc5b885577e7eb5f4bb665d59ad4c3c3d02336f6a88c1", +"280aa849df5decfb645429b5cf0032c2b35718cb6fc1dcec7736266f2d775002", +"69ae782a6a82b8532a92d042f5c6fb31729f3a664ef35f4b1fc5f295e8a53367", +"031bf0ea8333540667cf4dbfc3c42eee999c567f4de6e1e73e3ce4edd540fdca", +"03d9176063dfc3aefb4745646bee694ea9f2f8a020e65297137c4840362e986d", +"eeb2c0c2b94144cfdacb3cb7ec7d072542eeef13052258f2f4642e2c2008f6d3", +"c760f441017ddae392a8dcf1d696e5ebcfd94d564c03675cede5663b285af16d", +"a0f788702b89b1d271a1abe2bfc1942dec066072d2d3f730c01a839d000b9929", +"64cd6ea7c4b938ab5cf922be0a0186f4e4a8625bd638fd181a18beada1a39c74", +"9d5f44d5a330bc23dd8eb97ddc21efaa3c750bc46d9abc13777d80b8d8060604", +"7e219d62b739c9bf4e626ddc9e8c497b8a6d24d9052ad7d9176d69d7138edc24", +"4ce0e30a7def90afb5d886e0f569b9b02100a5ad34ae157badaeac2c4504c1d6", +"61259c5895440ea711034f7ebdc8adfd0fe4bb00cfed4c886654fc81d5c1a84f", +"488ad886f5bb5bb953b8ddf552a8fd8ca44413244a4680a84c886f0ffc507eeb", +"61aa79fe9fc48911ba8b398081cb155bcf14e6e8e83d1b841a439d957414851a", +"515cabb2025b42711646fc81a651031dcfc2f97c70f9f4ed3266a20de4de2d51", +"b205142a83648a3b042b1ede7da30ba12746a23b11e8d075349b17d6bbcb867d", +"a0a3c22f800e7cd63b1685d6f670ea9904e761b66378e373f4ef6367c6f7fa75", +"eb7092dac401b4574fe51762dda03960e86697abed02544ac5b6e64eb1b4bd2a", +"1fd99ad634b9eb3dc774907af4162e7314d56e1ed734d05ca8e608b60a9d1597", +"2f3158ceb743c15f97a59e3a8a6ff689aa8b50cba6039825012117ad1607e01c", +"c2a49fa150934affe9c501080df700b8dc073ed57ba87febfac8743e46d3138d", +"e9630738257e18af8aae0e5de9a0fd93a01d3018377b72a2ba72443c987570a5", +"7882a18362a3794d4d89e8a10d00e35fc539511e6babea172c9885dbae840aa9", +"31e67840a9a42ceddd3be18a42e891f8fbbcff35b14681a8c6c795a63e8811ae", +"de339fd5b37d5dad46fd9270a7e7a9e99d0c75bb1b0525dd9d266d2fc6c19cd7", +"1e4d2b827c7174693046a2c0b31410622acffa56d154d5f510af52f0b851bf1c", +"e24c041ee6a1f4fffc2bafb03164531ac8eba631bf2525b7c607a4ff9ddee854", +"4527eb021765205ee1a8326b86ffe2e243d0da4bd328118a1921bd080ca3b40f", +"0bd2dabab83ce32e087466c1e5b07a6c21e4e84028d057af6413e70f3f0133a9", +"af481e2d6b741da62f760c7324607ba467279ed2cc26a24846c9604d42092f8e", +"98fe2e5e4373e35bcdfd43cd98b146fd2ccf2673415f87201ba750c2c7d3b6fc", +"8d4d9c39678b266d5b9d167b4d76806d4807934881ad4a8c9db4b3b183ba9f37", +"ac6520d561c1d1e1341e1311869cc1e8a12bb3884b267d12c65b42cabd35afdb", +"9c173249c3684c30b78b7e229c6b46f9c06a752cbf7e3c69881e5a2229fd5cd1", +"fc94b65160000d3af4a0ce1e73af86ee39d5e1c2d6f543d1c8a7b5b754332d16", +"3a1ab7b6ba5b162440465af79fdfef30e36ec154a5cda9e0294628734a9de89b", +"23be7ee555e2a1f9c9b944c865da89a457f653485749cf2e13bdf8c8f0ff9e48", +"e6fff6c382176d23bab447d71ce87e7542303551de9f22d3e3566cfad438e9ca", +"b151009ce2a6f580c271af4c8229ec417bb388fc8634c8bb77b6eaf9baa90699", +"9cdd6931082f6f78f962159043e90cf8d3b717b97573cc81a7995ed5c7f2afd3", +"9a4b7a9e63a874cda9c82adfb592bd91be77baaf988b6a22bca45c8d24ff1c8f", +"0699d3d7e30aedde9c7c928fec7b42c3bc9ee074229ed14df55be8816be05def", +"41db86a3df4dec84ba4b5bc81b3c71b36710366972ec95aae67136445ad23cd0", +"0f443c93552e2cb76e9ce698c46a66b05be7d23e9a7098d8ae2c1720f57cb1a3", +"8f03c857b872f3c5abb2450489c71d906d3d96b2e6a9f3a466a7a53cf3287050", +"b4b9d8bff7d67683bc8479715d9ac68422324f3ee2313daab0b8fd78825d3d97", +"06abb57ed26e5ba5ea10c548e5acf2f4608d8b21b2a296d37eff0320c51b54c3", +"2a913432682d6dec351e66034b7f9cc2f64385974684e06b57dcd5e491730b0a", +"f5d965e2991b96606060352ee6aded1d24f2526c43c749809f25ac4a45141cb4", +"fd0744eddcea5bbae28b65daa28373241ccc009e56854382c16b9d17cf2017c3", +"20ad12bc366328da65d5aec6608704acccc58eb40fa2b67fd313dab2c6925ab1", +"08b75653a13105a350552565a8a6ac0dcb6393e8c06681ed08a852f56ab22ed1", +"97287725ff79c89566b653ba298f5695ce2e546f181552a9c0b689f63596dace", +"8438ad4b63db524e5c95ef084e5460b1649f834d2f0897c99ea20230cf01bba0", +"045bcc862bb05ad4d930553262026b858821f3a8777e9a0c19f614f8adc763db", +"2f197ba672c12fe3b6c16b61197e6cf67ddbd2178f6411638cfcdd53e80c736e", +"3cd813f9df72ffc561df4e1bbf1ece86f5365bd9b48d7f7169c7b20dcb2372f0", +"e56cb6f41c6991c63da50b9b5138ff31fc96ef5b23b16e69791abb9b66dcfc50", +"5aee7d5c6afac0c13a80ac03a559ae47592d5e4c104c12cabf1267fc8aca0e02", +"8fa7787a69a1602a4d323a5256e7198e6bec18916878979bad48baad4a46a2b3", +"48666e5b5b86483df943c40872d74a5f5dd3fa63b09f143217c1fd041a579534", +"48bc4ac3a0a3fbfaaa42450371553779f9554094c085c162235f42aaacc3e0b3", +"c1f4b1453ee08b1fa1ce45e5b3167a1e0853493041fa1837de255ef4810acd2c", +"bfbaee3d514c35bbcb02769a3912d98b9a873cb0a78255809184354cc429cfc7", +"c8fe90c3b3dbcfb7a7d40925a8f55021a394d1d17007fb753659fd6689f01b9b", +"300923f05fcad251ead00fc55773b4016f841e26ed6bbc36bb3a93ab5feaade5", +"eb19d975da3e1a7b398a98f6bc0255b11e5a44c7305bc48ddd6f30afc357582c", +"e042f94c7e39181965d64d12f45c978e190d27f715e241e66a639b3e841d82a2", +"bb3cb5bce05a454e72544adde97db2042459f58031b0f0c1d29c992777e0fb73", +"a2ed88b9159ff2d9b19bc2bae6694c238d6b80ceb8a993b46170c97857e59483", +"0e02e5472d03e5fbf4b2d4b818a778d04af8ce00be730c8bd7efa30bb430c8c7", +"3f2ecf3a1bc64fd431358b34af41e60b45da42af585bcda364f8894488813f14", +"5518611b2b9cf3985941d8d9a077967282b719331947da7099f2eb2d43d5892e", +"219f3b922ac3fa2735f437d6542a26b7d744ac8899b53cb50e7f059ebe1c2960", +"c7375cb79f52c69882c6c1ad66179bb1cedf83c7fe50362bd27ce4322ab89a5f", +"f8b56567d82df2a24a0fe656e59a653c9ff9fc332aa35b6890268ba95bf0471f", +"3823b06f22f403535f1ed1e4c0bdcba397c7362f9c66df370c63501f16b7535b", +"6aed957c964c83361e50f1047a34468e95a810d961ff08036e5c1727fa233879", +"cab865910ed8ade413af57a20af437d6cde5581b3d15d78bcf52ae5586ae1d3f", +"63ca106b2c0225549304c2c69186e2750c78d1a0daca533560ea04e7220ee275", +"f4f842eb867148c875473020426bad302cb6d53a2a511cfb98548aa0de351994", +"4bba76fba3e34ef57702888467eb2ec58968d3ad533ec5464d534eaaf3198644", +"954fc6d2f8581e4af6d1ac4935b84696b977349404cc339611324fd76c737c25", +"09c12608e99cb664901294cd6653928418e8c33d334fdcf83eaf5e829d8217d8", +"5e451cdfe1614ecd48f73d93f767d651686985c7a6bbd033a4325b76c5a9750e", +"f25f01f2117268e17ae1d9adf475424c872c49d09723aea986bf5709960cfc94", +"680dca1fb419aba7bde2a7bed26726addcbfd0f49f27cd1304be177e63083f7e", +"665de573c6e33a076b43c9910c8aa0c30cc36e0f93046e457a178faf6fe34db9", +"6de8b59377f776db437cb004cfb451f7a3b34803153d67876065fe0d8fac0079", +"1b1aad7d15d1f4c8a7117e093730e91d3c6056cdb933f4e9eb2fd05be7ca9f40", +"c3e6bce70ed4c65de4a39d56a01a3565a11923fdf786c225880728d53ea2143d", +"35ca5398197a5c50bcdc84c97c6ac62debec478537d1ee76838b12d6d71dedac", +"b65547e285ca6ab1af1c46776fe5e36e20879b6a3d639e47e10d846a5b6f1dd2", +"93ff5438847545f0cff55e7299f10e70e1445252bb2e188a6ff7186595588dd5", +"91a2f03502d4fb54854bb4a85ddcd4724e4caf0545ec2dd7103bc9ee001ac49e", +"7cf3d609de6b4210d442f17c2b38bb0674fa5a0543cdef473568dd16a2b30513", +"231be30df236bf79de2f886d8d9fbf98c412e563bcec75d6635c56e37f62d3ec", +"5c14b064500961a509bc9210fb85f67350f1618e10fd4ce40f5c11445e6b687e", +"304685258605cd9da92209731642c74b7830895aa5a5e949680b55cdef503c18", +"717b106c2921561a6911337ba203d76114ef9853a00cb096f6b7333493b469a2", +"68e2bcf484e504c0c0761139336942b41617a4d39688cc5719e30554271f165d", +"b68f347cb270120c52e9ff251581209da81425735e1ac520b884096e93bff3b0", +"4f64719f4492f54e796551afc6ba0417868fff7094839ca7a26dc3382069cf94", +"7d8398a134283859761dc63024cdad03ef281f53a3753b14ef6bf124c56ab007", +"93be193605b4a6decb7cb4312e3d45cbfbebc5869e619a209539bf58eeaae8d6", +"37b93829afe8369b53d9d5ed8df93b14aa4295879244d0b2d0fcaba3e5d38f57", +"f4c5d0af3f0f0ad299d889a47df833fc9b3c86ea3a4145c18578a726fb3d7b42", +"0e113ef2cbb645f8b806e417a52d32c166d40f0bfba2f2d569018aeb707f17b7", +"a0940d6a65a300139490c760e3072104cafe4ddc5bdd601a8577cf34fe56e56b", +"32c6c63ee6b0921b7862a4b1ab901790d883389a34eee7c4aa83c3b8773c3b34", +"67d8b3f272db8072b9f22013c6900409ee5b5db808f9b6698b72496e560eb9f2", +"62446c242dfe108964b2347b1ac672f95dc6533d5a18a8588b3eb85c12988ecc", +"057ce720effaabfd7c1579f7f2ab28d57879dedbcd8726c6d0e592e384987e8a", +"7134a8ffb3eafbf1083c31cccd2b66853b5851af67919c4cfdbb93d3ae68b512", +"5b72faf5c8b978a25a4f9e99a860caafc0a845fb6deebd79f88425ff933de029", +"9b5aac822f0d91e0e020f193c0045e70f783f444aabe46ea87dd8ce338ec1349", +"ec2a9dbd481b72ba4d8297ce9d739a1a640ffdeedf3d90c855b5951b25a941db", +"037abc468787fff73f8d3c8313e561c86d3949ead8858a50519a915ef54725df", +"4b2719d77688554c05e6ed5f685e3048a7c06ac2f2342e310046579d880ff394", +"b783ac6f17b3d8e7aee6249a39db9cd455bdd573eef26bfd4c1f56392ed96de8", +"52d29df5428ca3db30f9960dee0d90595f611638ef969202ced1e0990dfd9545", +"9ec000036ea4b44d4bd6cc0f0dd1b9dcd6b2911b242e3fc451cde8914569cdd8", +"ab7efa786f7ef4617edd4cf7d5dc029bfe6b3f808233bb47efd8aee7d646a746", +"d175178c5df2ee708924be190724ec5f3bf7a34395a42c8a473fa98c67e5ab29", +"f966e98098542440a54ddca84d6db81871fa2be1dfa8fde7f3a9b42a704e4f60", +"f1ae9980b1f4eaa7e0cd8c0c1f0ccb112661d86bfdb39f0bbe997288f7b5d144", +"d5aa0ec6c2031419c5326bbc523e56230a7cd97c0d30318b8ec07ef29c624866", +"94004f1e307c3a0ed43e0270049c61dadb0c6900c7b7181b58c4d768b3475e34", +"8785d3c76a07ad5c69f9326806299ce1f919035aed99b47b332d5cccdd9d4ded", +"7f0bb4fe2b411bfae5c97991e120f502bc1a3728e9ebabb9adfe4fa2634904bf", +"1cd6aba258d43b2528c54e4c9e9b87af87374d0c2f73fc29645282e06c1bdf4d", +"a06876e09cb6c2add17ac006b25b076131a359d28cc0672476db4b155a217cef", +"a664af53be46db53cbcd5ca312084976611fbc94737d25635fa04d6627324adf", +"17af26a0fc638c9153a57ca5357382d28d7e8f07f67ff02e9b5921431d3957a0", +"258dd927855bc24cd89620d2799f84f4c4b7fd565ed78c0db486786d9ce769d3", +"bedab1b1c93c0723a4260282c0fcd9cf77fdd57883abd3bc27fcc7ce895950cc", +"70a6e9ff6774eab796046213deaa9c54cf81773533a613bd3bcff0fdb4ed2d4b", +"4d71284351fdf86cbd880fa5f144643876b1043253d8c7c228dd830a16ddc0ff", +"046af3a0c38927f1d3259366d586dcc9309be00c720bd4fdf89e24f7e817e53b", +"1dab7aa7e13a55ab9031bbb7d95f3bc5a81b4fe1cdf664cd572150a3d5ddb87a", +"dc0a188b472a4af2c258fb79e237f5890f618a2b5b8b3db94531c64abe64b8cd", +"2879e934d1ca3c1955cddad5edd1b46b706583065180bb6a1d9fe674d1ddbd1c", +"61dd0e0f83b51ad52d26776b3ff5b60c362f5d69df6300da52f711240cc9c323", +"6c0283d6cdb921712848299169e1907742429908dca0c4a10a9303e930c63675", +"6c3718d84774f47846cb531b7034d5d6c512df69b6310a829a46734cabd4b07b", +"be2888ab0b2aff926b4d7d9fd51f91901cbddf9a9ad894ab102e73d6200d5d4b", +"f8438c7a53b0eb1d1b198d7acf540cec9db7aaee3a274f892d1f2bd522738b29", +"8c4edb54071e56342fb8ee5af6337541964708bb757a339092866ebd42a07262", +"8d423895ffd10527bcdbcd8bca0468716cda0c5a6fd8483e9dbf4e5bc6d5ebd4", +"2b7babdeb9388f53be9286ba816062c4f79a86a7905aed0b4e2a1439c9ff5c3f", +"df73b5a24f178df4dc47c3f8c36b8f219b519e73e88cd883ac1a2e1c05cd0fd4", +"8a55a4f179370e44c4f67dc31f88dcd25b9de90ed9dfcc2c26f73fa95c8e54e0", +"2f479813aed08440c90f05c349fbe0c1378801e30c6b1daa05f82f426196007d", +"b85f8721ee147e0f1d0893b2f12999caed4ff163bb2056a19dbbd4413a4bcd4c", +"6646bc7333822ffa55fea388ce4cf1372645333f63231fa99fd87ffc370bfcc2", +"fa545494c3b2f6a34bb2d7f1dae71979df159ef9b5204b5f0e4df265f4b2c949", +"8f648d2f81702a89404c19100a71c7146e52782ed486f081b0d29f91be613e92", +"9398548d6cf8f00c360e369701a26d8b1707f1fd1677c740a94c93dffd9e0533", +"af72c7a077898218a4ee2ca75fb2bb3f48b686e1e10dfa7d9474503f554e7d14", +"0acf2d678a416f19d7dcbca05c8ff935f0b66deb165e6bcd4adede640c803ab7", +"00d705df7e4d32dca8633c6ac9920f6b38da49b33eb6aa4de0822f23d91844e6", +"e2d646315b23795671ba4d8fe5e11ba2d87403907390adcd73a5a95992fc36ef", +"81e27781d8e21f909cc12bd273f13ce10082e24350115f49474d079de6bcb1eb", +"8ceceecc07521be40ee25297c5316d56756788b48e992d90d235ac49ca9854bf", +"53075780e0a466dadf948d8f5ce889e35fbe45e061ebbadfc0b6f4f1a8e5082b", +"c3cc975b73da6c110a1f2e85e263a1055b6863cc87ba577c24ceeb39dd8fc78c", +"ff3b31b767765ba551a4b150a76101060c2fe2c2e9bb68335b505313d4b4bf49", +"bd787ca8dd69df814de515acbd2f2340ca10f2d4c4e8a9d3d09830e35c03c430", +"9fd0509589d6f83538f5b7a2ea6a4d74a061ee1b4118ca833820683856aaa28a", +"219cb1f05a1ae32013f6f22bcda51bdfd651753206d28218e498ab2495ea6de2", +"50b500f9aac80d9ef6566d50a84038c149bc6bd10c5251a5162d4251f984c7af", +"9661ab74ebe8ae6291d5185688374431ae544e908082743b9e7bcf4d9345abf5", +"8e6ada5bc3323f297f3f3e8e244be2c711c354b5725f2d94a61adcf4ca4a41ae", +"5f3508474e41a1d6e0470ef9d881b7d4f94a9c06254b5e0c4b97487d232da456", +"6a1d5f6b4f459ff9f26eea409a7f01a130017235be79396ab49204b1292549be", +"7df5143600a53f3e6ef3709d3833c89fc6d5beae7ecf06c1c676f47cc3fa7b71", +"88e3c69af5ba0bb8f58ae6056ebd21e36f4fab123d375a3629ee92c402acf696", +"106241e45d9da92da84ae874ca22f9a79bf29e928913c4c9cd52428a3da6a2f9", +"e5fdcf01c5582c4b7537e57a0217676c062a67967a5679258f85fa990f1422ca", +"ddcf98e28823f484033ee08cfdd8ca05dbcb3e71e545a0744289e5e8780bcbf7", +"6cd65a624cbf1049aa14ee271e4f1495dae5ac3b48ecbab4dbc741d93e59d0af", +"5161a52ff061a78ed0a0f782ae93ba4a5b70d50d280dac9bfdad4d7c65f2933c", +"a7660e7a6acb691b984af1a99b6aa6d738fe6ac9592ad2de791930b0f8960f0d", +"55c9e5afd6710bb7143be7977700eeef76d045f1bcb54cf8b19bf46ab4acda0e", +"cb72367ff7618ef29b63b18ae2c2117f4d4ab3f2daae686467510d1433edbec4", +"0eee63cc7561e6909a0215fe338c2271ea5b09ec726abe605b2bf207266600fa", +"1c6c6f407aba94a67d6406b2b57dda4f46405b2092961a4c537205fbb9622718", +"96bec80f916d29574f229eb966ededebcb3660d81794e0ad8cff6fe4e7a6ef31", +"e535597cf230f8f59f386034232f56439ef518cacf3e564d98d3ad81c878756e", +"f8461b6f8a8b894fd4b7bdc0abcda42a0dd9849bd5f960b12485c1e284622597", +"494e43313b121b97a49dfcb0c2c0aa74aad3956c02585f00ea334a73381b78c3", +"e7deb5cfbae743d68bc230350e0075d5b309605343d3b16e0f8b84b52e98ea28", +"4074c7ef2f047d54ba02f5760187df82f58f5a2fce59e2698fda66edb22b75d4", +"788f0d52e2e9bfc0db0a8ee972fb2b4ad3d89a49a994c78184138adcc6e5c492", +"b2580a917e9f935e0ba3fce840346efe552cdeed0dc4609d42e40adaf169d3e4", +"3ff06318588ec7ec5524872e4f66e00298b840d464abfdcfc3e95848a58834cb", +"b5fb1967a921c1a0e23e67d998346afb92fcd7ff50a3133b919f8058ce058f96", +"e472d24de8e1ba5d97f42adc6e6c0dd7dabf119c1ef99e36b52d8524efcba420", +"00d60a0c0a8b1355c15931c3a729ee6d966bf62f7b18171746a75213a17d2c5b", +"a2d6ea0af7a5523da36883e03ee1d77a5d4b49eef097643f51c5cb4d3e066f41", +"3dc3c85870fd489a9d97ac9a27c655d6c481478160eae3b4c021deb8e64294be", +"5274dc12379d81494bc0a907f4a49c1cbaca59912e2e6415b0438949a4759b4e", +"ed6420353160d9b78f16238993d7f308ce0b6ea69fb48458af39f77fab595de1", +"bfac9e7574f5ba6ab95a7c41a55dd3730cf5778b10aa804c572bead9f9e7fbbe", +"063f377ca3a4572dcdb511e8086fa75c57a3d7d63b1cdfb9d84b2c57a1962487", +"4e3cc5f1c2097d8c6800482c8d39147a403a9a5080b0b9d777e1f00c815685b6", +"cdcf7bfbd085f9656a7a4e5562758849a97e245de07bebb369f752d426b341cd", +"a203f7aa44aa23c676ab2505362c59281b9ea8035817612b3bdf2672105bcde4", +"e537711d6bac1ecef536176e2ca96f0e1f49d2d97e5c7e59598bf6dbceb394f3", +"c42a5311c9455740f127beacd6486b7ea17808017ab26bae649049c2e1546389", +"802ffcc10c75831684179c7958b901c9c9fe618cd7f27444526c24d6f1b32288", +"eefc82010b4c7e42733d702c3869f9bc170fad49f47948787dbff27795f29562", +"d7ae4d948821c3d013491a081c3fbee437a6beb88e2c37a6ba4cf9578b1f35cf", +"a3be5b6c843f8c9f87e38b02f182017072b3df459a1a7e3f227fbcdeb1373d1b", +"5f81d3b68daf550de47f4b2831c1cfd822f59c56a7589d21109fd5d2a61f5211", +"85b6c84b4fbe49ccf6c91e5c9edfe7e0958c2734b4b96fde7a636599d1e529ea", +"9906c67e04c5ccb2921ffc06a6a101f598e5095cd5027a014865d58f65b435d7", +"733c2fdafdf992a9a4448dd1766861b433b91fb40d4aea0f089beea6e7fec1c7", +"801b1656646d2e028fb9fdb67245580b775efe29900c622c1f36d1907ffa43da", +"1f535f5d034b2ac0e568fef3494d6853f827ad1c59551f78a9894fe634233733", +"d445ece16d21807ab1d287109a775938ea09418befc20224a918882f7c81a619", +"396e5b31ee4c56a7080808e04f9c16deaedb9e757575657edd2236a426797f0e", +"1286458756197647e666f30b00951c4b8f9033c3f8708659fcacb9f87c99acbd", +"d8268bf2403e94ac9c58b0279c280d188a0fa17e72b733dbba67f3d034c7a7c7", +"eaee1b3b289cb0c12fa12df3d081a861e85cd5d8b28dfb6c49c7fa12c6d3c06f", +"0225feb554acb116bf399e804c206761345e212195848f7d574f336f891380d2", +"898c2dc8e72fac821b8ff1bda7103a5a7b32818b63f991b6a6626680d7635396", +"f95ea0508b9f21d93191966fe39d3f24d1f912758e9a94f0bf756d1f5c69a529", +"41939b9ce4b709b1c81c8c304e4238f140d06640848923dabf07d3e3028b39e4", +"3bc7809e5e37b5f67db4d3a5f9dfc69e2c96aa6eaf68c5b03a596dab012d079f", +"ece104f0b952dae0a2eba0a757ac135eea1de09dcb1fcf7a41b0a575ef561659", +"5954ddf026038976f185ac077935c2e57309f34fbf70bd0fa0e9ecb84ea78efa", +"2d8fa3fbd052969fa8259eabde5bd22738943c006a9e1218ed697896fb545488", +"4093baadf53c44636768afa3f6bc120d6b17eabdcf7b9eca7891cea65a725329", +"28f7a397b1ec84fc9e5f1f8071b824cc071210ca942618a1a95a0596a646d01e", +"5278197fcd1824ba790af2ddb72aa8ee16a0a313afaba3f7034344fc791c4c11", +"60027c1bd20f9849027966cdc4043a3d4057286b5e9d8d5e9966118195382932", +"eb0aa8996c12396c005ec1b90889e20cf7dfa05e1b4ed8838132b75066da7c11", +"b5704a3d26b8ce38e4af17ca90881fe2b8f6cd59bab9e70d9a9610d388f6cf44", +"4dc0081058331df702e3ce06f633eb99b6447f3d05d348ac7c5cb4e5b12b17c1", +"dac915b37f8c9333398938473ffdaa1e05ff6d03d3be7bea384d503eb62310f4", +"6bd4278c6a22a03127387fbe11a27dc7e926d506e72bddd67c9128db5ae79d23", +"b96b0fe632d8dcaaea82f128e5dee2eb9f1fe8b6d6dc8322cf5c5efcad147e63", +"7056db093774100e34a60c899481c7e9b998954699c6136540659ee985947e89", +"8a40e64b80a6068544de68edce9fac41e278471c86e3cef12991cb295527ceda", +"2bab3f2cab81d5f9d52be82b1342fff93d14b810a6f0bd0465cb387aa51c0a2d", +"cab3cdf822d993f077f6a6812bcac1740dc36ef24a0852dc0c79da7530824f08", +"6cc92eb7a9ea78d759f3fcf3675c31c71f7a1bf9f47483eb211d940bc1b2a2d2", +"4698904a9d5e8e35d1875977a6c75243bc895d6efec907108beb7916aee6fe9c", +"cdd5369767b8e3b2db8f59fdc468fbe5cfb38ac61be7163736b14d861a724235", +"da82ea51506e8abd2be0890a5010e6d727fc5a0bf6f401997832ffb93861fe50", +"e46899cd78a3adfcb03bca7f51d588b55c08066f18e0a0a0d462d73afb58b0e9", +"a02460ebff03b7cf95326527f13cf58eb0d99e61a8aa238927cb19300ae7dc74", +"3da58ba9e381d208aa8964c1e25e09e1207463dcff70062382314661f4ee8dd6", +"d6d4c2b7c224e34ced602e01574ee611003cad4ca3895584d720c9f999e6d487", +"5e1dde02072846b5a444dfd708a6e134eaee79bf8d0af96691b7ded7073c3957", +"961ace1aeae3f11523890212486fe4e7b7e74bae2d0dbbae6e69d0d26f4824a3", +"1205d967cab0569406fb616e8376f2a05c878c24143697a72880a8a4ae80ae72", +"beb17508e483c079270952af15ae6fefce3dbf176830a75b919a93fdff69cb32", +"db7fe099ade891eb1eca805c745ef3816d77814c8250309f7e7c5670444718de", +"27eb31611d11b5a263f7aba36162ea771407b3c30f481e0231dd45a13c3af447", +"bea2bcdc91abf90451f34dd569b90a65f394acda38f427d182eca52073be9822", +"8440c504ef7bf0273f4476335f218b44e0f32c5b49c27e84fff3ca399635e1b8", +"29ebc7fe9c2bd71e05ce4427430ab9ea9fc8f0f6302c6eb0f146bebf6787f6d6", +"9eb4389e99169d3d93c4d7ebb9b1f1c79f8c5534ec75e61cba5b7c606e803478", +"d3db1e77cd4bb36faa5f8b368e089f7d98d4b076bb8688f353ad35d42e99436c", +"17e444e6cf143ee9d631895cc56472806752e1967e2391dc3b43f670cafc4717", +"f4fc1c3d765ed81d4c7a3926364d5c65f9ceae3c8665d01d4d66d99515ff94b2", +"7e62866c76782d9e3685873a08caab233901150dc58696b800e26f2b51b83b1c", +"e1fd4d87ebf9be9b0ee16445495b6454688d621fd55cd0a48e51dac94a5b6328", +"c6c2401e683f9ff1f7a842f047d8348b547328d99c6e3b7c020e28e2d76f802e", +"3d7e2379aa346e10642fab0572552350f8ac27d39488d887d8b5d02c5934da9f", +"376ce969c7dd8df7b0ccc5a99946dafab757dfe83ff1162b60e7747ade770cf7", +"605f2b5c3c19c4fa6fbe2225b0ca6581cc91bb9791e5bf95e46449928d54369d", +"74a2d23794c84fa5c0fe24b2ac6d9ba6836edc6862d454076134f04d47226ae6", +"a96b1e91e1d0553f3e19a3aab84a229816f06f9b9cd8270461d2b0cd18907850", +"a7a0e98d9ffcdc98c901753f36310c8703295a89ac76f370dd7cf779fa300a40", +"c5927770b7704c09100a47dfe7601886cfe749e2a0efb751c809fde69a7e6bad", +"b3d575178e077c1ae72a7997e6ab114da2d018f2a448aa4cc2d16f9f55439319", +"84fdc1a1e915cae357bbed97393ad3ff808b8f32e9816770b2563f5553b69cad", +"6e14f95cbc3724c4970901139a1323e2ab33d995aab43d3589e0e04ffe095b50", +"b9853a28283219ae597b67ac6b497ea561bccf155e737307b3610e1d5fc458a4", +"6afdb674c5acc1dfaa30c879d9138ab16ca644337221c4f0d7bb0bfc6d8816d0", +"acbe5685751f2b93601b3b398bbc3892a4984bdc732b574f4a8c7a29cebc9c8f", +"4ded6220c6942f58492b1153d7e0890753c782b2591dbbed35f53e7638281ca3", +"0cca72a6471a124b710a974ebe899348c13aa2e8832a007ad2c31ef9d9f61f95", +"1da646f93b785043907635ea60dfa3dfaf7c8004dec3e60bd6e322f9adbb3a8a", +"9e921e39aea0b0d3ea7c62a9e0e8e74fc8e8b8a35e892399f86c84f9d71155f2", +"16d99335dd5705186137eb676101a21c79649df906bc3cc1407c884712d76456", +"7a55c56e81b7208e67bb9c37d86bd5f97f948cf9841746f9fae1ba07ee153239", +"cc40f50e13ca70396bd645d6c838527f613227587e90d78d477349e1f3e74fd9", +"66d3624fe63ba68c0387083a7e10ada01433d7b3ecb6c2d430c31ffb09d23064", +"fbbf422016f1e9ecc33873c5981fb28caa39aff1a1801a0affbcb5412ea1c0f7", +"d97dd52b18670c246b5b389106b35b96b05bf678f2ff17dc6b0e97023d3950d3", +"66503e5b733c8bd76efcc3bc7843e0a4e8eddcf53b82a252021c11262544badd", +"83d5b74d8457a63e906afd487bda8b1875aa147f8e179ee87ee9cc4ab0cf11e6", +"1a61d2adcde32e3d57be92cd7df0ab58a4ff91d45c410bfaba8a4ad53df2403c", +"b26911c3b64a8b8f5fcf3c63aa19817999b98e8dd2ca7f89e7e5e5cc2aa92f92", +"f643348c888b2e75476d306107bea5437b49a67195ffb58bed36ea7e4a73f849", +"416c5f21f2c823cff8c48ca91e15e61a7e4d95237ff093fc3f17c67466a55c08", +"17596be44371c32b667845aec2f6262d3e5cc977367b5666d5a85fca302664eb", +"3f0223b42a8ca21374ad8e9813a3772e48173610eebceef9b4a7df71f95b4061", +"cd390b3b760c981be2b472cc7bca1af1efe641820075fb5c93e680a621e05146", +"127ae7c479881992556b4db7dcd780b9a8ddf9e66ad6ba036d0fccc90f92ddfc", +"ae9c764976e53bd3d193e90d2188930feb85f75104aff65e0b699f7ea40363be", +"5ff0b645ccc18f67606b99d09bcf24d1177d111c9bbb2f57b0d215b5caf9bc85", +"c960f579dde942f2e40b20bced7823b5ab210742a69bda0f0e698e1551eeb6d9", +"a713a72d92d8ecd5ac03e693ef39a07a4098249b46f4a03105835154067ca999", +"5f447c93685f77d4df9a5c83f7938493853f24824b8b123d00a246087d0053e4", +"6495d86d2995256fd14b84d9e4f003892adc6a99a16739a6fc555621933d4214", +"c892deb36dc8c66669cb068fb19f78708d862e96e143395af893ec18f3abad8a", +"55653cfdf72d99322aa90f0ec4897665773f87546db9b3e6d516c8af3419d004", +"8a4ad9e18a6639b620a5518c182b966c97fdaa06f88d56e91c7ee99b1e43aa8d", +"2726b3b0f3bad42f2552a99e7bbf08d3d7dde44c820817790891c152146250aa", +"39cbb57ad363de140a7d1186480415dd9a3fb9c35d6888921be77cf58cd5c2e3", +"3fb855efbcb3fec4f757e93365cc9fd859350e521b34c2bf9e7b64bcd0dea74a", +"001d79a968a5f679b7b17276bc2d45700b575c0bf40befb52ec7687bc00d4364", +"346c1621eba9e080eb7d6fa8b2ece2abbd5e455f586abfb1121ade27bcebceec", +"4774b79f811437fa091a466403e405564086f879d2dc01f45a589726d8a6f752", +"91f512a6372949ef37fee9f44b9ec329c6d5f65dec5ae6c6241e599caf118888", +"3d2b535736849e72659b4b3d45cb13de6fac8e62e16967d22a9e76ea22b74ccd", +"6e2c001a8f5a48ffec63251a6bf775f4c0430b752f215b6c5c92e39de03287f1", +"8d9d7def8190f30ffc53cb7f164dc5a0d09a41c34e1373257a4189a468c1df47", +"18e4220eea2dd52a5f13854cba7ee28f7d37080b0e9060a83981c499da3ac59a", +"ad83d40410e9e11db4b266c11854f9e9185b5d90d67e9922941576a7b6a864fd", +"981b74ebc69354d8fdb6e521274027b9b281818372bcc77129be31a8ac13cf5d", +"27f4d324cf382295ec0f2e7b4a68a809091247f7818f82f704e9daed8fbd759c", +"09d215a9feba586688a6ca4fca243ac637f5aadda42d2b22c0baf9f970392a21", +"fb7ad2d0e085b771138355547e82b771482f952957a14f38f8d5f5eadc3c0d89", +"85948e46615dc2fd905d34ec40b9b6915ebab7e34682a0941c7ef83c3b28c4f6", +"f4be74d783dd07a06a09c00a65ce237689ecbb9ce9c80c8c73ca75bcb2cbcfca", +"e49e348e0f948e807c7a8edb3618ec36ec1c6134bf88494f21d676d91c2b2f68", +"b966aa04b58ae3be1e3561c0970015bc21f63b02c63dce11f919294f38eaba0e", +"c0d67d6cb71d5cd372060a6f508017d0e90167f111a5219a0008cfa5eb37b0fa", +"d3f7ffaba43fa508904ef332c8d136bcdaaf78b02f39bd764356f0043734e98f", +"762e993525994f47fe6c4856a6e085fc39a6d1958910096be529463ad466ccc4", +"0e27250947c4752966c4d0d4d2ad7f0199872f530403b06bc8af5a2ba92f058c", +"9e72d7d6b746f56748150add0a102c889f4557aa6f560845d96dda2da90e9b49", +"31f00827fa8c729d986a4f6d1e614010b7c89bd90ea7c5ca6815f76d828644e3", +"098613fb6ecf597581eaf7d9e9d7a5c65979c89a6b58f687b514b52a707e2304", +"9a3b446ab8432f17ec385828b4ddf857f601f484ce143c45ed584787465365b7", +"57aaf6c681b513e4c89a2e49a7bad2048d6946169d75407c4ce96ac61d80d277", +"4896cd2098056578af3292355516c06963776d8fcb7115c53da5e4b9e6a19a0b", +"d72ac7ce1612e1e0ee0b9f9af20c2c57d0cacf5b2c6f66e3df799ee8a1d20c86", +"75300adb34e13214f0ef195fcec89295d26eab98b457091ca12d0ceb8aa00d32", +"2c77ea91432117c5432597b7b5698abf07f9cc8a9f979cd5b7a7009c512d2e31", +"922e60a25fb0b5f77824f19aba23052e8e4e4bd329aa3003a5372ac3382b7832", +"51a98c63d91c29dd82f6819510bbf6a15d0d2e2eec78a59d7610937ff418a7f4", +"91014d045d15da8fac001b4e677e8e6095aa5bf801bea8cef8f8e809d40f3998", +"b02aa2d6a8158f5f6d4e1e669ec566ea4079b87096f2bbc19bcf196d51eaff7b", +"4ce2f41020168cbeea74e584a4fe46d792f1f94bcade7326b5e93cfbc6f9ea91", +"1c89165bf0811a71d664613101469e4315137e42fab5c7c6286372f0ab3f038c", +"e8dbaedc2527f7976c781191f00b50d270b87938d70a0f83c005764d9e2db2c7", +"ec7286cd86bd479efc98d8af58b5b3c13bd5b4733266485cfb72016797409355", +"a9bea7cb3138b05d4b45c906256e47ff208d9fa61af82007dc805982ba5549dd", +"463f0da046415fda3a49d52738fdad88bca8c30d94d644027e3cc043c72e4eba", +"a7ca317aa81981d99356cf6bce3b19aaed7dbbd0282879a363381007a717c74a", +"3b1747ab686c199e1e3fc18bfb2d8c6455f252cce3a6103e1d133c7e49fdcf55", +"913e0b1eb0ee7d90b7ac0e30573cd4c0e25e12f6fc87e924c6f267ff5e31e1be", +"d19932aa222ef2f0c261028dc60f5a2a150d725386425151b03ab9fbb2f10e4a", +"efb3bc5943651d444f2299345e01876e7948b976e0a3d7e60cc661e69c71315e", +"089aaf0d90529605e92de89f303b6daac6a8b3d2365ab1843a06763ff3a61b65", +"615760ca663a9287a9f172e8945a093e9b4b102e9e43d0ebc61547df63c5d1ca", +"5b71faa9dfe326cfca20c3e57f1d8570365f88fce2f835e8f7a8ddbf5f6723ea", +"844180c0a4f8625627a583a4eecf22bca9fca008b9e6ec8e22fb01b677de87d8", +"6c0a84a394234a3f4d575a43a7ace87c8cbf11320c5afb3b275f40c2399dfe9b", +"24dd36b911f4f44654ff6db007316ff4e5afa2b6e060ea81e125fc12a217cb03", +"ae2d18472d5f07759848ef26651aff9c0a358401741e7764640b38fabf3469a4", +"c15068be6e3b472f6d88cb6a6c5c63c79afb5ddc7e7f8f4292319d5f486770b1", +"8100221a73178f3e89eb2f5ae2159a57217701bc70dcfcd2b41b1e0990361c44", +"0e91714cc5efd828292f4245ef9875853fa464a5ac6400816652da269d36dda3", +"836ba1041dd73136db41d8c4b587fab80b840bfbbe4723e1d6fa16c7c60a9b9f", +"d1be6b877e67fae5ce663aec6aeb9656b2f30089c8e67d9cd9413006a9b3d61f", +"681808447f0e7f91dc11cd37c26e3e8f3f42d03e133b12ac3a8b0f65576526ce", +"326d0359570720abcbfbd94c8b2420484b6422f597a4100bf247fdbadf6679ef", +"cf26971fb787c2fe5af1618dc6b8bd461c34513e59cee4b85ade0e56a074bf3f", +"bf12d1867d278b420660fdeec8fe5a611848e761735faccfb98a49f8a5ec1902", +"4f47d16b0911139f7a9dc066c15101452871fcf9fea332b80adc078702f6dc75", +"94c0376bf8cc96c8a728db4ea9abc5b1daf2efb7be05dab3c6bf7ad4f462c358", +"cef7d3a0e0eab7cd1793003c1d5863bc165af3ec845a12c9c5e45774cca203c4", +"52450d3f0d8cd3b5f9147ae2cbc73693d8c0f71d74e0408d7f8af9164ffbf3ca", +"45ca5ba483123c7f3ca5154785b90f32a7189f3286f1821e36ba1ac48f4f3178", +"a97ed1e2a1279358e6456ece5bba564455f11d1ef0949fdeaa25168b377d7af1", +"f3d3d75973a0646d8b93fd27f4dc58c0f27800502dc773186fb3a28aeb53ec64", +"beaac4c5066b5def64a42807784e677c82daab1be5d6b2a8895e824712c5bf08", +"192c2f3efa73ad809cc1e800b03b88cda056945c507fc6aab629cde9a61a446d", +"d774e878ef009be9b0e4b5f48602559fddea82ad860643969158582a9c418185", +"e13327f3e617fcd80183997b742b35429e050839104c6a9762001d3728ad52a5", +"abb37929d054efac2bfa3a27e161c3ea83bab536cc5fd8357b7609cf76e410a2", +"a657f12435c3336e8def2c5f2f5c5a9fc2b7012bf68922aa94e0eebd488c4283", +"3b0dff844c1a76dcec5a6dfaf096346111920c330bb6501637a22a58d854d215", +"c461dc4e0e9b637fddebcb88b104cde0cebaf140298a4382ef94c6c9cdb0e63c", +"9ddeed510c3817eb7f1764f7cdddff50ea5d461af6c413cb982493bc346699f5", +"aaec6b4702067e61ad0f2ce31d308b56db21d7bb5af4f1509be5ae65e3ff20d9", +"82a8960f1d0601bc102de5035d7ab4dbd2558fd6348929fcfffa24b16cc54ab2", +"827ae2d8af0ea350e9093cffdb2e21244e1bd645ea48f21023fc085a67577755", +"0c09cb82b0d16302a4e67f80a98508cdd3e7d66e9c951a41241aea3ec3735901", +"1b25259739a4cb7863103bc3f259c929bc8296406f69bb15d102dbdbd823a84d", +"a9c777de2eea92af61dee0cfb16531d8df5fea9d68757bb86f3ba261c245ef1c", +"278a9746c0b868c3411b826682283c3466eb7a93b863f52bad360521d60b7a02", +"a43af7842b317c82776283341a9047073e802a12ce4cd65cf6c5a7759cedfe56", +"bb8cf5260a7ce66ce410e4e87d3449954be5f478f27e32beb05fa23dbf4b6614", +"a5cabbaefa3e764ded60f74344014d496bb07a56f535a6cc25dfbbfd9f6d110f", +"7f9bc724ff18d6838b0b1a9edf2f3c389039813dba2ea749970889a4777130b4", +"3a124998570182c6550c5a9947b89bc1aefc3c21eaeea91d227670cd27d8375b", +"3ec6d85de8af1626d4c4f324181df7812bff18f2146c493c9bdc6f48d0927be2", +"1746bc146bbb4a126fdf15448ef86191ef644fe9a48ce7a6648ce6e726a81817", +"a7992e04276bb786a9222166c1ef100d6c011a4553d864c09f7b85dc33b4c175", +"239c58758aa88e340d3c90bb354d2c6d7182904e95504d00d94de424c0f0d04c", +"f150324921e7f086290f507a9c15b4a119993b3b5c294bc43c40c7da94acecd4", +"844e1ddbbde58cddd1f0f0cfb828941401a48627e38e80c1e08fdaba1da0217a", +"1c3444aa8059dfe473ccc589077f3c7af11cb91e68ad6e7f42dd8e949236b70d", +"4b4a5fd88c4bc90b87addec6f07175323721de086ada49070af69ac80d5e2e82", +"9cc491797d5fa756094e953f824ef09d21f7db694e6ca95b44429e58ea9cc1e5", +"f0f2a616d0bfb0b4950adeead946180b05dcc8e374553180402939365605d2cb", +"b37b579e277ca91ba474d3423838f5e249d20ece433138db1cc799ae53722251", +"160611167dc9b4c26146db926011d738d8aa5737c40898ed1cc7229b037b084c", +"68f670165f20c4f8a08727c1dd6eb92b394faf04bc77e90fdf3da1033bb7c3fd", +"d33d151dfc599bf9da2d862c6bcf393df16edb7b223285d2d4438e53bca343d0", +"7a28f892c841766627de275b6f74035eb047a7de0c6e43aa5aa0c41ab5a3759b", +"0a8ac2a36c0c4126424fc41fa68eb69bc5f9211a912e89d5e8174b5e27d13cc7", +"08cdd7c9ae5549749aec7669cde41cfd3350a168ef14e22ff28ec57e2e800b54", +"656ff194ec9dc387b3989d1914b50d212c731ca0c752062772c55aa101b72fcd", +"0690c27ba648761ef907db118873161d4a86c1355c875a8abd699bbd0b75b155", +"cc2158a7d7c4d7a2136d6beeb01a3149a93cafcf5b13f07d4df811a23023899c", +"73352156347236ecbc2135509934bb7da79777cde9a6edf50bd3d741226dd875", +"80b657e3d565bfd82b87fc9dec95c4915bb969a449f8b7371196667fc69dca00", +"23db54db2353862ac26ebe4df2eed3cd92afa9d11699a3cebf2dba844197d502", +"bc288f2dd6357e905d87f08b531d949d982221dc77d316d38649f70487b14484", +"0f66d567180ad91a6cad0ae5ca20ad91d74f24cb06467d9b628dda5a5a4a0651", +"a79e61e5456d0580b6b010ba1c711404fbf7a6e9be3e36270014f5653be9ee40", +"419a96c8085276e9e341cf598cea6ac6e524d11bf8c63693c5e20b20a43a80f8", +"6e41f798e7ce8eb88f2a0326bdeb96148ae6071acba35e4b5a5c981f0a0b1dfb", +"75a2485b07644baa16e610b7fd06a67d6b9e884425439ab217e89e3e2e393eb7", +"2811dc3d1be43714d6ea9a41109bb90731bb83bbbc14919da1e5a6534a1f663a", +"ee276d2dc2a8ce1e9dfa50104c66efea41a893e3980409741fa3c149dd41ea40", +"f802785b2d0a170b2fe5081469792bf2daa1904688bfc0518a4e70511b641713", +"838a91750115716cd9be0aa19d2c466eebe4bcff2da969521e558e74e8ecf065", +"f03d62b972572008b6f93cdb6a15789cb9b703e32786da8d9b16029f5290edba", +"65a1235d78ca22063623735626e962dac4b5de67c7c5937685446f74413f7459", +"384d7a60507c144e6d622d7554b72a26a5a95572f1da765aa23cfa9548c33237", +"0319b25f8aceb5b26f50bd2492e0b58cee82a1ea363764778a75396b0d8806fe", +"d6208844b3cdcb1e47972002e5017cbd98097a8739713b7f660271fb821da095", +"90fe4aabd2952cec384983efc97cdcd430768b8f8e31992f8bc332a40af48222", +"5c940b126e3e4e80148782f32aada256737ca6479dd9fe53c74f0793f0861f88", +"b750d7a508fec57e5cb93986cd4367bad1f3e730e04e7c63cf4e6234ca4024eb", +"ffb67305dae82898cdf86830be6b416357c69ea1bf3749d912daee0b454e686f", +"8749f80ffdcb60630f9c8ee5da26565e0978af401c4447f93242d3918e45e83c", +"528f2c1b0cae1056a1f70d8c0cad7f47a30d75aa875a5db35e0a0474e4e9145e", +"21f6ee607f03d7e2d42a9fd11cbed242d09cae0c1b893658fc2e9a0989a4f470", +"89daa601b73311e42dee11cd34c8c16e2432471bc8b8f3d27a413fb4c59dd6dd", +"ba8f6e6fcee0d6ef74ff123967c1cc9d6d3e683ebd1e9adb099e2dc0929f0850", +"ef45d865633cef7768a05b1abdd79b9102ce0eb1aef614c7c3a7d8eb99f4964a", +"9a8339fbe8938cfc20b92e42be5135cf737fbf40933ee4f4614ebbadda64633f", +"746e9ba93f03f0f61df3713aaa8b6dad9f51cc3bf03cd67a0f8ab1a5c48adfcc", +"2a18f1e473b8f46b690cbb73f3a8826a97ab9ec26cd27bd96a14be86d55592e3", +"b863edfc7bba0538f087fea2c8763084bcc5b78a85b9190f039bc21d5fb4077f", +"22c09e3e63325f7e59d179503dd5a5c8cb47280a65c1804527e269f16d64257b", +"32868f50ec9960ecfdd38396f39eddfad2308d41f9c0e953bfb34332de83c9aa", +"6bfa6772064ae23a3f1f34e6bdff625582e5f8fe3562f00bd830e61358fb4034", +"9e0c47600ec33341362a32de8980709e1bba02c90cfb13f5e6f5db613a456f1b", +"59b8d543c4e3ab3dccbdc5afac4d11f6ea9e10569996f71c0a69bffa4a379cb6", +"a1f7ae3738a4d42bdc82805327c989e992590f666328b617be24d744f2c80d50", +"3cbab9d026c045b5322f17c0fb7d6575433343c63b79615d8bccc8837992b4dd", +"c083f58eec9aea4597267f71bfff9864deda483ec8aef047c5ca3ce79ea23bf8", +"261b872085835c90fc6f61c9a27dc5e40e02f922a9bc444f9c567971458df508", +"8ecd0fbe1a0bc6f32ebda1b99459ff5a9e82ff2733ef5f8b0420755659ccd109", +"696bcce1ade2f87a0690541f4e4e49a81300b98fa2b5505959c9b15223af8248", +"35ba7749a13eafcfd415c2d7ce4b08f3bbb0d72dc4e9852bbfcee181af86d5ad", +"72023407d0df8ddcf13f937491ad34554535d82b6f3c196c5a8941c7192aae4e", +"aa0fa1b01d03e4e3414e68897809c9303784566537eb511bd5a1ca9b9e93f0f6", +"ed8b98fcbdba093c32619cc68481c7511f0e80c1eca5c6da2dfef7fa0afa2663", +"cb3159d68e0b9d153e82570dd295418d73a69074a53878de896b13a82d4a87a9", +"f6b5c4a9424fff38f6dcd4d1f60b5f0feba7dbebf6b901d40e1907e0fe79ccdf", +"adba8f453d7db198bcd916ffa9b5d28b405ded23e684034a0b468de798e1989a", +"e0142419f42118b7731d4f8e5fedc15db9e80279eb5ef9bd8e1c52287f23d250", +"3cb68baa7d424517cef3cf785191f4a927f75bd5111765e24f541277b235a522", +"732409643c0f16d2989d82f2e9fa128e3c47cf5767510ae0076da2de9257d529", +"e47f93b6f682b6893010788aafd9046f411cafe2ee21bb351adcc14f8d479cf0", +"b4a8281c93d067a5b8c59f4f3161e542dc9175689bc883cffd3f8dd7cb634c66", +"865146eff87493d955018723da8ffe2a83473a8432a4cfdc673a712d15646c6f", +"ea84bc8af5740c0623e3a55383eb667e0aa45e7f72ce2e22eac1f4ebd4dab4b9", +"c878473a1c2605b8300e402bbc7d6d46d2bdbee6620171eec20791e9a0581e61", +"7f91eec3b26934d7add813437122aeea8bbd34f6003ec520544db0a655a2490b", +"9314fc64e7eaaeea0a1062745f600e285fcc082f6183d2f719396e5b7c2c6615", +"8364be172ce150337eb679b9b34a26692e6feb02af1546248613bf2e36150769", +"ab337e2c57995fa8267756e76770356d653887917210761043968f654b0d387b", +"936bd887c8436f5a08f6676754e2510ae99a324c6f0fc6942be813ca2ae2034e", +"e8a1c5d41b6730af5bafcfa8d402126ea8e3bcbdae8292139ebce25ebaf8b152", +"cd3bec82671635035091e03e9d24dbd6590f8495dc9e190036b70e02756db4a1", +"ee4751da17607f9802f355f7c5b2df169660c9eccc8157bdee610f78aed6df35", +"698ba5a7447a63aae6d9fe00a3124cb15b38da7b1467ab2930436f70b1c102b5", +"f67d44949634b892b2fe38c36b8991e94c808d67b818cc3203df3a65070b63b7", +"d50105061d189373582a6701ff841cbe43ff92e92a8f24550929ed50312b036e", +"367509085f0347369b4424b861471836f951b97444009aced38283fd987319e4", +"6b9b271efd11b12134f99bba51a52ac9e31924ac848e99806d86da9a5caf867e", +"1c458f8adfb5c24e6219e920287a37b0d96f812666ed48e0f287868ed38319be", +"3b6fbfe0f1a988161cb64a7cd4c9f065ec78913ad0cc60163a5f6810750d2286", +"298594c4def5a7a4c66864b1ca399b4a8f9789775e2cd933974799da178e1cda", +"f78506452b9582c177fefd3e3b3d08d247f6bb163f01e5827fba55e11e1dc153", +"336e6a832b0dcb3d2e7df5423002264710c983dc6853ff80cad2b5cb7b0414b8", +"c29939dfd53aab6da8588e79af11bdd4602c5628c0f16d547eae72a654298a26", +"28c8f13f9cc547564c8b577af1ac991db4e2e1735906e9187823b9010864cb6d", +"f04f8cbba9c6391662107ad4a1b9eacbab6ef0f7811e38fb0e1f1ef105ee406a", +"a3a9aafc306518c89d92c6296376c64c8589ffc9002ed44f57840903076ec7e8", +"68682de7f75ef220e63e5c26b13c4f831459f23ed40c5b3340c389b6df4f348d", +"5b9cef5284493e120b885e25859893ea9cfafdb979997036a3fb46916d642fcf", +"706da21f6d8d90809459a27f510625ae05c302a2f286fcd052b1b1816e368f30", +"dcaed1fa71f8c633eea1b44cdc74a05ed3896aed3e3ac67a9853fc48c67f20bd", +"4dfa573494f6274e088ff373ab905e1479571f0e599e04d559b08fc49f3ad947", +"4feaa74d3d0cf5e7dc369ed6882b21e211009404bb47e27cf20b5d53f02f4085", +"761c73195540b544eccedad184c4effd2b0053fa84d5ae4f81ada45df697d062", +"113204fe9135bc3d3008a8b3ba444f3530546f299774129e0d1799fef670fb35", +"a8ca117ae3556e2970c1ea6552f8de10b3d2d3c70d77d1452c38db53cdac73b7", +"dfbe33121c223233c241bc67a7325c1c2cf62fa78a19c7a9e4d453aedb7ea34e", +"9bee3ce928680e7ca082a139ab23786f0fc28acbf23022dfbc073c7940dcbdb9", +"1f915acd578fbc7fee5079b76e1dd31b5748b4985dd7878b2de452bc5d3caaa9", +"999797c6c679f4a0efb213fad56220df3ce9e06184559655930af3243da3a0f0", +"bdd5c8d0c2852b1cc69816e8b475fc0cc9dad0725ee623f4d3cd88cb106722bc", +"c38230ae273c88eddbef1a0b8650bc95fd00647514e1f97ae4259c386d230b1c", +"d40e8ea57179728160e81ba8e70452dbe69650ba2aa55e2a52d4af6a3d0869eb", +"473d72ddd3f89d2c29d0dd9636a42a5df01905b5fe89212015cb7484bc43159b", +"5427596611e73a2494666e337e30fe00a1cef14f5517a9dbe62224f3cda97fc7", +"89413b80b074fc928d5482269e438739b6a0211654bc31833c1d12e357318737", +"1ab845439f795f33a0a505042651d06d6bfe3a82cd101fec87afde8cf1359fbd", +"a8fdbbea8d798e25037d3760e2791d33675d245a919b872174b9a93daf449ad9", +"45fd9ce410be565a54dadd68a305497badbb911c7d107e585a616de51f62d35d", +"f8d40dc6475794d0104b1181bcc6d430866361755c6c9e65b1f851b9b445e612", +"c329b7fb906eccd869db66e54c1a74061c8d4ef2bd9c3ef99d1aca882ceb4401", +"6c1fc4b2ab8d7599c9f7e055bfcda026555cd2c791ee66c7d44b5b2b0647d4f4", +"6792f2cac05a8cc7003f85dec01081ee05ca421ff9f7db06e213d02d182bf68f", +"15f212bc09cf72c5b3c8e882230e9ab9e139045f0a4c828ac97d669c7dae7940", +"84e0603e3b52bf10844fffb049e0f0cada2eb09f23a6508502b7bddb097a7443", +"07319cf4b9a4d5bb01e5f47d8bcc6eba7abdaa1d5d28f5bb68a576e87b84a8a4", +"92d89fe23bb0f6bacf6aafb223b11ef92d19ecf13fa40b520e6ae6d3cabc7c09", +"4e50503a0c54a46edb1fc5b1f08cb34c0ee3f747a6835f39b080828dc74ffb53", +"7e325298dff9edfb5aa43f8e390d42c066a5032cc526aee746c6bea49eefe6f1", +"10be5eaa0d95c87e363d0466d85815590463d808960bc0b02f49eba0a22552ea", +"20ae5277c2242b2cb34dc1bfade488ffc05286cf1222b825e98fcbf4d73325b9", +"f06f753521a12846778d9433cc283977687b418651519bcedd858f45557cd0db", +"7ff7fc9b2dba107764709fe07a2a33767b68afd248548cba9b3761898cda7aff", +"3920b85d7095379fc66aa854372f6dd0bba86821ddc1ae34c600f3a8f747cafb", +"c7f94ecf8cba8b421afc55fd627b04ddd81618e05f84ada21e13d23835d054cf", +"9005ce579a43b1ea968532718d2df5cef8d1d9d0f63ff2571ff92fe3ddadb156", +"525cd797a08b9138221edc1e76e9457ea7f972d93ac6c5d644093b1f76bda9da", +"df473259f8f64c555fd042758c28301c96c05b5e076c155d76a114256b88e8fa", +"cbd16c3072a65bd78bd24e8842634293963f8bd0c23553dfb27bfd0f800b3bd1", +"89d70619c8bac49ccdea66141c7e319f00d7577e0236496b83b122899a03c860", +"e211d8b51ea9de542cb343ff475cc3a2cff4f61c8453b3b3ff38b5ac78fd0230", +"079ee68fac45b787c679fa60d57d9b7ab18bd72e77150d61891bfe50bd704263", +"b5d7861a6b5323f2eaa60f80193273219b7d51d26a1502d9a42b88a3ff1df36b", +"55cd37e46ccc0186c2ed4992a19f88882e136afb8f2df63a3b488218ad946aa3", +"63c378ae5da00acfcf8777e4961f1e19d10c001a4ffebe6f6787082eaf069ccb", +"3db56d3932d3b267e64507085e2e08f110f402614dd5ac4e712f7c4799d39939", +"71c4918d4b8f216b7bdba9036edb693507441ae36b06d1cef89227e20449b027", +"7284053e63d6ecdd6a1fd7652dfde9879db49e373a9100e5f0e809fc321973a0", +"18af62718fb01ca9be68d91ae75c8265fe9659a765e3ad180d545b762f1dc0b5", +"a14f3dc4df4f5699f98079d5ca1344becc3e494c83a5447c117770b0f20eb2fa", +"688580b38dc1a32a68a06266e80fd907435f075b2aa6c0623752ab90543243ac", +"7d06d0a0a5b620ee8b24a6e02f30d68af394be9ab29fd07b8265c50c7f28ea95", +"6386093518a0739037918c0d6f1ffbdb539aa7b8561ae2910c718113cf77a256", +"7784205cdbc3bcc107e2e458c9944e0254dc2fd7d7467ef8613816363cd46df5", +"e87657bff6894fb0716a6c53ec84095885c2e965ab65f12faf404e17206d3dc3", +"2d8392894196e39908eca31f2948247aea2441db010d4933a80a9b232a7b115a", +"5c983d13074006f675a33ba6794158c6ad278bd035bff1f9acf6185b9d7a1c58", +"9d175467aa29dd3b5e8728e15a6de49b99b762a24fa0648fe57604e4270675b4", +"60140c4d77f2f738db28698b49972ca2e46880051ab4e156f40b703964a7ab00", +"e19105cb5169fef6aa83b852317286f073808f418c69311b0ebe406ef87bbe19", +"3141a7488f918215edffd2eb130ea00c7a8e598cef6aa4d0cafae9c8bc1b951e", +"a5e7c8a504837f3c6d3da9133f86d0ad99b7068b819791994d0f88cd9c1107d7", +"a625706b3edda1762a075bba5299ba80e89e657961d2c828a7fc3008190c64b9", +"013ac4b904839fbb04c588f3da320e8f7accab6673a1a93646511faf653c240a", +"ebcb831312463f4d2661a000367c3e887485003bc9c90673028e8381f0c4eb22", +"7a71c865f58582726278042ebe35dc0a8ba36fba5fcdacbd32765d5322c6e766", +"ffe82e75847a6bf57be1da82ffccc7c3f59c987bf92013592146264c6907db90", +"347b14b8913c3aabb9858d076e0ed6ba7b90aa30f1fd1466864a54b258ed4a9d", +"2ccbd7c89259ec7955ee27d1ee2d2476c318ea558369bf70f0fc891c046bce6d", +"3fb12d1e75fb9ff231ae18839b4eef27e60acadf27353597b10ec7eed55ebdad", +"a60ac4cfdea2a39e53241ad5636f993bdef01463e81082de8b41b0a83b6d8654", +"f59aff6c192159047d7e64bd2bc62f425752d0a37a35a2b708289b4d419ad773", +"b6273514b4e0aa8f3d8347f5ec9a28c932cd0cab3bc1ffda46085004a17f20af", +"8f1b92becff113b726c379efc313abbf69a70894731dd50f88c61f60ec0deae5", +"b4402ed8a9e2dfc1eccec4c3ba65514db3dafdc1b4074c623fe34eb23ee1152e", +"aad99827508566c232737285acfc38d8c9414335fbbd8a08d9b19b8fc0676429", +"0c5443b1530d7d52653a1c28bab0b876f71c15f9c99329cf738cc95cd296cde4", +"768168b8b056017599f1532bc86c9ac482699f7796f43d096ea329825ed91c5d", +"a3b52ca9f7c5a9b92ca2999f7591d6562a93be5655d619fc4869d8d8a5e2bb1e", +"9137d59f9fab575066c34d0c8d1adbf06d76873e35264068512e6a001106ada7", +"ec5a0c6fa495873497640c0065e754632064fa04f145481fb2e039ba7493f5e5", +"09f0cb9e66a3089d8216fd855713078dce9b8cbab3905c63d2771f26f26ff186", +"c137fa9351f1d335316d242728c8d89f3e4347381696b119f82ebf1cd96860c3", +"379c9c1070725589a0fa9626a59bfab6de21a50e600815be7d6af70edf059f3c", +"7414b2a496fa424e5c0863f36fbf4299ba33828480dc75d2854e1c008f837b88", +"0e30768d6bdb4b26f6076b3bbd9526ab9d3a5784515520539cf18719d48d9568", +"63f09677ed807db69dfd95d5f9a265318daf569a8e4b99c704d1f0e7a3be4d6d", +"5d0e4d129fa1b5d8d26d9b9965b812020360999a6147b707dba37890bb2696ee", +"98622237d4e7ef1118f685e1cfbe8f78937b20ae02dfcd3a5859d6a4f1a08c81", +"6f9558627aa174d26b5c33d38cd7972fd1fcd7050313bce8503aa9ab62e8ba95", +"fae03a1b8eadd2075e249e8ff59ddb3ac07d267b4bc54ef2e8431246c0139c94", +"21c0e2c00d337d535ecec130bf4c059e30410ea58875979829427c54ea6f760d", +"6f6cb09b7f48eb6d579af6156c71e51e662a5af97f7bb7a5eb9ce1c9047c4c5f", +"9d50f89b92494f584adb7dd6ceda9c7905b0e57ff7c9c53af0eeb81cc3eb95bc", +"a8708f574de82ca496d423629b3d9df0644ef07bfb8f43e5ecfdfbb79afb5081", +"774c908f167d44cacb2540ee143858bcf8ae4c969d6adeb30fff0c256bc191b8", +"a38011cc11f7352179eac9e039152e00002e941592290175fc8c30abf77341fb", +"703df70ca47c3ef54ce613eed28d9f303113f0a0878708eb264892dfd863d1c3", +"b82f1257c40ceeb4c2a8cd54ee972dbf438f9feaf0052788f3815726642fdc33", +"9f7aaae569a8ca4d76a17f3a48d85cc1f926afa47ec0eedc829317cc832b43e4", +"1c0a1655ea3393cf7fc15557a248c09ce11df06453a130fb2a9461ae69c533cb", +"7b8756df16cf9d6f6ca5a32eae6ad70c3a0a9d176116559380ef84393cb44e69", +"82ecc4b7cf336b078777818ae90353f1357d058cd46812811c4a0930c56377a2", +"3ea8889ee4c95fb3331a35771a3cd78e8cdaaec9256c959ca9873f71a06e5066", +"2bee2342b541805242d28b860a6139431bc02672d34550a909ab85a455b8cfa1", +"85773081178a29401b1b3ca794d9c5bdc4b848d0f4cfd69a7f98bbc1497fe4ef", +"880f6862e2a492524b0b92f5768fcaca1a1f47db980288d8220523779ebf17cb", +"2147191098d5f49a09c8f647b00c5848b8553960b2a9e7df67be64724a561519", +"5cff2d191e70272427fbcfc4941de630b0c4afb41f230e41645a0a49d29017b9", +"e4e717b41ccd44bcfcb4b8b955ea847bd5f674ec89b2ec88ae3cd795ca1dc2dc", +"09c21ea5589264bc17ae372fea55b526138deacd72fe2c5157a2d4d076a12c20", +"2cc5627895ce8ea7b11c9f58b48c8cd14a962cbcecc508a29d9e794ba22de572", +"b8155da65549c3a4ba7459185cb1d4dfbc5c2c2bfab19c2ee172140b459a598c", +"6f818bbce13e7a6c7ed786484ef58840ac1b7461922905e7a1ce67f4fdf83c07", +"a756c7578e7c7714bf306cc5d18847856df650b635f08e2dac52cfeae828a2b2", +"f4e8459577e7008ec0445f378de02310c1a460473d63f7223cdc29c1cd944486", +"c1311a7f803f80b441e4395f44c578810e8ae6dee64a9db68a2a9cdaf6885d21", +"f480d78c920ee279f3b50f5fb06212b1b1ac62d7e21afae7bc6b2982bee2cca2", +"f042eb6320bb69942c5541e5acd8f64565ea956366d5bcfff90a807277233754", +"5c5ae6122a1931cab34425a93600a4fa919c9aeb0dcecfdd315286967982df57", +"586663c789495a0ae8d4aa255e2bac33b23cfd40a7259e90021786d62cf558eb", +"817e1591e4aa336dfd0bb42419920cf29832e62b9ec4b07daf023fe55ca4cc6f", +"2691e1028e87930f98bdffa54397897b1102f5e8f720720545df853b0b523c26", +"37c023be7b0c326770ca8c57de5bb86b5a802869627a2b0db966b9cb7a610d29", +"7f0a91f2a48a093fec14b6c6af2832531cb57cac7ef21152aac4f1b6c800cc95", +"b03141821bbac69485b9160261019aa97cda2e7c0d0580e0a4cb2a06d9a78bb9", +"f711d2c8dd8060c66b153fd5914ee1a861279ae759c43a6b3ad3fd5c8a7c6754", +"140bf8e541b3de9b4caaeb32f2b670a891f96e5ffc46bb860b1eba1db06691db", +"02f15ca84bd65b8bb6fa65b417bda821894a01059dcece02a30cd708f1b9066c", +"86932bdd2d74f632800d703dbc10c002f70c7e4f96b5162d388286bd22a3c4e7", +"8f6e2bf4c9710d22700b88755efccee724fb27daa0e27c106743070cf9461f50", +"caf42ad21f31c12de754242ad6b282a48a8ceadf3242c71d6192385b238e2fd7", +"baefc8b06f7965221ac8f2ddf3c73c0bf40d297ef34ba57fa9c0358c8e83fdc1", +"d59ffffc8fbc1fcccd315d954b945e0e0627af266b1272146383563a52e3bb67", +"30490dbb2234b1e3ff5f0ff22a25ca5ca277fe574f8cde80d456511080a1623a", +"dd20afb04e19397cbc6d38e70eb0eeb7afca7c775ed65c7f8023445c2c3ef353", +"fb0d306f8957a9512e727e3462ece44ea1a1f0cde3d11d9cf796aa92cdcd3a44", +"7046f234639d52b16a8162be089659e1e64b77038537307c09dfdd2e5de2b2b9", +"bbcb14225d91a9b91e4443ddc4e30bf9f950c6799833c4af363b74afb0668511", +"4cfe31007a14b52f8a91c7d896ff13b22d043b6b943ebfefd5589a8c5d86a1f8", +"f968896da3071ad93ce32c5604037321f20d329ed8ee7460e8bc1eb57699cc11", +"89431b8dc29ccc26985d898dd7aa4907fb287a3fcfd487393ee789e0daa4a49c", +"e443300575cee6a868c1599fd2fc76e3c689e544a77e6e85af4813824f1664ba", +"e1eeacf84f757e26d9a7a5074d4d7719448527f8b46ac88c081a52c98338e9cc", +"4b6ccdd0077738f002825ada3ff4c484c5a7c253bb1be1fd106b100d54fa994b", +"75a7d001d4431f4322ccbeaff6887a352912a745fb6fe62b444c9b3b73e88c83", +"91ad77f3dcdc62bdf81d147ea6bc03a1c48543c546f92612276358eec846ae70", +"75987063b20d93da9c44ceedf236ed9f203a03dfb8b30192501d75ceede38093", +"281f4014eaf8a3d01044382e57e0e3d2b03a1dba8c7da747b456470343564c51", +"b48fc59415e002664f633f2563786e80a89f369af0833b7a381cfd3302134820", +"97808dbceed22294e2ea05aa750ef944a60775eee96e02d28628dd0b115bfc3e", +"4a19454e927b2236bc99a3b116aa5322a4dffb4dbde8263437dd25c62ee1fbac", +"9eed3f264441982e6160fdab9adcda3a849d06f8c35abba81844e8ba2aa5f06f", +"d723fbc182eb8927878a86fef238e0293ad6c42dd41ed020d7b6f26fe224b3fd", +"995894adae1c301c9edfb54e0166db3787444a057b2407a33db046a4ed9a1059", +"635b5ff06b705b4af8296be77cb8273e6ca8dc756747aed4ba41b6a2a1fa4681", +"12109dd8de5f06839fbca58904b1113239f9c6b9e735e3c0771b8ef2cead7dbe", +"97c8ba54f5b546d68f38436cdee49991e1be80bc273c74473d445c5a33af6eda", +"1cba12fefa5c2b33472768317430074a5e3dc9746b44601eb2dd633b98803287", +"a16e5b8ad9206b62ed761868cb054e9deca8669a4456c8c4d750aecc7985c278", +"470e4221ed1035f2a001a3570f0c0448c325308963e5c49b630e1d7b0970d532", +"f5711470ca047d62872826af9ad44e3ec54386ca78d221e712dfbbae7e7d78e3", +"d4125d976bc796bdb29e071a6b0db0f024fa77499ebed03f5115f2864e16bb57", +"035f34539f9e91fb89cc28a20e34ac2fecfe32c6b488dca969c872e8a8b83889", +"955f8f4c2917f58ff30ea7c8a5a0a07915865174d761aeacb5bd38e5dff4b2bd", +"7cc23e8021a567fcbea9811f55d734f464d101249adbc0c9ab8864c902eba01e", +"16e3bdc3aed922dd27d1ed693e3568be82304adcfdc28aebcb815bafeb435854", +"f6f858dcec7528418257dceb1ba1b76242854aeffd4af8c48c95496071458a13", +"594b52dca9b06b6b5aea3211ac1cf5705783a7814d1cdb8d5a88834b84ba208e", +"950935223f090e4837b54e5d99e9223756ead5160993201e1883ba689006399b", +"47034fafc46dd2de661bc712d78026d362e05deb800fac4b8c4770fa5c45d349", +"4ac12fd7a91a336864cfd5f2d68b03cbf587be1f2c0d799d0d95fec2b4896cda", +"172ee74f2c13defd1ab4fb05bce2a2750f3c4d0952f8451fb5c736268bd36ff6", +"e01bbd0b324d1bc364e3061a4d5c292cd5dbf73cd2f39907ad8e000706186d10", +"d2165a3dc4f46981795609d095fe8b2aa8af73c59e72c3c823c6cb330c878375", +"8cd5682b6ec6f23904b3f827540d09680fb391b446494eafde0bf1bf2386d8a2", +"9168d29b8378f3dd2f062bb32cf24630c15819a94bf030a4a38737eb9c206713", +"e139a677f3c1df1d6804d5966f259b44bebefebb2f39f024d969b6388983acd0", +"a4566c29dabbb050d825acb7dca9a223df210d49a6896b6395d6dfbba0ddda7b", +"324e5bd3bedb728774c033194183a904ed80ef21c4e44aaa22593b728cac32bf", +"c3558e7105c25aa2cdb349d76e21b67aca8f8c147fc79bd31f23fc4d37e2380c", +"f8388f594f92679931b90fcd0111aa94b6d1a97bc277d543b841dee3d0dcb29c", +"66785ce5a8d57ad364d8caebea4d44777f6f6d98c9c1deb473f09cdef9048d63", +"26d7f755d0ab0e344ff40c4a3214108cc4f3d0f2c3f1e6605e24cc0b505bece0", +"e68ab8d8d03dbbe90d70959a901677c765e5a0217681a70720ead4c92e69f905", +"517d387a13cf0ac6e6f67580c63abdeaf3d2b1aab71606251369a4c62a682504", +"e2ca6f1b2cb0217a97918cadc778464843df5bc64ce53d5c316b51a88caba2ec", +"9a82462893ba03430facbbe10da9bd40ac580de81cd1cc1481e2b12040fd1eec", +"28aa63e4010e7edb1ea207cd0ce4894999dc6ffec2b6ee67650a7da941818aae", +"f1f5b8ac2522bede838ddbd30f403daaa718fa1717a427a4a55df5f13bced40c", +"6e07b825b8d9cebc2e54754e2d2a06260b65a4d3b1c8e75f3da3fb496c295c7c", +"71186c4afbb705ed27814f529739dc6e139aa6efbd73b16d47ca1f4d3f78aa66", +"47973698d9a0e0d4238e0d27c3afaa9aac5fd5eeb837548ed46e879b4fe58951", +"ab908c64d8073538d18dbc5ee2bdc05691f359cf0d698a39fe59680f70a64439", +"44f92f961f7d7dd641c4b3ea991a30c25d544b2ddbc6403856bec46c052b314a", +"44734a9c48b60cda6ebc42c26d6223e978afdb2e6506f0233e43041f99eb8db2", +"0dd54cd79fc5907513a86f9e11a7ff9c5dc5699d12513659f57212d689db27bd", +"dca86c04be7be584f77c1943fdaabeb2b354b1a3108d4a0a58f9b8c0572682ac", +"a1911651f37f490a61eeed6f8794006bbd809a2a6e6813c9e759076bf79f85f0", +"fbe71b4ee87040d1713657a57cb25d7b6df042bb9791f251892fb4968afb7f80", +"c4b069cb4eb9b22e0e98e19e3d6caede07d9ced397e04824c5a2d09d09ea0faf", +"12e6eb840a43904fe5ce3421511e46d860929e1f72a285a3510a6eecc4b671d2", +"a1240d6c584933d4f8de2d0d3bfebfc3adc78d63231a51c00a95db0d8235684b", +"18b703c384e73ded9e2d58fb83c6b76ceafcf743a4d757a89d922b64c59842d3", +"f01605ef7805d33875e54cf0e00ec4da251065f1b289733096c66df19241afd9", +"0c2aa05afa540554f76816b3bccba52fe655bf89b7c1baa0dccae3018624edbd", +"6c0fd58e39eb116c2a77ea129f3f2eb76cae47b2a202299b37ca6e0795f6e6a9", +"86a6e2174c3e73cb2ec30cc8cdf51dc04068cd33b9fc33c7e0fd8bfe2c5c6d94", +"a35d7735d0283ad84c52fad1a5e1e6e18655c8605ae1f780038c987b5737fb59", +"6f044006e0f8e505e0ade1518a4d48b29b60f5c3cd19e9d36ad0e27a9a7278a4", +"8e679ea1b4ad3c36c57cffaf9c81293ee6f30dcc3b356537480c973f97fb4fb3", +"7054852442e64431b1764b50666194bc04ad70e4a80f29e086b61c5b79e1160f", +"a22cc7f6ae35ae45c24c495aa46ef64d0c2b66a914ce7c21de38569266f3c7e9", +"8c8a89aa60d22d0342adfd66dadc73dd9481a74fe9fbbd8b65e6ec15251e131a", +"7d60d4e0f52d6bc73c1acaee6fe332f247f6e6adb84991a03122e3ac849c9250", +"dcfdc1e270942a9e5662883124afe0b531f724d3b2c4381e35d7e1c3d5c739fa", +"fdfe351d0f5b651456397520651fe34174ae571ce66f2e3dc76d814485ce19df", +"bdc84807b41c22bae70216d341cbc79e13a5258ffe518248ffe32e6f6cc54426", +"28f57a1a7d15228447f929cc8a79d0bfc646c237800afc675bdcc3c3bd441582", +"4b3e7110c0d2530df209fa6220cae437687de65c8e778fb4378f09f30606579f", +"1bdfa9f0e90b6b36badbb6e7d812a594cdd8c8aa907d8c2b3822bcfc8e9fa3ee", +"c28386d491a5fdccf9510419bb25314e110e51e2a2fcfb6890c821e7b2c0a4db", +"1150dc02fa2bff860d6090e60e05b8e614f8c4bd0771ae0264103ffede3ffbac", +"aa178b74dfc2fe81b24b773c656e632dba1892a7fe5c405dfab86972f4b967b9", +"e5c5c5120822c99307966363f6f025c229130f5fcbc0a06f58b5f15b7d78fdac", +"408081b83fde61681e3398b17f8c40b7f40ad59c5378c627bac692a8294457dc", +"b335e0400d8ae441941a233a9d6570883cba9e9e7ff5360e79fb5e944cf4eb90", +"755a87bb8f7582daf7b0c95b1dc403d00e8a758319409f0e68345ee2030b9df9", +"06b3e25ed585e703b959d91517a79a1937f368789a5006e41496adc614d1da11", +"eda102f79c40e83233a35b4fe49949873c68c1cee341bdfbdcca0fa88feec616", +"ae388b0eb6afa50ada6d7fd8cb40dc3c8b433dc863544111a792efd8d66c1e60", +"67abac191dd421c0b0d763c3436f1f4ab2a4f04c5ee12418ab28c4a608f8b7ed", +"ef20b902c3198281a247fadf84811e1b34aebe512969ec7f396974bbc1b27c8f", +"c2be8e65cbde6d22201bb536ef16aac4b7b5bf9608f4bf233ab92f4bbf65161d", +"1bf7e9eab022b8fe6ccb6be43e8732fe9f758977e492d82deda6fe020d0426a5", +"710cde0f9f0fcf5a61f94b0c89f95adbc85365594530abd4bd9fb7ac88bce722", +"dc1dfc01351ddad66b748a5d14273830045dd98f5da8ea31519bde3cd64a9e58", +"9ad93a390637940cdf4aba5af19b2039546ec2502340a952cbe2bd995caab163", +"e11f7b74d4a92600fc30643d0b23e0a40009fe1c92197d95824207c75c81e15c", +"cfe1b508511f26ede8468852f80f2394b92a109628c01dd414aff5ad2fe629bc", +"5d700bf1233e052b1504ab4ddbfd13a8bba907b9fffe16d1bae172e90e4dd680", +"cad5fb251c1d42fa7c405aca8d5c831cdca0c32de5cc1c579dd2feed00cf29b0", +"ca06af69bdf6c05fe07062a25ead6ef2ba332492b56926449e4d1eeaba399ede", +"5a74d5a8d4d40696780346512b3070cac61e17e1359e9bbcfef81f7706a88801", +"72f59b18126dd3391a44e9dcc754ccb9028b327e6f9d111deae18480de7d6fbd", +"bcad0a5f6f5ccb31fbf1fb451caf13e1fae721243cf1c7f8a456c8b6735256b6", +"4a307ff7f885fa8059999cc09b7243ef152f3e24db881c0e17abbd23b341c7a2", +"33c1f32af9477a8f7c548b3bbdc36f1e8718f51f69edc2e188651f296ef26ab9", +"1c8198bc9c979e7278e68ff1d45ba6b5b6709bff19a82f62b950522b97079a04", +"e0d3385b591316f3c6e23ed81d8dbe025386e236d1a1aaa5caa6ea216460c8b1", +"d03fb5d8424c9270a05ed1dd6b8a94bc8dfffa364dcaa94b552a50f25bdf3fdc", +"5375d950c2ac0e336844af01d419cfdb7637d7feb548512037adf8d2c6880c56", +"1cbd72230995f87cf262ce4a85a69c737e863b3d67cb69c666a0295bec5d07f5", +"d6f6f3c416af59cfb6fc6f3bc789cde377667984e0605264f92df509fa2bc575", +"e719e9b11201beaa1d8d0b8d65ad070dfdb921c43e224283b09eb09352421352", +"d8d129d6e734c1fb8483c7745677b1d4504137c1318785955054f432d1b44f57", +"c56ca3d8ca8652fdd942d03e06758f3fbdefca6f9b961d313e4acd3a7a83659f", +"4b88c8f2828749f49227e51300c3d008fd73d0201352c4602af41f57369267eb", +"41c97511fcdbc21d62e84104823028ab3cf743f83d5eb0e13742e7466ae5006d", +"30aaa591a9bbb197e853de5c7959784d9822e030bbaa46207857f6cd4b0ba80b", +"4bc8885e02aae8662c1717a198a5b5e567ec210c963d7e773b2e0d0d38d252dd", +"bb41bc9c730b1a15695c8bc9089cd5f1c192fa76ee88552a109bcd12ebf51f80", +"5eddfcaca70c38540eae624a5da94d25947a56cca5c479496d2736bb4a3b7b85", +"f635b6ad111915061cf4d144611747de8af6d31bc284d2230c33967b3ae173ba", +"155396dee935f33703c1edcec21ec872eee1947bf88a5b374dae028336f3758e", +"b9140369eedc9161975cc979fa619e4e205e5952956ed40c96e62d58242b9af6", +"d79b8abf6c1d4262ac8b06f5b1e7d5be0075e5f5ef549319cb27ff2249abe912", +"9d9bd71488f5e33c3974a2383ac182f9c827c86b233e6d17024fd5a52c9152fd", +"3f18fa6487ed3cef6850dbf294fafc00b86fbb27569b3f27066a883ba69a8ae1", +"8044950f2e5fd7052d528bdd0a29a6304aed1c90bbebaaca4a9cb29d697b6a7d", +"3d6e7da65afbfa39ade6ade1290bbbfc1334cadce95140c777fe472461d48af4", +"5c8ce38cced28b9e17026ae5b31938132b10aba90c19d9c6d6409929a1565598", +"fea9879e7da3c571ce7b413841442c9d2f0ca04984cf23177c9cb9c52f801c58", +"c69ce636cb3327830439f2cba8657cea64960d3948cfa5fc843f9cbce193f231", +"b1b478c53c757ccbaa756ba54451fc299d60af76987008552b6feaa09bdacf3b", +"5d06736dc3bf403b37ae30ddb53c6d3fd74eb46c02a2dfaf113d0a7c941321f9", +"00111f61ac86c945568b440c3fd67128722a5a65fc3f052c63e29ada9ed7416f", +"833c306a673b542a8e42de6fda9502652bed824d2205c91093c819040c763db3", +"c9638a580ee3aebbba0a5f7146e47275d8d19c05d3a8bb3e236933cd46d826b0", +"89a24b089567e61c6b73d56f27dfceafee35873377acfffe3a0101f63c161f2c", +"1d2129049016f97d52768e92e3ddf2d669915f3f1ad324b8ff23805671b0a88f", +"550445fbf17934c34228fb1aa8c60eed5d6cf5abb16bd9b0358e55ad0cd0c9f2", +"00f9ac308b2efaac33edae87a3042c2293acb4c6fc72a545482b59ac3c087cbf", +"cba59aa4f9f94cdfd6a57aa854124f86fe5f442dabee2a486529a213c6c45578", +"5d1f031d8232ced3ab69525e66916f88bdfc834d8a8f78c357affd55cdc6a43e", +"7ea10bd1058e36868598d5245894577187acb79aab0eb4875424951915410ed6", +"d5deec09f01ce38c74290dec6eb7cfbfa08ef363e9b3df98b9ab2a4009bc82a4", +"40f527c5ac5c1d14c28ec7816fd28f9913d816e06a964fe602bbb8616173c9f0", +"30befed588ef70f635ea5ffa034519b3cb0a31ce93da4f78be1fe55677d3294d", +"c6dc3529f1f595aedc4e1996c3beae1728ea73be494df48d8b995aea301749ea", +"ee2f6b4b245b84bb272d19861c08c795ddd66bf08428bfc389c5aaaa4adfc07d", +"34005381af49af8acbf4324d16a4b7ec45499c6cac7e707fa018f5588bd9bead", +"2fd9d204f7986715a302c2735679684960263f8f7b8e805f768acc37d98f347f", +"ebabd28b06197edb3f50c3c97fe77caeb643d23a3f6ebb9fd9d08b968b559450", +"4bd119ca5e4d3782d9a2aa30653407d4de122a98268ec1d55424e1aa718b6bb8", +"34845a0f5eca756d7768ee2f03186ec4d73d86f9ebeaaf96abddf8bb733641b6", +"c0747e733dc391450d3ea7ada5c8e0a943f6e2524846965454ee1fe5ea23fedb", +"039dc7156f8d2210a3136e670638ee236f560a84e2ec239db3c6728d5be10ba3", +"1123036bcf240cdcfb398a9dfdf4b17e4b0af4d02cfa89c728d2095aa4407026", +"0d692e5522d6b11e7002769ce6708342d0da684a54537ec49b2b6efd3d0f1d7f", +"45b16beee12f2b2acc88425bb2362c9372fc0b03f5b49766a00891858bef4e8c", +"d255c2af6fc2897d6c4c4eabb50a8414e173006e9f9080b611bcfbeaee826303", +"fa0f5c3c9f7632620c7bf9c19a86dae5d8f8f9ffa96795cdc2bb9acf3f22603a", +"5c76063a21074e67fa7846cb320abc057a17e15aad301028bf7155abcff60d2e", +"0571c659eea0a98ccbee0d8638803b5be1153f9421d026bfafc8efe95d3ca2f3", +"d1fbbd30fc61e23fbde66a1d308879cd5b55328a03f2716a6e053efef48af9e5", +"f21e8fc26bcce5dc36e0705ced70c738d1b0db8d7a6d6b64de4c07a08cf11de1", +"3da1f5b36793e7144a0be7b6e3912e6d5e8e84781504b7448f13618910c8e92b", +"9435e09487f12fa449e9e898d890af6eb2037fab659ac1b91aa157d8d3316212", +"9943a47e446a9057edac23e79f7de354a43008069ea29959938fdb5a29dac095", +"7076234d01acdfac2dbda44b6fa5f7b3d4625482e6e72481d9c90ed08e9b84d7", +"de5e0be0b53e7d50df9937b3e47b885a5d1cce0021a2f37f6fe71b9ac1f5aab7", +"0698d8461a2d58713a457d210365f96c4f31a0ba2ee9592d07adfc15fd64b297", +"b9adce74312e75a20002ea7e9a90ccdab10701f2b3dd2cbbf71a9b032cb2bb94", +"34a2a429f26fca7dda65ffafb631a773f9f7f36b12d1d0a2688303a883c554a1", +"a71a50c712810faa43a694af1014bb4cd3689c3c476cd46acefb2091fe9c68db", +"eea89443e674cad4c415dcb5f652b438575b0c33f125dda22a4a81fe36011b3c", +"488d47c1eec5ac7cdb1e68a17e775bf3de307b5c58fbdff8e2263c36e9841cdb", +"af80cd3ddb83088d409a9b1c9c7b6d13e32cfa28d68bbe88ec9bf78893b9703f", +"f9e5433b26afe90ec9483f41b8d2213d37cdff6297d285e905bd2c0208558f70", +"0450bb34b86e68dd331e8f4233a89b566f5cb4e65d27f8bec4db5dad248b4a2a", +"d229cb092f4a0bf337d779a823cc5ea78dcd208d2522d96c50d13ecea99b645c", +"0977d2f7d367d7b3a8ebb4b09463fa9bc5e2eeeb93dcde380a46bbbc2ba72a04", +"d28f1866de6029d2c2b64685ca5f23a3bba98e5867b0251d89432b84ab4a4b36", +"5b5fd1b7048bca4897e02fe879a1a9430a877bc058b386995ddf0e8df17b1465", +"2cfa68f7e111007295dd0b8800393f230915824973fe79ffa2b94efbc9d4f8ba", +"454d4c6c4fe2580a25ec5569fb27afb8b3c6fd879b255e257ca291ee92970682", +"983a56dc55c6977d039725e9918d00e98bd598593ca1d0847b4391273bf5cb50", +"31b80f8160396bb957223856d0bee2bdf03eb24245729bfeea0350a7d54cedd6", +"f7a23de4e7d6d45938d52cd1d5bec8509bf283a60f3845deedd77a3ab9acd3ff", +"66ff2fbfc811dd292770204e2c0bb341e79a3983e06d3dba57806cfce215aae5", +"c2ecc0f73ca9c98df4dc1f9a64fbc426097cc88dd0ce3bc7efd7c53a006d3ee8", +"15a5e73f8199fc632c395093f4755f0205f1d062c42cd4c555bf685196a41e75", +"54741594d1a6d78760e79a804c20d119cf73c5b7f2624045e1663e2d713705b6", +"dce835fb483ee256aba2e13f6341c5f161c77957ad0c0035e1dfd931510c694d", +"779a5345eea3680a5f485fd7f4afe55f8bf09a8f832f68088fc095e4dfa1a490", +"28a5fb4c5359819be76eb2e5072a6d7df64fc87dbb48550481ce0752f8961405", +"1b389c93b9724d7a6183ff9d15cff8ecc79504d86de136c63211529a0ba6098a", +"f441b2351a64a7831d87fbd74210fd2afb0d775bfe4d1efe0033714bc35719c5", +"c590c942c3200e9f59a0c9dbd88289269a1b225be2ee7f0c005afc568fe6f4b4", +"f7a19a04950cd2de96dcb6d40804b6aa7b93d13ee45ffa89d1aa6b0112f5b0ff", +"d7f53da808083724990ba8a9c8cd757275af9c31d60fb7afa399f7c8d05ac3cd", +"d9bd905c8dd80fa063f579e0b8665c85353ad28d199052eacd4b79cfe233d032", +"ee7408fe2e7f8b58e928251df122e81e05ccba3305ba2871c5df1e3d15092963", +"8612636b7e6b06f90ab40705a5166b8860d46bc5c3c77ec825b1f4578c830ffb", +"3169c26616f550251dfafa68db1f6b36133cb57af1da9cb62d94818e2af2af1d", +"3cbcbf14be745a3094dbdf5248740f1a095da9b4172821de1ff8897d3a804d88", +"91d677a0fb9060e510b0d174829cc12222035e41022b96103a7424ca3ce710c5", +"d63644564d2efb112b806c050a5b15fb9ce1ab9ac8d9758c6f81dbcfd12defff", +"d8e3df6b7998d2fff5d9f6040c3a3976d60d0ee91d0aee69124902b8c9a156d9", +"813c07eb9e82396e43b0a4015cdccd90df4e7b7ffd2cbf66b7e1080567f748f5", +"9ab2ebd561e7c6cba8cdf99a3875892e7da05b1ad54e186c3f3d50111fa7864c", +"9cce800c65bf76f69e573e3c88d7cd99c0ff058b3d6e15f220ec784e8c0bdfdb", +"fba4d5357789106ed6700e8227951ee21f3d2170147b10600865865687c112ca", +"e8c2d9a8504b514e292dcf0aba49c437ed98501a190719aff0297baf890a5b54", +"60b98b2eb469ed66b4a58f8839ec57064e85888248016ef7f04a2fbaf437c6f3", +"1332bb42b6e501b42f00d07316d40c2591ad9f4125b91b43dd4e1023998349af", +"10455a7811e789df84b0333f34235436273583641528f71a15aa561edfa776ad", +"de4555c325f275bb854a37a957373ccd78bfa5f643b606f8007e921dfdfb0c72", +"a2a11a24b04e17368320d4d4fc3124e193afa5e591fbc43057bba7c1e022e933", +"fa4f07518e2407aa364550fa344152b424fa0eeec4cb9a1b7eb946321ea1c5db", +"e64798d1f7cdbbaf3e8ddd681cc5b143d39fc263a399cfe8e39bdfdf4917e3bd", +"d8bfcc19f0adb907cbe6cf7a50b5f593a8a26f046742336389849bd8677ce4e0", +"0711c7798de5eb55ee04fc5a0336e70fb2553dc21b857c27f10228dc041dcf89", +"3d4dfc40c0ab2b136b4013c1cc7dd1ff12375e759acb607ec8e5fbf792eede95", +"a77b8ffe21b109c6cf566471e23e9b55be6ce6219ff3357a61bbc427527b3b8e", +"fe20ebb42b6a833c3575a551981f6a8cd233657a6d05375af38e63b8c96724dd", +"6d26ce0dc023798f08749c399d01a1aa5540431564fe34aa87fd38e8951153d7", +"64ae5823b7b1c003fcba38db5f20f801f59aa0baf4e4802c7a84edccf1251036", +"0b27656477bbf51496eef9545bae40975fc98e490073b813e5b87f4d2965e860", +"54a3d7657be58591499961fb36a96003037e6f3cac0733754a5ad259a4cb541b", +"c2c56999ea052ec49d03b25d7f3d92104b36546c3a53bf20992fc4336d3c8db4", +"89e0764625c9cb43c175bf4a2aef78f2c8b557243f37ffce7842b6ad2eb31adf", +"9f3060a2d77d7d2888f0f236d6d9d1f5a3fe7ca9baea663430829982f3363574", +"217867a517b1f8187d8fd94172d4e40289d14043c9df828ff81eb099fb8998d7", +"f111b475c585d2b5f72f35eeae5ccc006ced57e916d0c849091bd27baef60e09", +"375e39b62a5795d070f0ab56e0aca9c419b142f470995a6f9550457857edf640", +"dc9cecaac87ac628023484b4b73f96d8e54e41673f29329c36dee7b5097f2377", +"afa2b33afc3f8c126bc5f6399e222469636d96e8a4d78a09c16a35bcdc8432e3", +"03325e9459c1ae99fc443f7b8b3a3cb01b2a2b17a42b8121c5cd8c83b375fd75", +"dc9ece643406f374cd3359115f6078fcde350a42c5a3f0e5b985f9887d52ea3f", +"ed7077a3989b0a72c4687d0ca057043da823b59d954139e8b872cde65bc3d2c6", +"97d17d13b01c4744963f3e5ca6131fed27e6204e2ad2f78b04b40af3ff8416a9", +"19fb3c01e4be5f4b538b400a6da152cf59d1898b2eb2ee3da7832c53fd35d468", +"09daf2014fe3f03c96045a3ef2e6ab907f0c895a9e9e357bfbfa1b3c3d1abc1c", +"a9878880b62fa1c6f86db005d1b1f0d1615edc57231952adcaec28210274d336", +"66e70629313e8cb8a8cfb8a7f25943b01bbcd404a5c123e8efbf985768e733a7", +"e2876912d7c4f91b1286661b46926e048435a379a17b3813c86a992a8d81f489", +"187a0ff12839f15575af999e06b5cf3591556c7c4b39936dfd48a5e711f7b6aa", +"b0e8090e7ae2e597793230bb4f3daaccc32c880487e465573bdbbb78004af1ff", +"d90ebf4ddbe21e8865117fcc940ec652e771c6fda2d74b6193a128429ee5f8e3", +"344db13d1cac71db2cdb96a12a3fd08e88a7ff677065cf03646c14062db8c61e", +"0f47c8cb01428d3d1d8272363881e33a9b06c5caccaa6f9f99fa133b5a80bcd2", +"4ad11bb9ea5d19a6f169720ee170dabd382f5363936df80fc37b1c80b77d017d", +"85d59a9ce722db2f1f307c8038b8ce5bdd22d0aaddbfb11816e49cdf32c03d27", +"f3d442afecf9a7d5644111bf7e544a8e31d2a58f6558a85fb1145663f73c1a00", +"3be36e39f1aacb33a2b29084ce99a53398193ae5ba2cd316f004eaf527f0c73f", +"f5625f8964ede138f4c0c8617062b96406bdc8db6101e4ef70a780a8c0440ca0", +"c717f24a62c365d78e683c3613ee02e7303c64b63f5fe5b35cd9ff0049bce0b8", +"6ed26ec48723fa93c261054c395595d6079f55b1137a5670ede801f14c37f4ca", +"3345def35b72521392a1273b815d5cb23e8262a1620c3915ab0e220bda0e7f06", +"b24bf27ebecde1a71433710bcac9553b5ea9c1c002af96c2cdb832f1719605ab", +"11447e20b4e7ca4c7d4ef7b49ed412582414b35f4d457df735f6ce053067a2cb", +"54f4bc004eb3fde8620b8ad01c5f0ea0127635ee3a1d5a9d963cb9ba199333f1", +"fadb002b14a448d34bbf9d5dfce51bd0aebb4c81d87edbc12bddf59d01a50d6f", +"c6185699a63b8892b8d7b544ba59675b05f42d633eb6fcfad34bb1de4b8d0033", +"39f135b39f4207cdcd0f3f1dee591bbecf4ce2ee1541e10a14665a030068668e", +"1365b4f79a66619b8fb548df7b6d212a6b46ce295804b524075558838c582570", +"12b1e8304da65a1d9dc6f6ef9c23015678c25188ccc22f86d44ff8858d0279ad", +"7527d19d8d8ad24648d2bc964647f978fff63a54cc08c60ad76d5c3bf0e700aa", +"83f0912fca4eb0293555f49746173d2c84e33d58b89974723926d19b63f72091", +"6b823a95f305e1a0dc3c94b8e5a7316623b685810be033082caf73b46c98cee3", +"acb17565d92f814183e1f60447403fcd136c2d9f6baf7227e2358194c89d1dc8", +"ccecca8dc6083c5066b3618fc42d6fc27745c51ade1b82d332b2a6fceea44f1d", +"590c00626ac0f473740794be7264dff08dd93bcc895757ca8cfdb61fbfa2ae42", +"48fde2fe6916d631a6d9c142cebf0556b3a7c308f2b07d2dacff0748090ab7ca", +"2eabcf91e7cef03aa214bec4fb2d1db5f305588de5de1ffa15f707ceff038bda", +"db2dafd6ba4e1fe84d1b47361c4671dae4c52902a276e30f16e6c6baddf57662", +"08a507b4a8069c40c63d0256070520a12192df1154bcf73358c422286d2c93da", +"501f1dca1da3cd30ebdedc570901243597c4a41366bc4c4ce0bed7f9cff00c39", +"bd2d87ae5061c82abe0a96ae5841ca8867a82bc325737fa0e024e1a7c71d44b3", +"9582540c38fce4b277c47ce50be1cc3b53640c25ea14355033643f3766d60c66", +"a7e90a9bb2d8b2859c7d71df202e09178cea8ceb6c07c750bfb570506b273399", +"fff6d4a266c8d84f5b84baa172f79e1c4e45eaa49bef1b7b4416affa667eef98", +"eb3f8d9ed92b6b80c750aa909ac172774c006ac0170f41e3afa2b58ade0e6eb6", +"e0e325321e0d6f442c5f06d63cf22a71ac4f23664b67dd29599b9e4078b0bf14", +"68bb0c532ce0a655023a8fd4dd061ff26daa53008ce716557c83ec76859a6f1f", +"ee28e8d75b74dfc41ba2bc798e0037f23ca71f3c19f5cb8e56ff12e17b33cdb9", +"74c8060014daa6a8fe0ac7161cc5dc798537a80880e87feb91db18c6b737351f", +"a9fa3ab85d97ebcd2c84f49ecc659203ba90e8658fc82c7a3136be15d69503e3", +"22e841f0fc5c09a31b2fe74e822a0d3912da4e26e2d85a4f962158f69bcdee5b", +"f2933541b288fe96a9b2f93873fe0890cbfd03df9f7a227d2d620d292d965b58", +"f1831a16365de3675f4bc483e68efe71bc77b944bde5f1078102cdcedfe6a24c", +"8e5d2c73f16db3465998dd683d089d8e2aa810781e71d4973e84bc5eb6f0402d", +"6fcb105002890cbe38c3ffb6e2b911012f01fe5652b940952149605491b637d5", +"52196545e745227070c4d76229c4082406f2c00817ef4ed4ea296054930f3cea", +"2152369bb55306c2662ef31e432878b4a7d9cef62ebabf310301e965da618370", +"6e735436a7832b7db6b22c6693e727bfea45d55c50af9e310b4e791a1dd5707b", +"bc14e757792bab302541f79ea24e3ad24a70bcabe2e77d2af46d462e503ce98d", +"7af5d784ed2347849ed485aa544625cfc10da0bea977667e9c0dabed82130418", +"bee1bfbd8abaad8530780e8c8fc6acc2b7265bf0159c61ad0f638c1f8405ab79", +"189c856c7ae3d307934f59a1b064090b80e1f3eb81070e39aee4b6c80ecfe2c5", +"a035830e48e29e5b963038abb81e6fc553f31715d4265312e14ef667bccafa1a", +"0aba11501987b8d65a438a57b63936f44db152fd9164ed841fce268189f65fb8", +"312f8a15afecffa85fcce4cc111527be81c5f8945f0f28e70d6d938d931eb49f", +"354432c6a726ef3697f5df89bd938ec4902f6854e0aaaf5a7e030a1c1f8a9a96", +"b3f02f8b18412a6cf4531863fda8db79aeb31af12a15988f689fbeb7367f646b", +"2e6a965bbec7097f3e25cccc37c40e1fef6679befea3ae06cf274d36a4195af1", +"0f57515099a25b5632667161b03bf684bf0a8ad34db4cf33dd090f530eac2377", +"300e34a71b7968bee2e4ba7b96ede2da6709830e15bac7af061c7f5a28d0d02d", +"e51ef88f3ac585d5a776f026977dc229c81e68b360f8cef21dcbec48844f90ec", +"59c212243f01db0ac7de0ff0c9f64fd842d05c32476a2a2a90f22822592ca6c0", +"62de6453795c5938bc18cf6e5a19c547af1f69303a7b02bd747ea30b44319f6d", +"7545665d11e598abc210a6fd3cf4623f2fac368e6cd102fb49efe52615a6c834", +"78916d8645652a97356502ebbec2c258ec461ce55eec41c78a9eb7a7da43db72", +"3a495fdb1d66229ee0d2f7eacf9b889d116f6fe6bba4ffdaeee52fe8b845dffc", +"ea1a2c226edbd132f7ff4768fc4e43eb7f78719676e26dd57d187817dd53b853", +"0112433201659d58a096a9db311fa9d23eef66ebf81f8584b9fa29fdb89359c2", +"4e4bd49a3e505a86c07a46338a054c9a2b155c7680fd544628c793731a11bd99", +"0d0966f323d36b2bde35932698eda6d2b6f236a205c1ab4becba45132062483b", +"d3f6882c3dec342bb97a67cf7dfab7aedaebcc908c28c67a20f3397b48e76d2b", +"13fecde99dd23f1c5ff5c8130135ef51d7583692079d7e94a757095a82db0ab3", +"bab3cb6ca7624513c290056ca9d9dd768e1e24766c8274a9f16ed1d8305ff426", +"7986de51d61c642c7fab7f6dd02b33e4b5afc2e7d4bba2ae31650232b2256bea", +"87a2c54dc30a58883b1527e035ca6d366331fa98fdbbb4837ea8f0cc79d741e8", +"66e623f3dbd35672fe01ba210386c3f06f2b7ee5976567cb2a26206b14226bd6", +"b7c49d82a0e862e2357d9f7c51334f120cb6fb62c71849fade595597f4696dbd", +"fd7b01d012f67f58e12e076369da78c2b63f34b25598b3a114e59768ddbbb3a2", +"084f663a2e66dd6b9e06a90368555686ecd1026432cb863aae060a45ce09bf5c", +"24c04d16e5ec907fe3e2f8dadca1cb60da56c0be7ee7fe1a381bcaae45b7ddfc", +"0b73e5b6dac15f3f86904b7de03b5f187736679fcb8dad008ea8d99ef2f7baca", +"6c4c29a4d9b71a3d86b93a449d92ae04f37cc4f68edefe704e5c99742439df8f", +"710efdd2e808747139c0a3324334d05f9e24dbfd01edf79f5db78b4a99f2c42b", +"850a93d11a35c42c331ea1118385dcecc91cc7a14dbdeb362d2f0f7f86d362a0", +"0f95236f5fe00f99cc298898414505efe544c524e5fe5fbd8b407d3330abea64", +"1258be5437a50d58789e1d0c61750adc6da099930dd70c126fb6beb28d3eb6a2", +"6f56f6feed657cb9dd7fc7a59ddca27c4aab79dd869f46f266e892072454ebfb", +"74b00a8c7ee0342ee58dfa96fb877a04ac3edb50a2efa679940dd9a99dd76ee0", +"bca987d06080b6551ca1ab9140efd153a06e16904d8974d660e6e9274f3b31c8", +"cd98ca77faef729789d882a361305cac0757b119d3bf3138c39a426264fb8aa9", +"9bae1d40b1f917a027934f1772f8d28d72d400332bfc9ca196532b8cddf0497c", +"8d8ce538b3392d4c81401966c2482a0d81c0c5e5399e5e9c2f976a865410051d", +"834b926aae5bd04d788507b3272efe068f608ec4752c58d5c3e15e814fdf5a91", +"865d313772553067ddfe3d7e8b46f6304b519d30b8f7c81ddcb39a8e324a7537", +"e372a37847ad892cd11231769275ce755f332ba342551a99743a7593494e5159", +"5c458a84e32063db9a7ba4557fc358b25d59a2b06e02ad37b770b05d08fda2d9", +"d6b1cb49ab22800f4a865e89dbcb5ec8c6c752e047e79ad0daf03648a5acb6f1", +"810101a23bc648c46ff97c04d6a4a4eee229d73840688162978c5eea4fc77ee6", +"a47628794691fbe9ad4d4adc4a16137945f46a8ffea83bb2610d7f7ec8d02ad5", +"7e654703002c0dd479e350a9025ba807f74d8eb8e73ad0f31386f378ddb0508d", +"a4fada66644d57fd902ba5f9058afc10a79e87a2b3b494355dc178a57f9cf55b", +"099dafdb96d3ebe65b795b35ed22d67a922901b6fa2b8337c92289ea7e4699ae", +"6af574d4a3f3a76f0480e4351c2ef40cf6d3803e149b29007138697f1f52701d", +"e99a3eceb12cad45beb08064eda2587f35d9283e04f1c7167cb5a660bbcd40e2", +"312e9fe47f42db52dc67e0c25ca13104a8073630dd1bb586c02b175634cdb469", +"7732a0e3ca2c07ad4c0a823811ad5fe50d23e258bc876d905484809ce11af5da", +"6e23730cc2bf817dea1f097a519c6c1605f51d7ccd254ce7f1182f9c2a0155ba", +"094c91997fed511b9b2251dd5c4ffdbe4ff2ae8feb28246839b5dd50ecb72e2a", +"b69c622628dc20c41d7f4bb9e1cebdedd2fec401638ede17fd890dc4cdc2bb06", +"9fa4ea0abb521e74bdf866b58a5e95f1847414339eaa93dba4b5848e716d0516", +"65c2015cb1ec8679f9d6065f3e45633a56ff8c7606915b4544d1160209db0c5a", +"7917ab0dc977e14098f07bdc236ae5ca5a3d91ca6b18a58fa203794b0dcc9b14", +"c97af89a9475d93a2aeebe1392d14f1fd44352c3ad3e391d0a8829a084bdd5c3", +"06e1811a3da521ddd5d666766395e91b68c00227d8969c4b411413ffba0139d9", +"b53121907425ef5ba688a54f1cf2a36598bd070fdada6d9e96afbfc417f896af", +"69481cf47cf64685ccc68100279fdc8c99281cd262db350d41dae9285a500aeb", +"5b4a6f3ec40c0e6721cf2848c6ebf1b12f862eaeeac74b2677567c5e32754996", +"ec4b87ca63cec112765a1fe9236581ac5567c16be9e6c0cedcdb08283a8033ac", +"361ee24a89831b766198924ed196fd453142cc3cf08f15e9053652a29499d8e9", +"28bf4ba94b782e2791817d19d62f5a31a5d667f40f6eca9717c17597600cb11f", +"950dc7a272cdc559569325c25bde0f9407c18aa2c4da2c56f2d9cb68fd1cf1c1", +"ee5ee9952e61530a3932fee71d30e8cd39b8854708b042490507b7e1f8372ebf", +"ca5d35940745d1e532690f41f5d13ca83359f51c505e657d212522187b93c16c", +"ced917cfdeb5dd8944a5f36005991dc924d6ac3381a4b3d9c189d7b7476f1f8c", +"788c72af07fe924d9a123ffccf649829baca970f0422258be1f631ec9dbaea10", +"e6d2a99f5a493ee6ffff83763628685bfe5a379d7887076d1885ba2c4f63c35c", +"dbae38b074978923eb63cb4f1cb234c5b4d32562d4ddf1c6b54d74609d51a9c0", +"3629be2204f5f0b6acd3f65a2afe85d4d2d19d9b9d7874ab8300a441da628667", +"e34cc4735c87aab7078ea8d6d3c32ec9cc6cdbbd6a0856d22e58b0418e095e63", +"9f417a32dbc3cb2e077053f85d7dc0aff7d706f99ffa1a9354ebab33cdd7e94d", +"6eb72d5d1558dd1c754f718441ab50e79f3d87e2728a45bd1856e20662c8999c", +"86003177b4516596a8532d2e420d77dbe9996a139a5234bfb302b046a5771ec0", +"d191231663b36f07c88f4e17a401972629a3c9166f23e76242ca607fb130a322", +"33324d7e00beb6d9daf773452a55205f9625adddfd34b0ac0cace69bf35a77f9", +"e389596f0337c8e0f011d0befcedaa787a012ea63cb0b204a9b5012188e237e7", +"32fb14a5a33e69c9743dcdb7758fe961d22b1949acc22584253a4b8c3f19e99d", +"ed3f6a1342dcf2655aa958ebe375aa2e5b3fff62606417097b0e2fcbfd069ef9", +"d0e548351f498aecba1225d9bb2a141c7c4626ddd7af419c1e9d654d36a786f8", +"deecb1e16d170c891e208d3546784901ce2668f827364e72f123edc4572c5238", +"bf1bf8ec7e8526bdc9d7b349f74d5d4eb4ca2b75ee96c456d84dc108fb76aacd", +"68f51ffec69db08d1641d1a92194b82ac23017940dc29ba52b7889127a060f42", +"0ff60ba02bd223b8c76d26292d41de2f291ac96093929e4294aa32ebd924cae4", +"09be352a5dd60b3612413e04bde1efabadb801ed660ca0d48be24590b70946a4", +"fb6a93a024fee3c14806ea03c7dc494e53c49806ad9a2d27ed008aa63dfade7c", +"2fec51ea954112b77af5e584ddf478bfc1520fd2205b85f68bfcd4b6c909a791", +"d5b93229f11a527bf2528a0554b95ec5dc21fd451380332b17385543cd3c7a83", +"2cf261b770e40d61165f56df0f6865ec5249e2113b95d91aaf3f5ec5db468055", +"b3598df24340bca2767fb5cc23949a174461015e7ff7dfd0fb32c8eaa0d21aa7", +"94bb4e5edd99f628cb617b5ecf4cd5e838074c56de3c2ffea4e90d7a07410853", +"76db03fec95073190cd0faa9e5fb862e50d478cbc2581c4ea5de20f82abcf8ed", +"7e65af60e9c8babaf0dded1dae5d84bc6461c23b2b61a59cd2b9984f880b0abc", +"5110ca4464b82e4b1052b926328945326f4e466c1663a8fcac45dad23cb451dc", +"5feddc06542c849b33217e91abd13e8ebc208b35fbaef85bfbfe5cd88fbad54c", +"51b8347f38e38a12616700301b96c4802945f90f576497019ee885a7bb82a89a", +"ffbe8a79a8a9de1243f6c022358b2e84e051185636b79631c5e7a03ed8abb8ac", +"67b831d200d7a0ffb9f2ac6d0dd16a064e0b8c0b75b1d0cf0ac120542c6734b6", +"798d53de8c3c234a412d140fd3737cf2f40cfc02cc9f97a8b0aa611df717f9c4", +"f3091d0577b5201deaf460a40d636720308f7e74a2d8eeeb2be2fa5f97cfbb73", +"99c93d938649e1dd5d0e6af35ded02d574b60338f3f21aa031b57e3bcd6f9b24", +"8936c41426939fe5f07e1b5ea80ed2512cbee0c0c0818c443a5f46e6bd59aac3", +"c6ac2906b977ff57ca817dfe39bfdfd1271a2540e2fef5d00e678f5ba7e426b3", +"bb53f10dc3630537b9b746f2eb1d81e6f5c99e59081daec894fec9a2dcfcf06b", +"8170148e5d12bbd05811c45d42f65d85acf667e40c4c0f8ef4be09eb9031396f", +"7ac45f71b22b6a2dfe7410933b6dd3dd9a81aa120f2036bd80bc0eb476e8b1d6", +"4ff31616cea77b08aafe30723b2c26a998287c132ae2ab6e385051642891e375", +"82bc13847269f69b43bf20f546160f90fa3002f55b62923f8e2c1366a231ac8f", +"de8d8f130b29502bec3394c5a97faa613ea136e6869b8b0ea3e1b29bd1a47a78", +"74068dee21ec501fea4efb2a6fc041a1ecead0de85e7068d5c582c2a2d7411c3", +"995cf6864a28cc3a9d82c7c7ad4f1609148f0266f15eb697b559469c56511e9a", +"07b71bbdbaa89574a815f5ad6f1d696ff93f6a80f3ab54a5df7ec79ae52a1a4e", +"280428476ef9fa360b6e9334fc1085371a6991a8dea25d21c89bf52303d4ed27", +"6201e48a9feed9a81a8d7d444a05dacf5d9b5486d3ff7fa9691a1432b85a06b1", +"13a82a31476ab851296b0f836e541608b41f9de1251393b9d38bbd198b52f819", +"709156584c9b6aa7a0190f1871079e5a9fb38dcb31964bc6daedba75eb7e24a2", +"73453776866b747b32dea027e09724b278aa1d7297b49296f7ff6965a65b7342", +"b43f068822e680db306d3251dc1f28b60b45a6286f5309b2ce02f71726139656", +"ba97ba9d8a277042f04755cfd8d92c08f2ab7675c2da2b59cd1f8fcbcd5a69ce", +"92154cb01b0745d9aea6eb9e419824b1fca60886f9684a2c1dd948751c1cc687", +"4e3f7578572f4b54fae02b769e63357c6797453f9acad2636ecedbd309aeae25", +"b92567f94794880acdda0c59591716d223a39319e10259972d8e0a2640b9710c", +"d7c7fa415d51ce81c1259cd7f6582fa89a2c16619c92ba6797e70ce07077b877", +"bae5e289f1709552a03e09283a9c4e44e400c52de6923491b4983e3a97eb7087", +"83008d633b3f9725000f57cbfb5a1735b40dc93828f13b7926bfd9803d0a9af6", +"12c79c875a912d44341f779fc73831f207f11a0876d870e8ed35c80ae9fc66ec", +"5bcad6dc58b6a2568ef82fdfef18f5cc8e5f189cdf43423ddb4234745de64194", +"599de46d79b5c57a62676fefd1c08d80e8cf29663623c2556ab0981d302cd6ff", +"d94debb273dcbbfb93d9424a553508254538038983c7d4082da35c6205913494", +"4bf4b1814d5bf55ac3f0a36d2acd116631ace9241267da7101f07ddb8504ffd4", +"672f77faf0fe8727e4c26a6ea93826640ae1cd3d806be07f810e6ad4e0407fc1", +"71c7bcafa661757a838e608d87584fe02618f8ecf9f55bf4993fd0b33ce6d635", +"e8767a2d0201007cc3f195d20d509de95b9aa6a1bcc8ca0240ea2f7618af4f88", +"89b5005a00d77d1608540f2d27f6b140e0ef33a01ae9c1d1b6aea027bd00b16d", +"b5e88130b170d91a980ee8ea1a2952caeea152b318d38967a1bee8735ac94cd9", +"a02e2e9dd842efd5a5c16d4d8461dbd31455237ed09d0675d69dbafe0ca46c0b", +"6a1778bc880fee5e38b9ffd5dbe2c23d60f516855b97ebafec02a1185a06f9f9", +"efaf97118f119fd1c18c1062088a1f988cb682e5cc6d74b15d3a03d10d881854", +"f824e7342eabb60dea659908b2260d74af16e1dcb0cba75242578eb1e1f008ba", +"057bfb3b104bacbe19fb9467e3eb7c2a3947a7c9b2467d758f2cdd82947180a8", +"04d660433f8c890a814e683100f3cc8e5805d0925f9a9e3c42805a03e6b4a36f", +"a69d401c73704f7e9369158673cfc0a5b3c88e18497299136a06597281410747", +"fc8237f6d970b1366af7a0ebcccf86613a142bf91f3b81e7d3308f1b9535b6da", +"bd2671f710ae5a43c6ddf9bfb504c685cf5181358079180a2fc682db3e9675df", +"8a688ed2b74cda40548d4fe3f3c1740fb017f83bbcdaf985478eaab1365da4f8", +"57f115bb1460d42ed38bb9cc4e167fa41895256c739c1e9d57b26fd32bb8329e", +"c5f8bc0f398b69700b270a4f8cdc53199f84dafe40f3a09b72e37848aadc2aa1", +"4ab33c9ffa7bc55d05965fa9b3127920f9ab5c731dc8bf255e41504ec4f24293", +"9c39060aa8c4ae392b2a128aac74b19ae4fc648b0260af15c1cab6a5dcb22496", +"4651b78a941d1428530d8c198d2e689a7a3b630019e75dea624cbcb547a8c1aa", +"e91332f27f14db1c488f70d7dceea5a3168bdf176c2f209a1a96cb7a8fcc742e", +"3936b1dda4b305615b0d849e119eb6e4fd6fd3594d27230fc5f31d689ed5b0b6", +"61b7e3389045d323b3c9b03bfff11f90e17e4a642ca3b44c889d91ee264df9ad", +"347f1d83975d0d47f14d8d01b0d1d7164b956802838d9bef77a723795a629f22", +"15bd9c8055cae840f08ce601d2d3b460ee7af95cd52b1f4f671020fd3192edd8", +"dad41366b8fb11235ce8a1b6a0837d5758c08e0cf4c710ddd5bca41546c2d577", +"67c4904bf3fceb9d4aa6f233071e95abb4790b4861536123ba8f0b9360a8e1f2", +"9b65d251d0abee150692c91cb261d7517bcf26fe8395c275263fca21641c43a8", +"e5a8658b0d25cba6b4b0e02149531c400880f8e753f9c141de54358fd9c5d860", +"4fc857383643eb2573dae3c1d7549d86aa7ee7ca9ae22feca7587f542d34a6b3", +"14149db3e7b8d54265b09bc84cc6220d91cef8644acb4d9226e62d4e95648264", +"a85a28b5fd50e7fa846a74e37f29b820ed74f4fe2dfcd08b506bb54f4b64dae8", +"ce16c86afae21d8fda9e0b64ef7534529b86f5565e45d569068066d2a9c7ead1", +"201fe9ffc664c12b5b8d4d08d241aef5458a20da52bed83b8baff4565606ba9e", +"5e3d459986161589455cf0b7e112212fa173fbf20f65bf1f4588338e2b05620c", +"cd23db06e4fdbd08caaea9e27755c596c84efdc4fb34ca96cb44f8b8e3344aac", +"881f33f0b38947e6f66bf1b7ceb3bf55e7d4586c9ce107134bdc464dffc72d4d", +"ba4c3021b23a950e61460b9d2fcb8c9fe2b934ac4cf0cd2af9f44dcddc728fc0", +"6a008faab6f4ab963ab6871750e94cca4577fcd5b72c3206075192a0490f0fca", +"37d6718c76fcb03e05aefde7182ed4378368aad4dd06107066d83fad0fce5f85", +"0697e06274dd59e1a5464b4062a3687f7d3da144a0ff165956c449879809a832", +"3d9769a9b9f663f791c53eeb883808f27d8313405bdde5b7c0ede1797110e0a8", +"4b315e9ad6622d7bb257f890f663f440709a11fcad5ba2b1d480e8bdb3b89fd6", +"b51ba5049f7d61cc53d15d59313c9ce0b6946ea47d58758fe044798ffe59aca5", +"dfcc686e007da302916416f5a5b9a7b6d15cfc8f28c23e8960930afcd249a739", +"efae6869c74dfa9c6410f7a9294f7cca8af33edbc6e3536f95deb43e2b26efb6", +"85079a3049884bc1ea3c0a1e28beb44dfce94823a60cf1f91544244a68b7f0ca", +"3924149d3a71dd370a151d1a110defef096de93be043cf4aa09b8b765665efb9", +"56a1eec4146d50d985ca1f6d00c89892778dee916fb8ed206c09411090baf039", +"388565b9f8e5780db9c571f6b55593dd84b9a16fdb1b74f8b32d9a613ecef342", +"8313c1e235440b1f0566e1ffd62ddaeac071c3c38668502df5f94580fb0274f5", +"c8a6c5ca8fb8661dbf24e21b5a9fdd2bcb38ab008a9af4105e679c47ca0d9ab8", +"30c401c42baf178a2c88b4da2255c0b7c5a0baa8f42dabc7d5b545d0498c2126", +"30e4a0c0e43b6dbd67a3055e126ea11b491e0a6cd126e31dd9832c8f94d6fc14", +"f98462fca92caeb56a67097c835470ba1e790266d9d3c38a43f94c2130bd6786", +"850908f55bd7b512f20895253f09d2edbe775a8594b9d9649e714425ccec45d2", +"9dbafcbfc78f1f71993186add1d5bceeacac926ce70a133f3c52607abcc3ee97", +"df2a2982263273c225ee5ad2ca8f788293509bc8f1d52594d7d811959152d627", +"947d27c95f36dd28310e051499b86e022ba1c7640dcacadc0b23b722fdc36b95", +"f5b817445c2df1d76d7f57729ddfd3002c4cbb2c9310d580f0646297fbd40a23", +"3cba96b655ddc470cea3c92564889027d32e71e8552c9d0a38d7d1221676d484", +"5d6628d684f9038cfacd56ffff298c81b31348b1615377c36052cf2d6db3f377", +"142ef084bce05900e3573b85abd7d51e79e516481f675f0769a1eb325d824159", +"f3669de7875dec76e286e7d6cfc90751e7e88a9faca280c0ac94d73e71a80647", +"9d3f41ee6e5ce577cd11f891ab855135d71c9b924abfcbac52037b7b5d018aff", +"746f92b625d7fc474c0925da4db2162dcd435a749adbe5d3d67e9750733e0254", +"fffccc385a9b1a669136dec710e826d99da70d6e6a2aaebc6fce2202abd0be21", +"69e9dd45dc1b86f0c790e2af06e6474bb415b4b747dc7982093ff5c5b2bc3d61", +"bdc2b96de2957ba3b848c881aa026ff1f959230e1f2a352939fe9ecb8ef712fd", +"9c4b6a8295f1b5fc177c31ab1a9ef73e6c606425a452d972d1e2694d8c03447b", +"985ca0ea62cc60c2e075662589c1c703444c6611fa8940c7e7df7de25f403ea3", +"1072e4694a0b28a1f684669fe308d6914954a736616fa7526b72b12885f95a51", +"d4ce7496aa10faf9190362da54a9649c5703b9ba17d71e234164fab84a287e93", +"e00d7629fd8cb0fa5a80048300d94cab55746a9c21965f19a595d11b947f6b5d", +"24712f29874d05505fe4c74320dd51cce66def98b76fb1963b26afc1945b4c82", +"1442b1e915d2b58716dbb0da4c1c8ef7b0eb71166046d5cbff1031a01e21efb9", +"d683f4a8105bdbc816b5afb478845ebcddb8af1a3b24fdebbb1ec7b3e3fe0ddd", +"f391707cd506589a73d10a6dd2dc1d9e30208ec4ece130a91181a4927c57d051", +"f8edca70a6c8f62d463de6baca59072b797b0ce626649c99acdf3d6e38545903", +"01ee8800c65f6014a1e2dac189f34275cf25179f19eeefeaad35c12dc31539d7", +"2e820ffbb7e6c0cefd39b5f1b4b3dfd099acf5449be3aea4097245f2b08b1252", +"12806781c7b8efa84ca538d752183cc2bd0852bf0bfb7f5f1f646ef1a18e9cce", +"8acc603ac9514f7855327761c04d49f5ec931178a3e207e3627aad366cc9f21a", +"fff5fcaf7a52047be6ee64077b44aaec937275306033b09ebe03a8a6e85817bf", +"07743db28742237f90b9cddd4f5611ba1d1f53d71d56a6268f563080185919ff", +"3d41cf85b32a6bce7f766fffe54d399be503d7d4310e9e39aaa32d31a7831817", +"b9ba348846b31c40da668f48351ab4e11712468d3213958df83083f9da85dab9", +"46eec803f5e0d0412a0ee1ec4d9910bdbc66e55a420a1f2b7269059c8ea6d051", +"f25ccee0fa4a91d275179ef2761f0a7e79b0b561b0a8020d890ccd8e85e7e57d", +"51296e74b398a4a00fef0397690ec0df4a03b8ecb9e4e315df71e4280094556c", +"29cec9f021239f053eb04668e7a46c7908580f00e1a21a055590216531bcade8", +"580e0bcc88214146b362948cd2154b9100d2a895c1741106165cefe487199054", +"709948ccee295a472105e289dc342bb69862941bc983d78918a8a426fda65d14", +"859b3b38d452ef89a659cf90b2ab373426e5784b82268b1361761917f28d625d", +"5fd8db2754cc035eca5273f259a7392cf7208dd577b607c52afd943990dd34eb", +"491148bbdd57a7fa3317b0f643bd22cdedd983e2b5a45ad72580ace15b30ce81", +"aef36205585544f7f93ff7822b3a95628191f9823b0cbb68cf96157bfe9d5918", +"cac2d97e43ffa089e71994bb8565690c348a7769d0cc2752f9d4822e49a90bfe", +"bf895a54b0c2c31d18435ae46b70ee47f1ecd6e2f51f3ee59cd6eef092495488", +"a5a584c1bf3bc1ada56598d97506e82f38261440874af90dc6c21712bde9ba87", +"eeff905bc1556d393129ca90c28416c18925229b7d8c6087e44ff78a9ae9a164", +"f80da0fd9aee3bfcc6ba2da8be50a709d11c390d304650c37ed5216209694149", +"5f2842a66f2641c128340337063d0f34d5ed4e6bde2a9d268e54fd4cb7317492", +"36c6d7c5d4cf186d97911d13aeb06a37ae2ffcd2f99a4161fcd494150ce841cc", +"0dbd7be61768e4e716cbd33e40c4249b381d563609076305d01b84418055dc0c", +"67ac48d7da0db58b0af12005631b1c9072a74bf8d3e40e75189e0a5bd53c7a0e", +"48b57178b5b829dfb580ecc26574a05e866f6242b86de0e0e83899b01c710116", +"ccd7761a162c39d1de08a37039ecfce8336d33e9a4590b348ca0b9f53c2633cd", +"d5c92eb2a4ea105a1f4331eb49899c00d37bbd498a0097db32ef077c9356a4ca", +"7b874fe174a874fe777d48cd30ba10edf3675858644fe7e888f54bd5ab4fe80e", +"7b410ab6bef4f847ed227ad0470742b425978678a22afdf3f57b2306713d64ae", +"ed9f5648ad024995113b031b7e7e28f14bb8a8b158ab623e1bfa47369ab6549c", +"7039f15f6a34f84a97d74fbccf7df103aa4e5df20919c5cbd9239eaa20e7a30c", +"faceb9ac544738ce1d3d7869e15ff79b4dc642bc9691ee785c0c629988a44d6a", +"9f486ebbd5f48a4f6240bda2b0df8aebe6735388f11f248c0b5dbb463f0b28f4", +"d98a7db3ed34c2fa7b0d72a17da9da49e4623e7928769ed659fd0cf68822fa13", +"4de9d497ec70162d2e494b69b079b3ff3285623c8f708f0f29779d6379b4fa82", +"432ab9b414968295f7357b75c9d4c124e0b86b547233dfea5c86a51ff59468c1", +"3fe980438d308ddbf6f6370d8b2de9f9c26a17d0ea9b17108ea6aa57fa852c0e", +"8004f2962ed7b8b436bfb922ecd0de64ee6c093cf26a518a9f97b4e248e88a6e", +"e23c724e9dc9eb215bb603f5ec69f33a7a863d01e0c33cabfd52be253c826a75", +"ac8c10dd4462585c7a4dd9fa94841349a0c3ab8e473df5a4c2d42dcb0c1d86f1", +"51b8bf059dfca4ef0288f050c9536b16f646b84a62008ea6d3668a194574766a", +"d4f0e7ab0e8643b608fe22c68747cdcdede17e1c87753db6df06227377b6aab5", +"3dd09627ff359ffd7d81cdee83cc2f91746b3b9eb2ee6f08ebf795b788dd5551", +"bd3cc90f2f7c72b6a092ea1848828ca9461ebf249df3cc71d0b3aae1020420e5", +"34b61da7d49a8dcb63806f824af5f9e71411084e9f3d82c2b23f670c353aa6f1", +"1ed1e232ed6faf18f7b9111005949cc3be914e0e77d72eef1bb58a439c208da8", +"251434fc78bf0dc080c3b1868fc26e7863ed49c9da982a9fb3768819e7451c0f", +"8d74cdf783d64ec2801a5c9018a3687bd644ac42d5f3bc77224a47b90dd359a9", +"15bb8444baf346a30a87af9f53b26a3ca4be96a70a168b387ec7ba3f839c1c78", +"f9bc168e7952e99aae7e9f519216a3c749a3120b373a9cd6a648cac0182540f1", +"a2f304e5655bb15ebcb38ff17d815fab5ef607b9a6dc680f58ff92f47249032d", +"afe05592e2aad56bd8e04eae47195187fa54defa5c3d7264cda0733f64e4c3b5", +"4e21f32c2d60be8f71c8167760b3c96ab9b83e4bef4a2b9bf23f2439f4c030f8", +"f05c213f692e0b291c10a6adac7ea9baf4a1ffbd9442599535fff09cf278cf66", +"9a5b9d06ef2abf01182ec070a002f89da53db6e62d09d1182a01d3a45af0dadc", +"a39678926e85713d5ab0202fe8056d5db3265fccfa27247315bbacb3c74c3d89", +"1f908190610cf904552c33c5e4c384cb5adf9994e240973824c7316fd1b12528", +"e611412545b9f7a1ef0feb6b4d2a0824f51e654445cba2247f0e8d7b08118cc2", +"2b4ce090fdade2c9cde0122f338a527db86dad3b7cd35f25ccd08004348ad08d", +"957e3dd7be77c9679a5508cff9efbd4034aa1a6d39d1ed5564e3138bae30bd90", +"36cbf593ac8db5a927448625b29bd3f415a5583b323811142e1ebb8e7cd28d4c", +"f791cf14638b9f654f96bbafe52ceea831e2556bea9de2033e5e67aedb4c2c0b", +"bd0b9a6b77c80a21f6b2862bde084972eac85886328881eeeeb868f83a1fffd6", +"14101f3ad8e341ee6f4dc1698c7017d4442e477bc7f6ec0208de5989689afc8a", +"0910ce5a8ec25f0ea0b68854b0a8d3e8eb0c62eb07b524cbc573eaa9c515978a", +"88588f6378a395181fccbf09117c6bf7ff237d6647406fb17d9c37d9a25149d8", +"0635c503df05d4c7792522000bd03f24eb9d861ce394f3226b9f060c7d020ceb", +"78213d2c3b55606c37144b41238f9c542f2570fa701d1645ba58d38b2e838fec", +"3c38affe90916235f0bcc55f352fcaba80ac4c04dc8e2168515f366be1afb2b2", +"481d8c328debe15f933c5d933f95d4402bd443ce841cbcf1655ead76d0c1672b", +"ec7c3f7409cd0151c00c44d756ab583fe5fed84f8814db603cd3fda6b3bf3cba", +"9d42898a238cf252ce4d2895d413d8d3c96101e15aa905a9efb7d8cbdbdf5134", +"9b1d1eaf04a618ecf910da6b91c861319166b9aafc3058ffd24cd77c14f47e47", +"0f9d246f968425d82d181e49d8e450be32c5bb712cdddd6d84e919b6f3c20cbc", +"b0c051e09b83e3b3c3d7dccbcc23b57fd1a98f6714265d5dec0a5dc74af8a8ac", +"e0e75863a67a6621376fa95bf7518d2df07c03f85226ec80c9d2e751c5af9dcf", +"0d2f1793b999109f5997e0a3163090c1369cbd67d9f7054084d6a5c33cd690f3", +"74b87c61314c7b135af9cabf5144088ef35aba29dc97478cd2837d99d2d1c661", +"15fe5fb8fe24ca483dfaa3cc86a477c14fb43f6dfdb8a2f562f2d467038d88e0", +"3108cc6b027e6237946bc42e74970b46a097c9e6f3d4d0d6e9e0c62126e27b6b", +"516e334f04398b626d016e0c56ee6c0c63d77cd0face69426dadbe9d4be62c54", +"9e6b43fd39f9f0c3144997005d7ddcdb3bbd64af92bd78c6cf47eb1b088705d8", +"3d80f3ef189bc602d14e512c6167ab9b6ecfe474488d29486a58b129666dce3b", +"c570983c576a8b69fb5da570e314f6fca38a4bafa24920890827023a37f3fa54", +"f9e253c799476886deb040a4fb0d699e4d9f5931ae14e40232219f8fe55693cd", +"a0848915cb9eef4a07d82159e4ef0540634e60a2d1c2bd8483abe53a489d8869", +"dcc34853771f1edc6c43fdf08fc89266735f9dfb19ec871eeae7367778ba2cae", +"190fc6168a57c4771c15a7d272903a80e395cd0c87b288e748e52eac36b6e801", +"fe4c776e320b074e04ee47d31c9bc5b86389700fb48766f37f10bd4f667df076", +"b28792a3d802c6e4c5ab62a0abe4e5127e7223ca18342d4271c7169dc92f7678", +"bb06f10a5ec821e9e1e37e7dc1e0dd202874d654533130e1af010717cb4ba389", +"5b71a31a8221ac35b163d2bb9ef77a66573c92549afcc0cec4e03e1234981607", +"4e4b14e0d955bfa053ce4a7c3245ab1253406b480f70247d7a5ba92636d13926", +"422d2381214bacf030261fde77f51d0b5b747ceb8d55794e7a2d438521449c22", +"c365c722abaa90b1a2dd9cf4038cb0b497159d84a4fbea01cb4927dd4a5778bf", +"17373fef0a0014bbf38918157428a4bf11736288c6e7841b35664cefbddfc5d4", +"880e5231be0dfa9722db86a78f55c93fb220f9f7244301b03b3d2a2659e425a5", +"b9b515b6171b47940809366f5d58591a56063db03fc39f678a03cb2b455f9428", +"2658579ef19fc30c16a573883ac4b0d97b943fe7f4c8d3437fb94c4aeff6240f", +"ec5c4a6b03d10c95aa67a4fb95a913ddedca98b373525af4920e2c6d171957f8", +"42ad6fba1ce3228dbcfbdf82322b4f2a600fa5a294b5c8226d8c201736d8ee78", +"4fb635d71dbef87e1e9b7208c98bbfaceac8865135a6e0be3d584ff581170534", +"57f2c10948eb0b47aa36ebbd374b4a557c7162e4cd12ef3eacbb11d1f8655814", +"d128466ffc4c05b162bb51ba979b747b90c98f9704ebb2832ee773c3ac195e46", +"22eaa05b8c51743b4e6569b976e10bc293b987e5c6c18c26b65c0a9ad64bbb85", +"ae093579f0656707564e77bf66fc8362ba2bd9139120dc55e686e6ec7c5539bd", +"bb0c45fa302281328b2e3a3c639f127e404f424f05540bdb16104287cc6d201e", +"fe0d8ed75c2a91d8b32327c190b73809f9e9f4ea41f227a2527205b1b85a12b7", +"4b1b4052c3cf9110e0530f47c0567f072aca8a457abd767de06408867b3ad9a0", +"101633e4083da4249bc01bdfce62c02034afb5b6c8a4bde9077a66efa61aa173", +"e41855740cf3aa23299d8693f1117f7fa9fb469b482885f787d008e5469f0338", +"a21d6161f3a480950627a73a4a616821f1b5242bf254a80e248c819f8710e967", +"6da8b39dd3ddccf7b72b010006db7832dd7f5e15c7bd3168333463fa617407a7", +"3f3f53269ca1ca4f8d4f6b68d88475467fa679d8892d951ac35613b69e738aa8", +"c179d7559de70342e5d495d3f98ea803a1d5ecd7e965c0546b694a0960f676b9", +"590931dbe64e9d29c99643015608a68c27fae5a69ea2be4bd26fdec2733a63e0", +"d5794254038013648a6bb807b81d1991cf33fbb49a05bdb9a92a66b54ca088d8", +"dc4cbd599bdca611ddc90a4ec272b3741ffac658c5afd409cc4585ef211f22c5", +"9aadf6cd0aa79eb81ce45a9a444341bb2d7e9896f3525da2f4918ebae3a453a6", +"9153a496c8c2b2a763b309fdd7421936006521f7f30f1dc293c84f3a6d432dc4", +"42f9674f74d582eee387396d1b9e5ab7d82de80e30b2258a79eab8343129b81b", +"55870eb28ab7890c01af34865100a9dea848e86321f519fbb65e28fe677299b4", +"a5df47167a1f2328533338aec3e1ce461e62a54d285ff9af7cd93e286d0d7bb6", +"995c63406978ab13032fd8122e894e096329a0afb9e676a5f065fed3c2337ee2", +"0eecc0e8090dc51caf6527fc05b576f94639d172f047d4cde2d56872ce967637", +"651024c2f5b403fd90d78d9d39a5706738d775bfbfdcd3e87b8fe8c45076c8f5", +"492a62304b27b891995b2cb4dccd8fb87e678e9e72c0bf638349e5f0af7437a5", +"de43c9bb210eaaa271e6ab2082beb64b911aa3548171ba8a9ca18b6f4a6776f3", +"f24a966f093207d23fa00d96d5bf534059b3a6bfabb411b4c3848d3e9f99f97b", +"8d48ec0c5083ab0f5428bf26360e7221fe02ae2fb371c18902d27241b072522e", +"92018026d521ced65ea9bf282b6d1dbee8f72ab32cb8b12b7fa515bda4be60e8", +"e45dab08f0604e49556872a99f5cbcdf8193110102dee8a0a257be9c08c226d4", +"db294a844a56a26cc99e66bc257c08d9d5067ed6370d9b42c2939a05e4d55c9f", +"d702de002954b3ceed1ff3fd628118d74a42cf0bc887413866ac2403b642aaf4", +"59cda35f306894e1b6e9a4b427ef92d6bfd624fcd19e95c6740fb7cc5ce60a72", +"769b64eb3be285fe2cf9fd2fe9113dead0408eab705fd28488151840615221ea", +"1c8e1f67804dbadd9f2750f2e558714bad0121fead00636150310040fe21306c", +"36e4a20442fe3193db200877dd46d763ce79e3855915a56d631bf12136aebf10", +"dd909ca02273d411a7eee842a79a0c9beda9f979d3a813ce4ae5a0484207545f", +"f652a8c9af4391f7b6eb4216107db16cdffb9a60ad55f9ed432930b3db265461", +"b97b13203de7d12249db2b0620df7cb2427e6285ae52fdfb28aa5c2ad8b77360", +"0df4c61bd5dbbae778a595f568b9e5c62be14de45c40294df5795c31d43f9221", +"e0fc996ba9cdeb4b29655c0c85546e17c14411a15961f5fbb3f989f7f5f8aa52", +"aafa02310fcdb44433418ed35683a5a142e2dd00946d16a82dcf827d56730975", +"baf45c152415f62e545f316c9048cae5e19d8b6da792633fd82b55e49262f892", +"09a17e84ea6c120ee4bea8a25223630f04363561517ded52656cdae5feaf0762", +"0cfd610197f872fcd0a7e8bdd65a26be95cff63303cf709015d0d5b90f802af9", +"929861f3e45bee090aee9980e5dfe277568db1b2d9db4efc222e02386efffc88", +"90d6ae0a84ad2090229f539c045edd6892048878d2801d51af0ee066772d8503", +"836b141e7f726d7bce6a5eb0c24600d4146f9bf06335974c64f22f4336f87492", +"2b7365c4a9627a49f73d7c76dbf597cd73119455865633ac903a2cb9a163ce61", +"24642b82df86907c73b133b78448f501fb5b9408e3370bdcc91ff2c1ccb5e7cd", +"713402a6359166b60f3be43795f9311852b6d94ed4230f96f44e690e13794eff", +"a700974154138071451286d230df62539b20b41a00c913cb86777ed412d2b28b", +"2dc24214f7bd00bf8949ca36dac389fdc6f44ac4e2ab2cb6a2b6974efaebde80", +"0c8fe350a0473f8dc81146ce3863302f420dc463986915f7f9ff7bf802c2ed66", +"9c3b58309b3cfd129f6f46a803732446716002f5c51b5784d10a9b7bfed4888c", +"7ca89786bad85851d5c7f07c78af2a55d82d3d012630132ccbc48dac54821c93", +"34ec8111bc0ef280f95b1eb3fb27d5cec116efdf5f341f5c5f882e35bdb1be2a", +"2152cc53b274dcbe7b114aee9a6062c810ea20fb5ac75d8d141937f5f071e4d7", +"5b29d088c5ab34723733e608fad80e0e6c7ca95373617570380f8834bb5b7411", +"9356d379aaad972d8ccd01b42526052de6992979bf7cbfff7864aa33f86c42e1", +"5a4e2ae70b108a9881be2f89b28980e48d4e81614af16be4b6a749511e0d4acc", +"f1ed307454b5e56cf88555e5f8db1f3b179d772c12913cc92b8cd13cbf5e0aa6", +"8ecf7c0bba830dfd339172926ab4ae8fa3d2bc054286b6838bb41e1a519ba5f5", +"4453efd52ecf49e8021779e34de0b95d02270538568c15ac35afc836a2a5183d", +"eff5a7de30fd63106979278777865c9b3a07aa7699ece648e5609e707df2dc81", +"8dc13ba68b4259cf1498be2fa79bce99fcf62f90f3118ca375d92aae6ff894af", +"ace8e863d1501b656c6431d66be68484feb97b54b937fe5c976aa6978f2973a9", +"8a96b0bbfc7bb493fb7e192239818fe7b58a35a161044e112789c75022da555a", +"ea7b142483ddf0f065469fb386afa9a91a74aacaf3fc57a85facd78abfae5708", +"dfe4a9f0e8d7d221b52667d74d138157fb35a789165605d4a36da6d5aa9d11b3", +"888957fb19c8bb1bfb3143090b919b41ddf535783921ec5c525ae0598fbf4b69", +"cd755ee383e30efdbf74e7cd9c5b418a9501b9e4cc74a3c568a5b61854179a24", +"574e7f44eff9051a23dca024f731e3fec9c15e4250ee23884dad95c9fe2c7111", +"ac01477e202086e27a661ede0fee7b0b4f2759a83ea8cd9c9ced3b2bfa05de1c", +"e59bebf466bcfa540d475fb9503b4a75aa40cba89b903b32350212b5b653eb09", +"c816fb1a46e97bc1736a2144d57d9b7c61202685325e4902077ac08afe66db1e", +"65b6aaaae223dd472d64cba5f68996c7fde4f69c4c28b91dc65d9c5cbcc6649f", +"d46e76a0c8559b76ac00bf5a2f5df1aba3aa2fa8dcbfa9ab288027813b627f24", +"ec6a92468759cecaa854e9b0155d4a94e52c897fe96607d592218664d1165a11", +"5f6a12139bc9380b924bed4e16e30596b9119ff7bf3accc871b9bdbf27dba0f0", +"c2d4276a8be2affe6e5e7824baeade0c504e7cbe2d7a2d527a349827b21a18af", +"c26eb08243882c7b3ddf1751978866658b7e1d52016de9b2a59a86d0744660d4", +"4fb823aab5e47c7fe40075d699a7056390cf0b96d8ac78cd85dd417785f3452e", +"5b389164f3c8e9d099b5a59c4c5f97eebf5d5419b665527a6baebb0be3b4b78f", +"015a14c8dc0d3bfe379bb05f6bbd7aa54dfa9f119d6056212f9ff48119926921", +"1814376f9bcbbe6bb15cbcf2121ac0f31816a8c92b93febb9e915286b003abe3", +"201a923eed5a3c78dd4a9b54bce519965821eb0893630c05f3984acce21e7bb4", +"447d49d08a3eb6183af91cacf1920c169b421bd7a6628edc236f7d7b73b9bd1c", +"0fbc97ea33417d675744066d45fe70e7136ccca645bf88a3c1dbff3745ccfaad", +"fa35790c97843ff87c1f30df70ae076ab34f741d7586dfc24fbb39d4dc18c829", +"f4dfee86be991393c624a01df5a7521ffced29857759662a305e991d0940acf2", +"e90f47ea0dc21ebae48a352d34ad97e415a601627f292d3f2ef490be17648bd4", +"16783f0e4a5d86534514280720695825665b2b897b5b1df5e105a1b3b8f4558f", +"8519bad4672807650d689487a8f1ed93ea0d75875b5e5bd3b1e02e5b436fa358", +"f76c2690fe2a61749face27bf143e74a4d334fd201a91c579e96c6388354b544", +"add41ddecdda826244d2df981768186914ba33f995ed6eb7dcca3c6f5bbb8f55", +"76fea291951f12f6d4ef5c3e99c10df53622959cf5406b70360d78fa0d194bf1", +"ef1d43f65222f84cebd543a8853c2b788914c77d6e86a586a1d8907f936d47fe", +"2951bf02e2bd928f995391b37ac9bf9bf777d7bf4c8208968af57cb086111718", +"73939fdfa8c8bad14c374e25fcffb8ca2a2dab2779f9e505a5b6c0f1b73f3bc6", +"07bf86e8efd6b1df59a943ac27f62309b1f364b0dbf245f572651ddef1497c82", +"93d44af05c9ab8002147f4b206d32f9b023f5373737be39cb4ca7fdad2e9975c", +"07394777bd2115041214a71e2f22717d53d81c6e6e69f7687df2b4b49b14bd4e", +"ceb2038750e3a6026f622889e891579d4d24e91adb4c5ec9ff12e008441fa3b8", +"9e5fee97f5d24c2d8d17c24ef56d802b60ef074f71ca2170021a24d3b32ee3a0", +"bfb16aa39bdcc9f429a27f16138ec7362e0becac0788969d24dbd13b6c885902", +"b0fc8033334327cbbdc961a44fc3c4029fe81b88fe41132803626954dc567d0b", +"1cdb93b6305c24ef228d28eff9643bbfc83edb58df735081e00430d7770457e2", +"0d8d190939273706c5b2f6c73d6da73526d153e6d58dd1dc93800913795d0895", +"f45b6b9a7416aa5aa83b0f033bedb4bd248b9fe7f7f04085f582c5d64639e53b", +"f280dfe5b266019e2ddca06230f2e1fa70d96ba8c0636bf2bd05aeb9a4be30a8", +"842c88684ed2da4e7446839bf1e2ffc71b202e31a6fae8b456ba1afe1cbf4de1", +"1b4b0f9e0e1d08e59b714da5f7017e2acab322b5e1bbe305d2140a4605e41a12", +"b7f4586645360c412431e16a1bea0cc3b86f98fcd1a49ce7874f28338698d8ba", +"61beed27fb1975d7f6eeebd07d7c29cc0abbfb3c601feaa2733aaa1d96a9d388", +"0b3f5d55eeb521db6d723bf179198224b11003ecf1ec11f182c3eca7112abbc5", +"a9759617348388b645b07127c8c9049925662dcdf2eba504a50303bac77b14c3", +"86fff7253255f412eb4456c3ef80318671c081e96aaec27b7b90ae0b51449311", +"3aa22deb849e0e2c9ebaf0ce145628b863735436504851bb058b85d616f21d8d", +"9dd0d35dc1b94d282ec222772347403d9f777c12b215dda4727faf422cdbb611", +"36d3492ad5f22e98f29ce0cfe575ce63a3f7a04b599580d33ad8b5251a396fa4", +"39f1049316b05952f89dfe0ea3b895645487a99d8fed522b94390d08e904ffa4", +"1a389eba980f86431b48dd0ddc86deffab02abc169e77a78a607a472f2fcb785", +"9bcfbc5196c5a252b6d87d75ba3c6d13cf509dba984ab34b262a7b40cf24cad9", +"b94771d38835fa037b5f1b6314c96be80516b3d31522b5734d239d9348959499", +"88221d4150b9f0d1724979d1cdd961cad637ebfd4bd23bf7f9e05205e8749233", +"56b466475a3fdf26e5cda437a675f907962b12d32e8f208a3b78921dc8a37859", +"0b4cd564fc665c688935228518f7e80f889a63f8edf44ceb4ed19bcbcf8c1f38", +"3a9a0495bff6e5011e19da484a234cd4fdf65ae8a1edeb676b18ffcbefe4acb9", +"74e9f6a5b336e4270aac4396cc3d38202050b84d65d580f0aa4255400814aa3a", +"79acac470c7608cf3eb70ccc80aae09760370e022f5fe2dc560f5a21ad29917c", +"35881bb03a8166d27115e5162f0b75d0f507cac7c1e5a2d0fa7bd07cc3302b72", +"568b435df91db90f7ad0fb90d5d9b4cc6def653ebe975877e28c81a14d3320a7", +"9bc230fc05458503dac38f290fa1bcdc13d7d6d5832dd01533fe554a845d847f", +"b304149388deb1bb3f4058237cbfe6127b407ed8ecf161024794077da85710c3", +"2df4840889eb2d34a57abf0a49da348d9dfb82f0c68c4a0fba22495baa9f5c59", +"a6358193d34904ba8bb1deed398848a161bcd9b231d9bd657d21fe185f5e6ac1", +"b49104faded5904d0174f084e5c54c16135ae896e0d8d4a94b5d3f1e5d0a90ce", +"ef118a317e9872ffdccec6fb495f8dc947d8b5e493d83eef30800fec515cb1ef", +"56a628c6ec9d0e183c4bf8e631c9b19627bef283e927c5de83f45ab3bdd911bc", +"6c11b26933f3063dc3bea2a0fcac465057ca73fbf67b9590fa1168a7a351efcb", +"afcf59575e99e413bf40fd7d17c614e761a0930039a2c701d4e237ad9bb18a84", +"b1143e6e3d6b5a5280781620c9b15ad715091743bc7de4c0ab310f25d02086d4", +"38a2f71bb34781c85a2ab780ddec07ef7dd7a13ce23165ab11725eb4aa8391d0", +"420fd0ec6f59a1503da37ce8467a6d554944922794cbe8750ce5aa0a09810268", +"8c38b32c9ac7190ca6694e1e7c0da40022e3e9a02445e70aea54ea0869a0dc01", +"0f1fb147da2fdfe64d6ece99f07e3a06eefbc841824fe110e436ef96b35a1d22", +"d2679cc0ca41f8be79a4d2543dcc7fdf2f42ea3c5d3a04a643be427fdce0b539", +"f72124836a446d30325251979447bf01e31c7549b2edff61591f74c8d495e0f6", +"94514171a18c90b3d1805067978ebd1c9e8420ea54b9133ce5e7519c6827e79c", +"a331f1cf43f0add8ae45d1e27ff743e691becfb5a47ddf0afbf335d9d8c5963b", +"3b172eed6acbea718b20e244cdcff248833dcf7b3a7c72aec6285764c60ae547", +"53af8b0766e295e36c2a825107fae5e9afebc612c129d6c9b9f49f31088d3f18", +"217ee161fa60f64ca4261bd85212cb45e29365773117f1be89fe6a4cdd943382", +"716108281f794bceb4706ac207fe8820b3d6ea90139b7ee5e490a2aa7d7ad09a", +"3d392ccb5022723a96ed2b7840998e2f6ea3e5d76ad05adbfde0c7ac908e15c3", +"2a493a9719c03c11032e65e6994219c19f4c81ee1b9590c11f1da4bd972bba98", +"6f108fe86ba5f26e94037c1e0ea6626e0fa526d6af14735e091f6f02a90521e0", +"556c97252f2d2525d9440e18eec7a8673e4518a45efc9b0d2ba9ee5588d4c39c", +"06c5707fd408f1507c79596bf4d1b51cbb39e8eec0ad206744767cd7cc326ed6", +"5db3d17bd5574703597e2fb846b421a4e1debe5c588958a7b03c0cd800bcc7f2", +"2f1582d9863eab19abc98a8f1bf820245af8955bb691bafdf75351a8290b677f", +"1a905d194085bc5f81a51d6c1410978f04193c410cafd322304c9d33ab6b23b6", +"5e7ebf071b1d7b76d502be1580c1e58dfd274b74b17b9e5e510f1afa1c369c6a", +"c2539fcb4fea2efdb1b7459f3598115adfa5caffa1e91556d76b4d471aef8225", +"56673e60252efb5acdc122410a5cf43c9f5a6a56022509fb125b8c4a0b773a1a", +"3844f46b5b7fee24773142097b3dc80c397014cc2dedf024454b9193fb1e2d0d", +"500cd05774a3216b487adb5abf1325fe8f3fad0feb4a4911b5c9f85078bdf7f0", +"14da81f02e68a29953f3a8c395b1ca46cf4818a98a3314458cde3ff77b84d3db", +"785ffb4252448b4a42cf915ac83c4c0cf0b71421e3c5bcfa81f0834a373a30c0", +"64c8923e3688c047aa6ddf17b96a79100024d2a6bb159585cbba90b1684045d6", +"818c13eaf4509e2c83ad278749a4b8163acd53f8a99bea07b58e23c5e5d08c00", +"ff0f62bb8a40886ac861660b18af1b333a8eee725591cc0a7242650c28b515d9", +"846660cc77403b96fb502e1a9f0021793371087654d11febf841a6104bb58ce1", +"a1dcb5610150ae850afe21806b404c36be87e5849b289038a3a4857d9dba6148", +"19c23828a873fe92274bf38dfce673e65caf3e09ff8484ea033858689c480d0e", +"76e776e16c6fa3c026df105ef700edcdd249cd589aa45c601b79ef76ba786b5a", +"4b6a8df6a02588ab1d0781d6f817abb53d27e8b142986a36da8d3b1aee32ad7b", +"0cf9bbb08d0ef4d77b5e0e83b18d92a713a8198a7b8c4de9d4c25b92a5014386", +"43cdae759e6c10cdad953e07d348814685318b379b2600f79e6b39aa17ab7462", +"e98e3e9f81cbc3d9e748e65ddeb3d001721efe02893412c151430943179d5620", +"f50aafcafa591482176abb913264d5d551e2e2569785944f07348c523885e86a", +"da4de266cb78083af982e454488e4defb19d79f28916301e86f05e1dc10b89b5", +"8beb4806122ac789520d3e71ee205ef1f418003f2c3a4dbcf048bc68713fef71", +"afd1c214f49650101d7fbec35b2aa188eddfd01070970e0d4328c921e357077c", +"f893d36a99e2e7c7258af907966964270e2c7d9505cd9b0192a880946f94846b", +"81a0fe44119b491d5beea54e9253c58f5aa030c9ea41273434fdeaed4df51eaf", +"00e51cd4fe109ce4a505e00ce348e04ff3e841925f4164c073d84d638a3bf14e", +"50d7b599606078402612e0f650a24326ca65360d4946cc7b62a23dc0639e0033", +"7e7f03108e7c6ecacb9b0b901530c7d97856948c3bbabad7df89948f3b9225e0", +"e8eb8b5817f4bdb34ff22f224b8ebb70c50319787019643adfdf5900280c0fd5", +"de0ea5004bacb86063e4a7cb57f58640142cdc001a2a11a64aa73fb3e804b3e1", +"8aabe62b04d72c6d151d3f9618cad28f28679d122994f772667fa1ee0bcc880a", +"24e3949aaf10e4155e3cb7be1f4eac2db9dea04e31cf904c99737fd5322dd0da", +"14c5b91a3bbb1ec6e219938a1e099cc72ec44b1a33bd6dc746c7227c522b061f", +"f911c270502646024c02ad50e01c618c2e0580df3344ff172f6670ae8013749e", +"c37fb468455a38e9eff5a78444c564262c330a1f7b07da38c70220e23d1ff168", +"a485c18a1e807694f57393562f8887c215c854a4b569c309bf83a1ebebeaf947", +"9cf543dfa3de324aca1c3df864e025bb978564f44c00adb52edd9e1b05319f65", +"5ae7ad71f06b2727223b76b1bdbae43aae5e3000936988b13ca4507ae0252ddb", +"927a44fe896ad089a853dbe41bbec352e70091fcdc56d1661ada059c2b29f887", +"c2d5332e5697b475921fbe72297b2d6f303fdb66203b61af57158f6f082a3469", +"5432a8729521fccbc677edc93f2adbc205672eaddb4a288fdf7bc30e37529293", +"1537e2b4cdeba15f41e670f69fb69f1c340b0a87c0369ec55fe298f1f4e227da", +"bf3242cbcf103a1b6d739f7e419577469c33846d03b5caa91e35edcd9ee9b130", +"9a08c19ce90878d34fb68fbb19f0f66be2c612a56ecee5e312a6a7ccb70bac88", +"422b73a3b6617212157f5b4265b2122f87b3e0d08eb1df624ed65f7abb1b79c1", +"6d8f052e2b4e842617c8341c666d10e573f950d009d15610e3b4a297783c26fe", +"b9552bcc1e51485dbc28710da11eb5ed97e0881a9dd289ee62185c8ddbc1da4e", +"6b1ef950cc199b43cd6cebf32aadd664da90d5808c797eeb947de621e1857aa7", +"cb1778ee4faa3202110ef0aa3c1a45d325bef6dee6d85c3c94f954eb726e56b2", +"3559ad9fd210412cd932e61ab27d9e7b1e81acf50016716d9c218e5f99dc5d1e", +"ed04834fdaf8d436f3749ca77af0dd96384c19f3ac8dfdfe3b0c68ff66bebd53", +"5ad89a12c41ab9773828961d358e1411877f13863a7be21cc76c73e26dc68674", +"fdec7f88568b3eec307146a2d1f170922f7e135c910da21c6a7aa446d39f76ed", +"5a6773112b8156643f2ad4dd4734e500e459a05c245399b71b0344affbef5e52", +"81112756accded95114f25478a7aa7d5e7bec6f36a3c415b2ca14e466bbf2e5c", +"e92895fea3a705929e77f06c3384ecd6d30ffe2057a0f5b654d2bdc680e0427c", +"158b6d0cbe9424146a32819da7c743756ea9c18c7ce97ca7c3556ba03ac38c0c", +"2bd577b17c182d9d2fa2d97c28a2ba99abcca77034dd68a68ec6fa2c36cb52b9", +"046ac6bd6752e9d3496289d761a49b2a4787a33e253401172ff36392d42db73b", +"d6f32fe3b0e7f75ff480cd94ac77d44c8a82b301b5d1380984eaf7fb961368c1", +"973cd86eaeaab97fa821a8015bf075682cd08c523f9b8b2ce5f91955639ba791", +"fea6e9ce4923da1ad89a8b28fd808496d370e0ed3ce2edb58866bb0cf94f9c80", +"607090de17c627a10e062c9da4f1814c5a1a3800b62515612c2445e0867b8891", +"685bd9a78f0e4c931d5c2bc64811713395af006a61bbb70f682e6760d2ad1e2d", +"9de6110647892e3de1d6d5c72bc19c1fe9a3f9ff6bc6b02b98bae900444ad46c", +"04e59b502bcd1886fb516e0095a819699e21bdb138694497d3b65d913a0ed011", +"e409e3d99ce2007ae1beb5621584caeaacf84d8388dd01ad0fe2b17524b79551", +"197e435c3fd80a2549037504aa0d4d450505ae69e09615c9ab87bf6f50eac1b3", +"36b47ccc67090621f690c296ffbc2bb92aa366b9467896ee93cef2168f364331", +"fc30a58741bc5dd02131689c6423017b60a70a795da14beb0f6f1fd2db3e12fb", +"06f66590d610151f94870cfb2be27b38dd798bad8d0c71c3ea558d09ea8911a8", +"8b560155c1240016b6f1400bb53026b0791c7c0d784f0afd2a31251a9b5c5dd5", +"661717ec666317605d3ccc4656e8dfc6d98556db2853c393ef0deb9a0b4f6fe1", +"e475b8b813803365d0434f1cce9de3b9606435b1b766d78612cbf218249a82b9", +"63942840802540be07abdd2e42569863f6af11c3be95725c1bc1aa316eb805b0", +"3e343bb863be7a3526da971802089970ed815e75a6b8f66e97e4d7751634f695", +"a9c381ab918c40680100ee69e73cd6eb69bf7904ca242b2f2ee9b9715312c3d5", +"ffa1cb97b886bf2c0eb4c3792c245d21850fdcaf25f8c5550a2d225f83ede50a", +"89f1d65504c630c0d1f51b7c63e667d688642859844da35836b5b6ee679c0f60", +"7dad48614024ba1e2becdcef5017e14c02b77eba6d1853b2387655853abcd728", +"45eb2e3ca7c3da8382684412a7eca888eeea8dcac7bab4fc791248675af414cd", +"6bd1806df951cc6074b1c1019e3cf27ab493e421bda830ad54c2965badda0a2a", +"0b8afd413460360a6fb5c4c42241d3bc1811c36461b2cc17d23984773e638e70", +"ab888d167025c10ff0f32d7597b7dbcee65be7ca9effc3a62ff8c70dd7359867", +"c1a25668b3a665e9bc4eca605bb028355a9edd8533f9af7db56595f93fdf476e", +"30af43bb9eba76c66515354203eb790f122af9fe9464312247275f039946c061", +"d4b91a6e34cdda9bee28142e877a18c47e6e2bf6a84cdafae98a1547a8585909", +"cf03b2a0b02d34d50b7132beedb5e078960dd5beb0d6f9f940734ee15ca7eefa", +"31b5aaebf0a1ec18f0431b7e5245ade279cec5a41971176dbe04af4322f2f8a2", +"5c579402f5524a4aa730b3f77d3eb5d9b46ee68158288745c515dd1b65c1e666", +"0deff07842092b641c5f003f85b19eb7c03be71191ee1b4cd7af6581edbaeb76", +"a445248a3e231617bd46cb6ff42e1273bcea04f86394e3088279594d4b58f36b", +"6000ac5b4e2494cdd384e7236e12bb8d2b5cb706cf6336ab7567683b442dd224", +"ec15cc1674e7a9d565d941132971ac7db252f684fb4fd4066427444a01fda5ec", +"f348935fdc5e0dbf9a922209b8c57a9c5258b572156f9e51b8b1a5292ffb2bf2", +"2829cd35adc5226ca37e0dfe5e33dccb4d444e07add3662505aea5a835c8464b", +"1ec355cda7203d797b1765948ab00bca9d915a4408533fc316ff141293fbeb3b", +"3811d5d6aa94be52baa7c848298d8069d8ea5e9c162f0b68102a25d6e0c25405", +"1f7484ddcf8984cca5e1ab7e4a6215b009ac9993c45703fd2d59781e191983b7", +"4dd70c7440fc53726c040a6f0c6c05e11c8c3035afaf93e692e35b2915be55b9", +"70434916738ea44ca15798387f0ac552d2b1a0736e0846ba19d883c7601ebf55", +"296a822931db48fd38c0f4a02e3a54cbdf16fae0eb6bd2f320926eb5a9eabce8", +"c37c81f56f36d4c08d3afd5072bf8779b12949abfd2d081af230ada474cdd45d", +"5477a03998c161fc5813272756d0874943b56c1cfd302704062e578d289a97e2", +"130bb15c66e7692c7b9006bce044ef5b59dccf9447000f4ee7f1bd42e07015e0", +"c39d2c12b2138bb255d3fe116fbfe0a1e8c956c859aa20f2cef45732445d21ee", +"530f7730d4a6670ba2435635a04db586dbb1985ce86459086d96cb46be174a53", +"20ea71321c433eb6221f9f7baac2d61cb06446315d517e3eee827b6f73469306", +"88410e3cec606eeb8f541059246f9eec996f8c9f0e34b8a26cb4dc074eab74a1", +"5d48dd389bdca9ad50b448c514be1568f401ca191deab793a2021bde0ee2644d", +"2140946d1c013b4b95eac0a05c16ddb3e5e486dcd6b23068f8cce0dc449db812", +"a7a0e921e577b1d7f084d30100026daeca01b578d487a9c72cf90a43e5f3b0e3", +"903d757c6a163a3ea489cf28de8c37ff4fb81b7977912a1f86747971752e5ccf", +"31e2713d95f01d1fd4547b633d74afb3222ccfe73db304a335501e955f80ac39", +"928b99bbe32706c82832dbcdd25cfb96d57ddf5e531c09c0631761922dbb1cbd", +"f366961a77955c9bd8ec42a120a97b0e798417b53517e31d0eaf7e8259de4e72", +"0b9216a8f249d1c2ea14d6b7e34f4596258161f743ef0bc2b27f98ee2b558ab2", +"05d4d804ecb533a1c9a0ab8c8a7289b0f4707afa603e4956b88e5de047a4f5c9", +"f1ae6b8a1e3ae6a8746d867ac44807b86dc72889f415eb543bbe6018bf2c6dde", +"e1c6c8650c2e262f08e1f37f1bcb80f9bb1ccd466274360559c345288cb602d5", +"f9c129d6ee345b631a7987abffd588e043245fe23e2dfde3d7c1fb5ba9bdda34", +"c062954d1a4e68cd1ae2e972d8dfb540dd12ecaa72269e5f62943fb9957437dd", +"3fa1b8b75472a924de57a6fe84c1f64afce2e7629f9759983e823b18bfdae81c", +"c016cf9ad8fdcccf33a39eb7878cfe6ceec4a716d4234f47562d99d0cbfc026a", +"a0f8cc8aa6c395c9205ec70aee522e2def7b6b1e99cd71cf0aa0de6533c1a56a", +"ff2d506b129521e85a2602086216efb73d5c309ca284b20b15bb63f1a2d8c7da", +"08bfae28c6be7c32ae8bbea86186e47402cbc3ce70e69e753613782c656b74ac", +"d147362789d7f704b438161d0cf1228df40c67600fc19b048070d735d0c3262b", +"dd79d1b9e73c7d05ea6480cd6ed89a9a2bb2abad67303107b1cb09c7ae0e299f", +"c087efa078a3d8b212ac0407b6a90164469eef361c6fee9c5b56b4540834d61a", +"f87fed9368cd06659cad3d76a6b8dba70c6b8d92974779fbdc7598afc869be36", +"dabf1c832bc4f6d086ab654a775ddaa8522ceb770747ce57cafc2ebb09e302dc", +"090c0fff85c93a288d6dd82f59e6c0fe2503d6c8ef1991979399663e91557b26", +"15403b21c9121226a27f89e7c077f70f431f51006ec1c6c0035f8c6a035c98ad", +"c3de1cc3e93ac56d6d779c989a8ee2c6eb658d8bda3eaa8939554bb01c1a9cae", +"711a81921099383eccfdd7c47f267954a706e47a89bf6d7265d52d74622f1791", +"54cd546bda2927cd8af5d9ff7d649ab35e8a5d92f5c3a8708ec336532899ef4d", +"2ca16f592bbf77c311e2b49636d8891a8452847f8a0f61c5374df3603e11e8e0", +"1acccfd9134a3d7587a85b32f414eec72066891045336e9bb6c1ecd35202c6b4", +"f8115a1b958fe276baa599d951a8bb118f6317c3c5550000aee5487a5eb8a8bc", +"dad984f66d522bb2f71a9a1a5454ed10f090ac48d220a3ae9cf429959a3cab05", +"c59c09f01974c399ed6948943cc3445530c972ca51031e8b8f0fb1ef3b987156", +"41ae8d981e75a54402220564a078d5ffcae1a74650b0a98667e073a3610927b4", +"e570cb7aabc88811c2ceedf37ce64b814253abc5f5cef9ecc88ae9e6789a9c2d", +"97de7da8c04ddfeaaa8cbfadbb63e29a9b5bb0d7ba27d4214c9c363817fca8bf", +"05e831824eaf2b0610b6f511471aadfec9eb2b088db70359d161a51e767085ab", +"0ff69d3309fc80237f4fff87aea27cf1db5b03a5653de099f7543ab8ef84278c", +"8e3a97745b758681bfb940dba95d63a759cb63d10c55ffdb98843ac014fbc260", +"6ea95aa5a2134d9c6185aae6b51e92b4ff44e811cb95577307aa552eda00eea6", +"7c27d4483e0e236b3d8b9b365cc9af3fa466e9d5ece2293fef59922fd3f258cd", +"93004dc6c470dd1978d0e9d3e357dc0a2a13401313fb5e1a2e069a14d3e09161", +"96b017365ee89d873059bbd4188c8c85397c1e4a55c79dfcaa6f4840b15e2bc4", +"239abad46af37f676d73b160d653d43d8bcd21818accdec68c2ccfe05b767c1e", +"666550865ed723e6fa63a862160a016be5e964668913b2fc94a66986a3973a26", +"ad1a22a1a56f9ece93a9f60ea1cfc99b28953b6d1c7d37f5267185a3b38dd2e3", +"f338ef264d9e2889cef554c3b0535d2f04d4d63028a5b31326c774b523099a6a", +"4e330922eb98fe785af7e08855ffc179635a6446fbcc7b03037d08ea53a47a7a", +"e51cbe6271c5e60d69f81019e9bfe11d65f3ff458f812e55d256b22e7b27cc31", +"da848fde5d4a9c17326561f2b2a02de1f111a6c66a0f08dd0357fed1adcfff78", +"7dd2f192fee367ea866844722f8dd9714417a24bd19fa4e66573758eba95e1b1", +"f67a541344e08927d4f96275f4e60c86a99eb07284cabcf713eb9d84290dcc95", +"9b62adb07745b8afb1e48877dc2915f48c5dea19500f8657accd2d41efae5f97", +"97d1b69d879a0e150649e823ddb65b643dba602061f5a3e3c4d11bbb857b4506", +"1744dae99992254e3478c80221703953b4e7a327b18a7e196c22cb7e873e245a", +"651d3d15cd67c720584b591285ee8c59090a0ebd63ecddb25df96deb56c4beb8", +"3c1133976b67df863b5ec1cc1ca7bef319fa89ebbf69e61b68e0ca238558e194", +"fcf954cf99fe5627f8b2cea1fa76d56872a2ba7c5f1666bb8640227263e3e6bd", +"5abeccf01f8074a8004b4c10492de2274eb6fe70fbee2f07da0e9693751fbd4c", +"86e07b9b5052aa371de87260945555c358155f7bc461e8d5b50ebd09d19c8380", +"f839a3289a26053c41f8ef47d4f91eb0f6fc7be7f41eb2c64b0052710ce2ade0", +"8ddb7fc82ec8baf3c4d3135da8f40d6468925d01dccb01678351e6a19dd8a187", +"f5d2ffa85eef5d9b1e0be73942a7c8d179e2fb1d81124dd9d6778d2940068a7c", +"09e6c65b9cd9b595b77cfe5c31d6c9eeb28540bba0494d90939639309d600e39", +"2856bb4324a31a7a27f801efc888bf85a6cf8f7df4d19c1013df008a3602ad15", +"c25f73bc82e3d7d6f4197d3a7073c5f84599bb6ad646e6b543c0b22b2446c3e3", +"e5475fa306f96418265f79674ff58bd5b4690bed88805f6ac4bec0dd488613d0", +"b3a06366e8d84f9da4c9bf85ec1b8aa8eceec3b95543f214c621ee6b2580ea62", +"eb6a5bf60f77d656d6205509c2611f69bec685d147b5e5538d4d03fda578d3de", +"d7a8297ba332ac9a518fadbe70bff80ff76686f1ccd459e92e0a2925730f0690", +"6b474f895489a8052f928dc8b258f80ae24331fab35650b150a11c396bcc378e", +"ba0777fd2d720c9dfd4da24b7eb3f2c658c5f872263d8c52584d6d1d3634ef5e", +"57f933fabaa4675f1cb84d09daed8df8ba6ee2497eb1ed746be1093119a08945", +"dbde1f7d2b5d2b15b4afc82f05d8fa176257aa5ff3c68c91b57f878dda5218e2", +"a7d7b66f67ff1cbb5f22d0d4cc6ab42611e31b9d9dcfd69dde62c5aa3b416265", +"2064b4c6f0b236c18a55737212689f86a1c0884e3a6bf03e58cb951ce351d7ef", +"0791e8bb1bf90039eae6e473095074f0d01a5394c7d09fa3c71fe9b6fcaf5d99", +"e00e93dbb808e4912e4fd9c9227f41fcc7ec70e684f30f8fe9a9a1d36c8b0733", +"4b85d1b19deb737cbbb1944c6968a331f4b7f1ff7f21186b3a588bb6762962c3", +"4c89859bf389749528907bfddbf85e9ff13448f1f52a72c78f3ff4fe0e3625dc", +"bb29d52e697f3c0dbdacf5cf75704bade769a25d9e5acff57858f65e8acb10f8", +"ca8e4aee00a80c31b5b249efa9cc3aef465cb810f918e3fa1a3ae0e4d8fcd839", +"f4886352cc49a8aadaa77022f051c560a0fc461cedf3a429e4fae07346fbf746", +"f35ec75fdc77f1b7e6ccca702bf66dd787e6c3165d4037cf2943c59e5fe63839", +"7d238f7bac06633852f23b44df8a96d60cdabb24233c6f2db52a4006480ddfd8", +"5982a1b482ecbf33192fd3425c05210bcf500dcbb3df20abbb7f21cc73d18766", +"b1040c4ab8942c097e1a0a8de7bf2a97cf798ab1b0402fe729ad3b9faa69c65c", +"5bfddc04dd473fa5f66d64953db2ec54345bb9243eade72e08185fd90ce4a39e", +"0490bdc0933495c993bc551d6b34b7dbcb89a807f39d12d9118de54c3b2443fa", +"3e2ecff17d797418a1a5d76f187e04589c435a654cc673cd26cf956d2004824b", +"c0eab7e1d5cab91a06e028929ccfe0dc8f3f8c5b324ac3ff628c12d7cfc117e9", +"1e8dd11dc725fe3f65f3310412b9d56e9d8081609aed8cb55bce973cd515fc72", +"a80873c4a3c0cd43ea1624042aa72fa1de93730e52cf4aa5a9f6523d92d97790", +"96b1d4f4bd34d18af1d5b9a3c2b813050f629173e94b6399fa44b5b6ed83b63e", +"cecdf0e17c2e8d917566d25e6d77c2781192e12906850167c6f84b1a718a7923", +"03bae5863b110431739d2343b865245c9c1ab4da8204cd5fbf5545590d31edc0", +"624317485b0dea29daaec6fd6bb7fda31cc6ea00e3efb5efed4aff14d0068a4c", +"350cb1f7231b1ff85666c2fbd77dd406ac72691de726950ee7ccceef1579761f", +"7b23a22bf29c5870802b9daa3d1d8bcc08da974f59380623e196403b635f9789", +"bc84ccb74968514ef2ff5b56caeefb1692fe64eebb38bb8e361519e06f92085f", +"953b9a5b1ce8907ad7497c3735052c8efca51c927cb2c72bdd8ef5c8093ad928", +"8bb000328b309efdb77d70800b3b22522b3b1df74db78e64c07688ce32b6889d", +"45da9ce932d5a27515b5706ae3ccbb005ac08efb276a9f4e49becceab45aa940", +"3a1f94014461289c06602f0807c16c202660ea67599271b339bb10cc246f85bd", +"575996b053ef583b96a624e9b3bb2088392bb97d5b9e55a5c1b78e54d01ba157", +"ae1939b4f3fbb1157fca1fd940b68ff3cc432bac82dbd159447d5b7dfe1d4555", +"52ab7cf7292824041f8ad6d45a64b8cf768a19b5d21e6b53cc8072d2c3df69e9", +"effd174ac956c3bda653da0d21d84fa5a1bc0a12513632a69c70717002159629", +"c9ade905c330beba2e75418d2b41e589630104465748217220c87fbe466ba118", +"dbc8a49e7c149d4993d5dd89b58723a69c6fa9ac8a7e53133cc43ffb999edfce", +"2ebbfcd5db130aedae0c2053d8f415dfccf3be21c197b937218feb47b81c388e", +"f34b2aefe1c19bc24688576d3e0a49c80fb529accf23e15f11eafdcd0b9180ce", +"83cb3b3708d13b1e4e54b5248710ba3efd3fdd3f1b5eb3178d232732a89c488b", +"5016303f5807c16d2228e13f9e46eff5fce2c138d7845b86046b4ea6987d9f7f", +"1f667f83543442da94b0a414d9cb0e06d66216e91efb083ca883ee6ae1421d2e", +"7c5375fb3241169471150411cd1f3c34f3b7b20817b41ec80d521e88c15aca28", +"5174a2c02a0f2296568a4e157411c000433c89d718e54603780133b4b18e66a8", +"542aa5ed26b440a5cfeafb15044601b3bb5cd6c0941dcfffe3542ef71969dfac", +"c88aeb00c1d40dc6177eaa3db6513ce2c628551ef79c20fdf05e9421eb54a8a8", +"542b2f4401a7f5aaa3781aef83889f7f48f9dfb3eefe6c8b51ab1f37ff916735", +"d2832a6064389fc2fef5b60c07d448f0fd8281bd0830371ec93bce9531d74c6a", +"f67dc4fb5cd09c2513b293727f428cc5e87510431ec94fed40652fdaf0248673", +"84f48b67aef958b5a386b607949f45486266b34c5f9ba3d1c902f7d33ad99f93", +"67a54aa6bdbe711d12100192fcb22dbfd0df60852736485537dc2dbc4d53f43d", +"3407a06ca624ad62cdd5e94764aa5eea9b31e42ed1ffcde8ee443756c844cba6", +"137198dc7ae259859b651cd2f0951627bf701eccff266621fbb50069efca09c9", +"a3347bd5ebe0e33bb53666eacac3d7f98de8525dd9d5bb4140f177a21410d95e", +"7ce153a8e9b6e5c2edf91fc7166e509122d817fb1eee02eb97f9ad78af8838e9", +"3261ea87aa7e8d461e6f7e8d0b1e6d5418e1ea542ab21472eeb0bc2c997fb01f", +"07799483358740d519f7cc57e0426027eb33ab3012fa5d565f5cb2fadadde7d8", +"70f0a1af051e29d1f96ea95b5957e3dc193d17a093f6c7d5a2b386bef82132d7", +"16fd8522e6aa9171c97571b7f14cda309c75fc34293c2989da0634522d9dbc0e", +"5cbdf4b59550643e41cf560cfef5da0a1b4b8eddbb3d924be04410376a0ddad9", +"325272eda32504f899194b2729578ab8a8c0e3a6582102b683bddd1ae7f1200b", +"f51119d7b4a3f6c79f0efcbbeda584f1ab39591fd8e13a342c463d289e10b481", +"315eb3c7e15dc51ec505f2149e50ecbcc034df54486c64fb2017902d7a557693", +"1eafaeb3781779227102720ef8c9384a058e60b55fa87cbe02eb5e5a9bc8a6d9", +"71f0f8938aa0805e168a744e482bc5ba215a4718ee07f0e37d6c4dca84fe5a44", +"78b4d2e161768ac9d84a4a67f18a1fc759d49353f6c3ef0a39c49d7b91431fc7", +"8de3a0296dfd7dc41476adfc06e4bdf447227e5dd0d4e9317e32bf2231b900eb", +"2d29120f2cde14ab562dd6f20a358e175748ebabe994b2cdcae8400c6b899b70", +"445c1c4156a3795f3f320f307b2fba6ea957d8448f4b10c5241aea2adfbb764d", +"2a57dae85320543d698e929ad435447f58e772b1e01f56366903dc43702ce609", +"31fc57f6acd3b33ae7697f3eeaf817de2a6a9f437f152c4255fe2a25de292be8", +"cd0fb3c38bfab2e68e32030e1661596a7c99445de845012119702ff713156095", +"b55ecb85efc83512e8a464ecac7e1a02e5e827461665645fe8260d7c295159cb", +"1fc1339bc37de2a72b9b15309409727aed46bd2cb568f33e709f897c54dfdc9a", +"e393ec0a65eec730e361f6540121aa1dd2cbd24dda389ca4a717b9613d2570c0", +"e5faeb238a40eb9e6b37c49d4d56a950cbb84427d1bb4815a7e9381402f15e6c", +"b09703aa3602aebfafd0a9a26f8e961e5ab02fea46b8763ccb8777fd574c3870", +"3f31000b38fb2fb8093129ea7f84096f933f4820c6b255faae747b1b0a293c89", +"6dcdd3ea6074fdb9e7f33b965e693c53fb6b48d88f573b085475374d43b30746", +"49f37ffdf9f4f3842cc0d02e370634b3592588c2cc68675ca30cd40e2d6b2a06", +"8110af875a723e73df94f7181ebd915eafe93e462c635cb69ef14f78593c961e", +"8155fff8375a8f96dc2e1693c5ffe5030d5db906161cd339d849afd515acb075", +"8e7fe0d1e7dd571f8a298d45c06ba5f3352ad6d4ae32bc051c0fcbc8533c0f86", +"36667f7cc0f220a4c8d0bc503abe00f8dff1839c22d77d4a525004fc38f7fee9", +"365d15023ad3cf01dc64efd7abee52477376594bf13b9cf73973fffeef536dd4", +"f366fecda0ac75bf2b7be980fc1facad7343fbd1863f9733e7ae204284f0fb3e", +"cb3c4426481a51d65eebfb245da73157ba1a20e72aac408f1f6b813e598be767", +"1dfbd942249f1014eff7f699ce0ac12cad661c5973d959894d9701c8d3b4480e", +"aa802720724db4fad5a7acb36640161a964aa9cf01e917768efb377ffe94d744", +"21d58cf7235903a0f65ee22cc10ab07c86df62e006da7985d86f9b08dcc0753c", +"7be958c8be68558542213fadf2a5670838b7c45090530fb0c678e1456ecce593", +"6671528f5ebdaaa75afba0ae30f0a19d11b101420a5b1c45b719fa79b94451b1", +"fd3617676ee8f8aa39c079ca7d90194b87ff560ee44cedde57bfb8fdedefcc11", +"44d41293db71865529359eae42bf18a97d57a1fe52cf2f4b4417eb58f34e095b", +"ab3d9b4567447335874824ef294eb60b243d06edab62ea112f489030faf3ce9e", +"a62ef44c94bde81531ec6938d9971802c6b01492bb487d552052ef83707de93f", +"3b2c413bb3a8ef0fb32d6ecd455d3a0c40381c2fc88def6a87a69f8ed259e04b", +"f71e037227fbac50ac83b2e6c2c235ca3469d0ac4f78a5fa2bb2b1cde92075d0", +"cfd8a64634075a5701a232a9d8c88cff05a4246732353b6e6838d0a3d32f673e", +"6adeed92841ff92692ec0aea23818ecb1d4e037b2ab4326877067b2c797cc809", +"bfc10f019886a48b1ceab70bd9f9da45a648340fc3dfa63039c74876d8c9e749", +"d7a2791bd4b91f83b028ea464ccb71471cadeb15432ba0febec6b444b762a7d7", +"e526ab77add39fa2bfb4e6d47f69adfa4a24afd9eb068629324b526ac89e87a1", +"5ad2b66b5b883ec2bdd3eb2fde47998a3f629f3274b156543b5cefa200833962", +"7c936af2f89636ab56e1ec7065a42fdf2c82b4177d42a48326198b66cb1a0e5d", +"e1567e80b846ec7b28f3c5fa09a0c05b7799b5b70eea2db7f5ee1dfa65f8b7a9", +"14dc80bf5e4b280a9f7b4d7bb2f8f784a233d62c3bc9a7a5b6de1efe9dce79cb", +"9457b7551379a290f80d2eba79f9913dd417e21113cc3691597c3519ec39ad38", +"aa63d04ba24a2009b744888777e8047ad6f4ef85fc21293fddc6853ba64901c1", +"fc2da1d1b58533985ce7d2d4903c2fd404f65926abcfb21e4992feddfef95d92", +"4a2f0a8a367403530586ea1e2262b810eb1ec80594ce0f4ee8eb3c69f13a7344", +"11d2e1f351394e9afd3fd561d758c1e23d65723f66daf08268ba07e794bdb29d", +"751b69a45135c8d1c16ccaa8b340aba4831336796158f28b216c341bd508f8e0", +"a6ac5c479e0bfb55603721c86e6783b6f29f1e803e8761485ff80de5ea4609b8", +"a67bb2e009d28be0ed05ddf58bb05cbf590993ac586a84ed65b95d7d7f363542", +"b54ea07a9da21367d1e83e979163f6dac8d05e6a748b95328b25d4fbc9546520", +"3f2897592602fb41bf4f244a80bfa043fb3c6d29b827673dc9933c0beb6d9de0", +"ea43ac646f84855be48e1b8b6dd3bccf11c2d183d15aaff37730e27a7f0aa1ab", +"97e0ac16d62be1b320b11d23a2ce09671604115889c09f4070d4666a26821693", +"79c511a99a426c64edf129f2c272fa7f3580e1f3b41a838d714ed3aa50c4d991", +"a5a042f615d022be9f2a11d88e4ab384f656d9119cb013543da577d340687dc0", +"d01a918d6de91bb81972a85d28e944bd7b3c1bd37180e144744063dbe1f7b5dd", +"0b097e47f264d3ca34718261b92242841ba9d21316c4eb682d54fd48e14d0fcb", +"6644192bedca2dcb2c4b193a91b91edf32193b4e7e524cf42676cadbff6aa8bd", +"ca371c508eedf480c948d905fc079b3e5ee06bc49e9678a56a606438f98c9936", +"ec525a76e5c361d26c1f2bd1edf8c9813365d174a09c5ac26d3fe23d9e35b0ba", +"bc596a5c0f7a4279d0981c065b0c257d1f2bb646346cea24e2a63aadd1688136", +"652dc85226a255c97233f1f5e3ca922b9436e304d326c1eff0d83a3f0e6a865b", +"99c654266f47c0be61f965505b6e903fa0d586a04a743e85399d5a0a83630cb3", +"2c44b87c4974b394417728de9649e96f2e5d3d3ecaed7a3ff55d63b3076faa90", +"0843a125fdce55b2c87431a295a90732d02e8e547fb0b8e53e9c088d9e2e441b", +"e2042fa7d7cb4da795c38bb17b76bfb94965e0164c0930362db0c28bc06e1fb2", +"a2d9970cb9bd25610e22db0370bb8ac8acd2f5e751948f4eb4562c6594a3ada0", +"c799c8e22fa4b7c700730ecc48cf22504e0a704ab51ab5b4082e9b75bed526dd", +"32e58de7c1d857c53e68d4b2c8de373123cf118c96c4251ccfd7e5db263444fa", +"0ea513eb2ec4233f630be2a7e770334c022ed693860018c73af89e010bd71d19", +"aadf9484d610b7f412ea68773b9633d9c042993fbd304534b27338bc3f8896f9", +"3629de258182cf4ac67476ad6eb39e8f8611eaceb0404b61d23f3833eb0abea9", +"9f2ac6a4ed7ae1e9b2b2826377c13cadfe9985aae0a1d7629833c9806cc83047", +"9dfb8ba71da5da1f65f919718be9bd2dc0c7cff251f5550e539831e52f9b7077", +"e73f78835675822d94944052ec91c5ac0ba28f4fe1cc58f6e1306ee167ad51af", +"95051a11278974f4ebdebd576b0c5c0b53e0d1c4935f299e279632798ce749e3", +"dbb3b10ca1a6d0653bc1a9800de240020d6ac9d89fbebd77d6eb530401c8e5e8", +"041fd3a18ce6785ac452ce452ddc21ba73de2e1788e9f4926d7912c66f089bb1", +"6badc744d4a19ce3e492d3b5d8b644c754a6110debdc45d4c5da8f336c8540eb", +"6fbd1c31fec886bc6763e714c59f1da081bb32edb3cd70ab395f0549cc482df1", +"8b9a53853b065136c05816068fe0ae929e570f4357f8d58bcc10655c2a2c0cf4", +"7758863fceda314b0a792657d3ed3a71194d496caf394e72fa6436007b6c0dd9", +"59267ee7cda1f660422d54c2782ba0cd26cfb1455057a6cb6106c58aa6205d0c", +"2d5bf0bf0777b2f914164de89bba6321000f656814e15779be1818c8cb0f3ed2", +"4a66e0756a826dc8552acb17a13d9ad41d7f7ab8b15d54182c0d18146052e08b", +"c685209916e58ab1377d531a6395edd26eb65c4eb2c8d0647ba452322ae93161", +"1eeb8c671001d602cdb54aa7ccf3f4ba3e36ccb8669db6070c786d680827d544", +"d14ec7c4d565b5679e996fe49e0b416432ea7e6760665bc233c7a65f69621b0a", +"42b8a782d76195d7813e3e69aad347199ae3a3824e4ddd116be7f834bbafd8bd", +"89eebff0a487f183bd971cf0a3348c8f8bd845a5e4b400d8512007f550578ec5", +"2200d50c8f74236f7832f49cff4bf957ed8be648236855355d5340028f2397d8", +"520cfb216ab710cf0a8aa110aaa0f57eb86bcc900cbc3da339d326ed61318f9f", +"2bff20008caad03ecf8cc9fc863174c6ee00679f8adcf3a806b2c188e6a7c841", +"837a1b88d7d0a0e2894e75e6276c0496643c2b3f6c56f94187886cc2c8b62cd1", +"78e10c08c78c032bc73f74cb922398e1004c512e0b8cc7f9e157821e25a12141", +"d70ab4ba77730304d5d4ba1a890536f689dd31f1c92c16f5f210e4c3a1fdb036", +"62794f160c85497efb4fc594bb01d711dd4872d14382afb9edb362bd4d7b163a", +"695ee155ab71adde88b72d58cbbfa6f83bdf2cb79a151f248abea26dca04e88d", +"0a8f5a644bfb2e88ca620f47b7cc3fb1b2ca4ffa1e7cad4f9c5e56c6c84d3959", +"4bc5ffddb0975fa58846c332680c56b72083a3a0b4054f6b8a3f91ff8c3636a4", +"e0d08d26194d481fedc967d5b879018f1940bca1edf672117f1d53076063b6e5", +"6df0d989547117ba1bd109a3e35329ffa085bb7f1f7b5663dfde4cf70a10317e", +"63b6af2f0253db0765575ae7a46c76e329a5553b36f5bf1d42737041181d8f10", +"50b51ed2452a11166e7e33d33e33669cb7206919850f396d983a84c1c3f343f9", +"4e48a3e426557e4283d6235739f19e00cbbc971dbae626a7e732211f88c4b116", +"229399c8cb0e5809c0e2c61b589d7e4a6ea418ebb78e13a7f3246619c9b01fda", +"0dd41dac84856681fda3f92cf58341277a74e3b3a3328c4ee8d24187a571bad8", +"5c5aac8345686bd466014ef256a503bc896f4186366fb000daea1f90a4143377", +"9741f2cc66730e45feb28d904dae8d5586797fe832aa78c66b200a1616a526e2", +"74a04f6734ad88b61468eaecbba86e10fc5132c461dec26a15daac9b6e3ca191", +"9abb2fd6c1f5a4908c6dce248d3f2b0cd159df57967bab38414c52a1d6048451", +"dcd522b3588c7adb0418454539e1a929fff936f211e5a20383fdcbc3ad8751b9", +"fe27a2d8cca2be43ad124e6cb45bde5d6851e3d2eb9702e8a3bda55f7eaa01b6", +"e5b9466fd3be35d969f4620fbc02fb8d5259b3246f5ca759153276a5b82570bc", +"db40e013c249571a1c29c19f1076986838edb1e9314911d0742fb666d732d47e", +"60e5386bfece072856f5ce14db38704422ca9b5300fc495309bf583f836d0a37", +"f51a2c88523a07e0df9abcedbce12e768359a02704093ab088176f9e64880317", +"b8f18f00a6cc8e5677ac0315cb7fbad9c8bc7d2a743db76811bb51968cf4f360", +"fac85863976a508e52720e25d80ab2c1740c524da9d5918c5c5264eca745d6b6", +"19019666b54a00ae519ec32e73a848416d41acab276a4dad5f7318bb9ac96fea", +"cbdb60aaf21ddbe880c3ee8bf6bd214dec5d6d8c7cb5fe764ca940b810b8cc80", +"f89b2a5849c961c420c17819571c34982427b9a05c5ede886df502643418eab3", +"383d662597b21a91deb0debc277af70dde2f294bba8ec4c11e96fb56e9c3daa5", +"73ce2b15f1c5ff2a0ff77ba7ea7cdbb8540858f51302e466e99dbca2289d55b8", +"ae325995c208b33ed9c03f6a1f2489a6596f1fefa5e1d0c20d71832eb0c4ff61", +"f3ab26267ca7a03dd265362982db98daa7c6e0aee3e808ea089386a7a8312bf0", +"8b9b054283e089fcb7831d7ce527a1fbbb58996fcd5aaf6a1d3fcd2dac910a39", +"0d6fc0af665e0f5fb1e1999c6bd6bf6e3eea3cb4f24a22cb8b12a0363d234748", +"b229b43b878148d2561aa606318e4a4d85554f08e53f1fcaa43c6cfcf76dbdb4", +"579ec8430527135d872bb3587b5fc13412031bb8897c6e29f05df1cd2af3f249", +"4f20d9828a7edec67439e2d8a2e9d8e0b0e38aa8dd310a2572aea0f30b75b0d3", +"d61ef56b2451012bf8ee777670644bcfef1c55f314e0ea97ea8fa049bec38bd0", +"e1b968d03fc7bfb177094d14770e1d76cbe19ed2569e9bc169455cb83e37c90d", +"04ba9160b2dd5ec462d5e5012ad7a3486b24e195e239c4e44e3ae5c99c7c54d7", +"e8f453621d9c0cfdbf7e2db81d20438edb164556ec0fda938c10773e8205199c", +"cb253b802541b10c21fc48ddaba1c38c0fcde10d607171ca1b3f21893c0a73b6", +"df13e2ae15dd7e9c3d34973724990022c2a8711ddf0dc6654685f91ee5fec87b", +"4295326514606b504fa78a0ea8c057c24c714ea57525e87a121d54b49ecc6d26", +"15d49af2d10c487c9d517f68b5bdb0ead1d692c1839e7292c47d9c6d61893dcf", +"aa55d2da04ca33101ce4cede60d9bfd5a346e0f561f146fa0150056e69118fd9", +"899a60eb9fe3eebedc12dda882d03b61dfa8baaae45902433f3eeefb69a5c708", +"3de020a13bdb46b349eb1913f00268d85c3283eba8c7989be1d73286d1290611", +"ab783bbf9f00e2bd402a52dadb00a06710fffcb8740d4a822c52070292026cfc", +"6b3c735b23898912c4df1fc5d4bdfb50f08519205bab10fea62fdabde57750fd", +"f78c827099a5ebc612c5961ea844a23193229463ee792212a836585e28872856", +"b935f6a4cc18eeedefb78d3498ec8b4dc6bb61d038b9df282686d09f8aac7d53", +"6df6c2d7ce660939b23a921f29a4f7077ff9fb998bb62c1e5e66ab125ef375e3", +"80a119c5c30eeec46ce473e27e1a24f0541e80bfef5b38ae7c2d0b3526f11abc", +"832386f97e25dabb6d6bf0c8020c6572407d824b11f9d089037491515b50d46c", +"846123cb02f04ff614664339caac3d2e0a0b1f0bd090983cfdf60e302f005af1", +"81fced3ba74758ceb872813e94ce557ffe9cb2bc61fe4d18b16c8f40f68d4e70", +"793734e5e7974cec5ea1753af5a9471b67edf56bfd97748c28ace6392238d0e1", +"82165279ff017d9ab50e7aba8aa5b6e7c40dce4c88d69dcaa7cecf50ab313606", +"7a8e0f41d6e6dda7033f8a8e456df113d2eba6bc790742cbf67009eecb0eb580", +"d62de862d1945bc26b1ab03a52f824935ff56f1cbf42a576cd30a1bfc56e1dfe", +"3114959a1e65e7740c3f3df693b81912601eb82926c0176020bc659edf9b7760", +"1d9e3caa9bcda611782372d0186a69bc20759f65a10e6566cb79a375bd5343de", +"178e2d42ee98128abdb6e6b9a6590167076dfd05d333aa354e612f06170664d5", +"bf9c272e608b3d9967c7aca7407286ccb187f774e36521fa684b9ce66d43bf60", +"65e2138c619783e9887eb8693da7c354792785fa7cb7f80a30c7a427eb01ab2e", +"5cd18385589d5ade763ad4f72b942d08028294b5731dc27bb88c983f93e8c652", +"c570b306209c8311893b1d3de24c41637335a5bea007c5b2c5a7d072c82637e3", +"a394a3dc3d212253a75c8d1e5ba23ac91a5aa5597c979208da5f046019cfe93a", +"29aabdcec34d19c0ad54a62c32dea3df1d2c30cce5161abfdf77ad04333414fb", +"72cc00ce506867a2515a279fc5dbfd97dd3654261ba7723d36997e5637cf8d8f", +"05d6ba704be1c15b03f503c341076c384605bc14eedaaa7c65c949aeaf9766d3", +"e809ced51d8913022e7d949ab42dced250e121e309b118320dc3d2b46b210c43", +"a1e6f8a6da1a9a63eeace306f5f25fb32dac58301afc7908dc98236618396579", +"e420bf6566ddbe45a6a56bb68ff442bd0a5fa966789883d049c9f6ef57a38d44", +"9b052e4d7df178aa3a5ecc063f6c0bf1fb3214a745d0be4cd9854309ff84ad79", +"83899a6e4500677dfbf1c727fcc538bb2760c326e417e82e7b55c03065dec520", +"8b9bfd1407a6648fb1e213157458e1444bbd0d9663515d543409ada3cf276619", +"f6e99477cc73a2ead882c02771a456ea6e96f72f5ece5ea165cd43b6eae7985c", +"1f5743b37436f40b8bb9d9620549f468e45fd1dc481c60647f3f2f01055da0a4", +"e20293078d5f784ac7d24532097b9d39b5ed1ceb0ffd72f0b5bcd44136bd981c", +"23c9584bc4430d83f740b5f991f8bc91e46b7e334aa1885b57a898431fc70dde", +"dc517082db819cb62317c704efd749d6c31e3c98271d12d3ed7fc363027170f4", +"fdb202022b7703200883df3686c970735d5431366bb080f897b6ee064f57d62e", +"66a284eb6a431b4d9686191c41ebbddd452b97ed6dc58e3dbf726d748f10efc1", +"21bcd9dfc8feef98ca2aa9306fbdfd8bed0f8c2da3d6b04dce3463af1eb676be", +"11dad124ab5bdf74bbbf653bdf0e1286ff715dc0fda04d4e4078b1693593ea62", +"feb3a69ce75b51da3a120fbb66e6255e9cd97520defabb79c1b812b3374a9fae", +"5ece3889c1753afa6f8949ba0d2959d5c07c2c0d50e7d30c91776739093ff94a", +"83e21a4166e71ac1eca43d29f4aa850416dc8462dfa808c7ce1e3affd5a46cbc", +"1ab4803a92eca1586329084df13a0e7876a8e144afd76951ae209222ea41c2ca", +"ac2507c9cc2b5f3201f09b1f757fce7fea029311b620ba116adb7f7527b2dbb3", +"6cfb3cfa41d3ed8aef017355f256baa5f1d25b29d4fa159ad1e7989c07248a62", +"1569eb5661775381952b58d67b6aa7e4ce17c322849638b12fdf1b66af8a9d5a", +"696fd560f01fd40d6e9f41f3662ded09af9967c6f84c228fb5ab4677d1d0d37e", +"a6177179254b203b5b3d8097a222c673ab8e885df8e7050e6dbc30ef5f5bbf17", +"cfe32f794a75d1ac4f1e68c1e01892c56b76a3a630fc1ebea2835f4c93102b86", +"0c1493482f3199c044f0a9f8ebb83a262544df349809d3aef5c0f19254235fc7", +"712afc0cf65e12be4da48dcfabd800d34d13704b5dad99256ba184dddb70583e", +"e88149bef1cf7f539db9266c2329f1469f128823e45fbc33f2ad7210d3a7203b", +"485a1e59e27585d965346c2a1e6cb67f31f6d882a64d5723cb920efc78bab1f1", +"a044b5bba2c396c527007c23ddd46980025378ef872065d051f7acfec1966177", +"26532f71cc42053fdd39ca16ae2e3aaed9689dd41613902377dd2ac403a793f2", +"03f3b3550d100d8fb265d7c553c16b3699aabb5628bf0cfacb2608847304bc6f", +"ce9bd96b1535e791f3e2d73d949e46b8e08364832cf075643ce07232beb939eb", +"65d9617ec0b5b5bc4471410a7afe2b38bdcea5464fb611ef671389f7d7a0622b", +"2578d2daa8e07914e47f852b4873804db065e98f57d64a2bdc4774cf4ecaddbd", +"bf5cd21dcc7308ae1e508ac9edc52491017e30c8abcff7a266f4684b18555dfa", +"2b134c99cd4e74b86aca4fcf3af32940f4060a106d9b750634d58a6643d70d8f", +"4ec3b5e65ce6179b75c7b3f84d3cd899a3a092ea36f1255340228a61ef2c88c5", +"175474b6b1bbc93c4f85bfd7a30040fe2557b2c30ac7c99abbbd9d416defc5bf", +"04b98ccd521cd715ddef40d5d75d46a3c7ce6416be6aa61b8e51fd6374dd9b7b", +"4f4057f3593f2c693402d0dbda82b71f4b6db7658dcd2e9b8dc16b3f71598fad", +"ec456aadeb12bb8d7d55f40536c4bc4ee435cf09c763a31dda594c964885f7f9", +"3d29c741dc75924bd5cce6b3854fde27419fb6b3cfb675e2b18d08b0013508a0", +"53d950890c4244e8dbd82336afe012eefde038428156d193d504e2630a5f61a9", +"07426656b6e9e984f955e4c8bcabb541da5b4c0e6c98c7a644f39d5fa12576d3", +"720d7a60dc73b27fa50203fb5a8bf9c2439499439327da2ba21b198611e50cda", +"08ef1e78a4095367e127c7636e39b18f5902d90b4ab3eedfd32ed2d1093726ba", +"9677a00f7ec11f94da6e891bcda249b7cdfa9fbaeaef2122ada7d54bc695991c", +"6426a9f053bb1c2305f3af532a587297bee5d7a985462664853928f0b4c9806a", +"f4bb02bca116967c6bf3f16ed0b366cc36c1159a75eea2f5b81c19e411f59904", +"290d50f8df1565f71fdcd67f62041e27294ca38a4d9342cc160c20b430fe72be", +"83c58937d81962c11ef39d84386438a80ec20b6f40eda53486f56c9c9a451d35", +"f0c4d4594c70cfe64e2a0f3d8ccd8140799ae1405bcd9e9319943021f132ce4b", +"bee8ad0d0f3b3c739137f8d5d819bc30696fcf4afe0b7721547f7abe44b1fcf6", +"bdd2ae2a5c3a48fb8bfc49abd5825e3779521a8ae5640cbad0e021ce8db9d09f", +"ca128a4a25314fca484d7da59b72bc33c28f746c16f7b1c61bbb5462b7184048", +"aaaa34fbbce3383dcf4ea32475a0e0a32774ffc724e840190db16bf190c34be1", +"29364198c66e564a01e3f0a4a9fa5fd63ed5a483fddca42ca07b7686af4f00c7", +"517c2f3b6804bdb7cac012e8e2845003fca081c285ca1d5b68e9300cb5383999", +"e823d854e6e7e4d7484d245f75936d07b021e58728a347b8f7294f6010083890", +"234cfebabdf4d86761197ba3fefe7c9c398a9769db9b4f3fe494c9c40bb52aa8", +"7aea63730475f78c5fd7ac7024422bceedc91000eba7f19e1c65aaca1b6e2254", +"58a5e69fa928a295fdafdc33012e15d495a2e59899d9d203a79e226959a79e04", +"5cd83673e485e607009d180f6870699cfcdd2903cc4e843041b1154f15a82629", +"15d1a6b53368bab3265bb436990240ad9ca9638aafae6b565fb3ebbfb3937fd2", +"9cee2d55ea102faecae5a88a54124cdc8ef445bf1ca81c5dfd3be73653269952", +"3f59aaa9a6143e337ba0b40a7bd4c61e20e3b6d7e951c16dbb31312d51b604f3", +"349cb04b0334d7e6356f685332151465992a23216a7d9a1723590889d379ea47", +"7aed9dfc3e59496f50b20db4e2712c792af9a9ca381498d548e25e759fe82fcf", +"3013ad46a372bec10d3fea3b36ae5e50d717022e742a0fb0cafcae5f09384d03", +"4cd600a052071b7332fe6e1128ae57cc26790380e0d608e67e55e0efc0ef0857", +"6dff74f3fadfc512a89b2af803d782eca182bacf855f81437ea7288a0d2ed57e", +"3ab437c68d324d95c9c22337dc54e44c35f7076ce3bce2b7878cd339a60aa94f", +"e5c0a41f3cd0a84ffb9453018fa0fb4f97b730bb6c59ff330739f171810e75eb", +"bd287d8d771a29de860c1b74fa7fa9bd16cc4a2b410e65cb64a903eb3fee048f", +"6a5c4d0ab45a1aca80589f36a7238af0b0e2719f384bdbd75779b88b747ad04a", +"227f7c47d230f868317776ef6ca6b1eddd8315d61004824258e750d0513d5f6c", +"a3adbead210a6d2fa560f441354f09c71f695e792604b59127ad3af53b2a12bf", +"fd666abae3f3cd985bade5b089391c6547d6604eff3bfc779c03cea5dfb7f688", +"41697ee8c0c57ad63c51c1aae8ef5cc81ddaab3708d95fc45b872ad35c26edfa", +"198fe38e4de28af59069a6d6df1543487694e5c7075c7bcab9534394e6d44160", +"893662124d62666f32c6bb9fbfd87d041f0524cf21da856d72868c11d9d657d1", +"c781f8cd7dcfd3b66f24b2c21531a19e9fd602647ca60fe57bc0eb6715051305", +"937a145f0304716cffa2bba32ffa6cb5bedd36de1c6c03a50a21cdef5ee9e853", +"5e18ae53b3f462710671bbda695561fac63793ba0d9d85ca1447550747d7d0c1", +"abc3ce9b610e4c2bdf1c41706366e6c84eac7dab6657199f85315173dbf64e89", +"601b9ddeee1f4abc1d0544b82cb4a19b1bf75111b1ef4b9455b90e0f970b148f", +"0c1899b4e5cd4ab52aa5c88aa06cfc9e2cd252e7708e41f47f1f1953d6a4d977", +"472b247652475d72a024277d3ce24239f0807fb2bdeba74ee6a4892b56ff4a1d", +"b11856f44f96451067583d9429f2f675eb2439c8073722b315232e1d1c3d47a5", +"b5de438a2bd5f16f8f7e5df0d64b4004f5cc96060ce6a6304c607b07c8964642", +"c1c0e93e54663673f53a4e774d8d6b6d79199d1300731ee2250fe6cdac5cb368", +"7dd6538900cfa180554f65c39b920fee6ce6b09ef27351ef30d0855833acf48d", +"905beea766571e433d5b41512ba86c11c5e504a430be567842ea1e974c12f6e5", +"bb53327ec03747acd863c89f4ec22131718a1591fc5af9f9030a0bda8ff2e033", +"897967a478729fd11f19fe5d8301dfb66eb32bb8687c13afd7c13a78f23a5531", +"ff0539ac4defb4eb74e8da337dad1d7162668b08b01862e64d16ba6b79e4142b", +"c4c8c4b4e2c79189654ff9f7d88393e5814c09f19140f04e55b1f9dfb8148d59", +"e1b77c220a5d6ac006f83ad892e476a27b3bbf4b6f32612d6d20261f3e11d4f4", +"87ee281e2cf217a72c31ef2f075bb0cc7cf483af12f4face8a9b663ef0747aa8", +"5bf51ea0e46a91a5c3efa666ae144bd7353aa755942459a48769574d3a2c83f4", +"99ce519cfbe39d9910bac61c05b13d3b42f8918905df146cc2a6700d528a5b4f", +"faba3f4ba4e7416f035bb807e50873cd8c510d2d001ca449d5a7586e8acf36b2", +"6b1a7204b857bb6070bb8f65d70494af8b5f99275205a876011b9ec202d57cae", +"289a6c36994466d9800971f5797735a9ad4355404f255bfe479309226cefe2d5", +"abe02d0f1a34494af55d5b6fa2f159deb7d69041e3cf96138c99531d7321a8be", +"f03418fff3b22bde2f6c64af1c0734ca3eebc2d1b2ce0743be2a8cea9a0f1849", +"1deeb28dd349ffa2560a82ea3d682be183af204e8683bcaca175414896d9bcea", +"9e9dbeef98833556ca38c34d5d29f94d160e893a0bdf8a379369b239ec918933", +"14e3dba3758c1ce86a93f7b93a15f5920f3a88e34776609857ca3c1068ca24ae", +"aafa88e7f286c104e41f086a41692e56d4ee84bfd088c3a943249daa3b2dc1dd", +"4ef1b6ef735dcb41721edcebe5645d4c926af523489692b7b261310a677ea04c", +"b6473bc360635211cda8177545c9a80a05da57bdef1e43cbcf1bb10ad463231c", +"ceb716d7192820552381885584f494c4665a70985a5ae6d96a316b37e60bfe33", +"6cc6fe5c12bcab2c150560fb5e1f0d1f2c8fc80dfacb213231680f101762a84f", +"f5b639eb8813deed223214a7b8bd8184f020e50e6c83e74fc6f1eb8e6eb59745", +"319c346f1375204e421bff10a2de0d37adee62634b2fa6dc046714f14c08260f", +"db956d6f8d75e426cc1448cacdd556af8d1e1a490e5abdf326dcacdbc3c47755", +"b35174260312d791235b829dc13e9ced9f5255a9a8b4748e68d8d46340a11a9f", +"2cc32cc26d079899052956e200f521a5a9e911f68717fc44fc21012bee6ba976", +"6e80c831fb835d9e7de258a3418de08ac8006bfe1b4c8c8c6eb8a4a3220659c9", +"26e1e005b59d0999efd04218ff177abfec30a01eb6af567ba7f4fd8ed13b819c", +"b54f35f180e7b49c77e1f620460a2f69742af2113cba7536dbb4bd9e72b076a6", +"0bfde97052ae7f8bf710d2d93f6e2f7c4312014d0b159c6f9afc7f65cf34c664", +"3f7ab26ffa64ec0734da75276bac98d6eef8218c3c21766a21af0515e1ddf7d1", +"dc5ab4ea3cf0dc8b9d4f0e5a9d72eea4c94e4296162141cd74d1e4f38673624a", +"b0f09ad016f16bb6c189339795f62a150aa329e1cffb50314264264d9e9c6e4f", +"4cdb46ec69c6cb00cf418bb27b4ab0922f4cf707f00898edd8817dac625a0669", +"6c5ccb215cada17f9c38a5b55b7bdbaf2b2cecafb51b788e90bdb7acd2427483", +"75fbc30212bc16de3550a681fdaf75dc602840746a5dd76ceec9e3804907bd01", +"98b3e5f730dbb19c6d3f0370df3486127a14dee10a4e4e37453176b8d1f6569d", +"434c39f79095462aa2d21850625189af5902fedb9ec65beae14fa89cde1a959a", +"f1dd35ab80bdec905888defc11730bdf87e8da0a244fc004f700a47d1f683f39", +"117aaf616ad1547be97c088bff1fe0cf701a76b5576ab1a5064dac4bddf0ea24", +"c61011693530095d135921adfc03ea8bf166a4861a4a3b69f02d9b4bade06f0f", +"430ab6a8fb6781286c80dd543519aa3a3991aca37564ab3c7fb8260823dbcd0c", +"4ac190b404b4d841049551c0f10e5f3a00cd5a9b10c4b2f2ca9808dfa50bbe7b", +"104a014e4049e1eff26ce38f890f2b8b0f49ebe249d28ebbde3a96da3d2fcb71", +"e59470cf6ca60d8205310cbd27d6bc278d15957985b753bfa89d78ddad2cf5fe", +"16aaeb7ec76c76b97cf688d1b2294d800019c50f0a2ccc17f23c4159c8f70aef", +"546306d2bc307645534e58e2364dcd5bf342084ce38366b961ecaba61c5cb02b", +"f52f6fdfc0b7cc8c49e9ba70889a7c45f64cc9ed1fe3b978f6cb6a7d72c3c3c5", +"93e9bad50782458a1581abd5b4e3a70be5d23ab7927dfb5a5104e40a4aef703e", +"5ebbd1965e6100d31529e1511a93f04fc72f02e85f5fcf104245ac4dffd9ce38", +"08c78f50113ca5a4629203104541dfc746a0f4011e5dcafce88348c945374425", +"a0b90885689dd8aeea0979ea1ecfc36d8c3a4bc43894e40d3992ccdebf9216f3", +"2724a1b44749938f9af004e45680a2466833190ccf0c28e3eaf2611f08d4277e", +"3194b8eb6b2107062564ca40873f311d68119f75edd6b933e77c1f90bca84ee9", +"15ac79322fa796009cd5f393d5bab90805854d9fb3276c67fcc9449dd454a277", +"6c6e704aad88ad3eeb78332fafe378d8feb7174549eb23e2ec2232406e71b447", +"8230a1fd29e3dc4ae4891e3be52206c7a55da54b44ad8ae6f6234b45ce37f753", +"47a157bc3749e388fd638575c3d7fc0346a11de33f5503a1398fcfe9a9326fac", +"414ecc95ed3912528755c64943b7e6961b6024f22ab322d25eec64687bff8c4f", +"1ba62612eff3453fba41013c31bd4376ca3c34d021b4bd2939dd432dde6251b1", +"af264d1b737eb930b641a5a92e568c22a30efb18d2055ba9b67f026873edc480", +"bab7e3d687e5410bdf48892c4766ecc02e093363608c7a2eddfa02f45cb863f5", +"7e981ee9ad6af81b6b893151cf26abc3c94d1720a01183b9254c6b592dcefb36", +"dfc4c728a6708d3e19ba894b79271ba453b7eb48fdd21d8f33ee224970b7e647", +"93eeb04421798c8af37e8315e4b610eee2161884facacdc79fdd0582c1dea0f0", +"07d94cdc2da4f04cabe5eda0197b7371599dc4fec2be2033f89ffd4ce38ec8c6", +"4a582ac99ecfffe714896c25d87bf5b46b492bbeef45806d4907d21189eb3660", +"e2d41b1417544f35d823bdaceac0674dae5111d9fc45f13cf495cbdd2e8c0d13", +"b432754b67b15f67efaab14c6297822427bde405bd9337e3aa1e9d028f42e47c", +"d92ce68d69df3e594c312c016fc4ba75001e137cb99304da1ab15d3ae09d8dc0", +"a66a23b610e7e51079f6d9f522ec9f6df279e99bcdc922a55ca043828edb9b0a", +"bf925bbd3b48ecff6a2c837288fd1b9ce5849426c76dce143078a2ab249d1995", +"445bac364df75a38b2d7f1eca270603c6bb6fba3a860a0749bf1fbd6a68f9245", +"c0576a5b7ab4de2b02a0745dae9c78842d11d644ebc1bfbf7b34528b063209e5", +"b1eca3f6ac95aacc306520efdee1b71902d7d2d2d16f6e999570d2415d63cd82", +"a7da5c9f86c0e0580e691c0e8a754e25180ddea243c4253f90e6d2dc28d39fe9", +"207dd5b925fd3bf40b54a38aafeb6ff57694c739e0a6186256c171eeeb680f70", +"a6145f6b2badaeea9b7ffd053131ca0a8375856abdfdce0d4fb29f357d8b78ce", +"fd36ee59e1ce3b900bba48f18c9f0f0c480c12c4bfc1da1eec771c60d54210bd", +"bc52886407409c20cda9615d316e7bfc1d055bad292fc1b044a9045667651d90", +"97aaa42dc27ee11d9928d800f9cb5e94fe1238d98fb2fa4d3c3312cfa79ececd", +"67f75a12b2c2b1cb3f667d22f93253005f739b82e08508c4b7d9f25ea4eeea8b", +"e061b6323b310936e8406df50677edffc087f3252504ade98e299bc42d55db8d", +"c0102f56e43338f66dadda74c977445e9ecd7354950973280bdd8f7c41431fc1", +"cf7187f109d9a7508d791ea946555b9bc14b235908642a522ebdb1da88301646", +"2f6f5c0e62d7d588185b138d243e1f9a4b527cd962b2f1f1bfc9c52b5ae0fe10", +"5515285ee161a6d0e92b746fdb4a0d3d8727045e79e36fa76f20e0ffc132d987", +"3559ae081d4927a809fc10cc833617dcdff12222a06f8fcf3e280a0769a6405d", +"d856eab2c8d9e123b115fca70d78facc68c789e9f5acbaff251459b967f051d9", +"b93ccefc04a653af10abe9bec36c0b466ee6207beaf6fc0ea6cfdffcdd87c8aa", +"46214fdef4207c98b4c38a99f64ecc5f1a080643c01e264276cfd5b16876a43b", +"f91b6e0d6c8de00d3375bace650f7254a31ee55d90aa68a4615d41e2b0b84933", +"faeecb2339b5b4c93c207953d6e6e9bf519c78e0b618b9cb08474645dbad4b44", +"343e5d6656fd67b87977efa9619e5ccd8a3624a9bf6fb1d7b211ab3f53265a0f", +"9583efd358b4181a3edc971167a097d72b61f7656ccb810623f23b6b88bbbadb", +"5a92c524070ddea51b1f4d4dd01b2570b2cfbf80746fdfbedbc91da79d697f2b", +"6b575ca6c2ea30a60e647720efa8ba7e01498514c8785cf2894f363d3cd40fbc", +"e6278fae46b87933a015183d6a9f7375e79c3c9d53f507edf53be779bdc0cdd2", +"5645311f22f1ac98df3acafdde5881f52236e8b07bd2c55975e76b588e5ad807", +"8a9f5238a30730e1f48702a4b796cac78c54ab0cf9ed0b60fd0fe46fe84899e1", +"bd90c7210bb44b61dab5bfd0241f5b52eccac77577a94f9871f500f3c415605a", +"2d3c5f724205b28e40ba20266c431e6a899041ceea1ef6532b4a25b2a8e241a6", +"89edfd3453a10f6ce442b656437923479ca3b35b820f4ecac1c1b6e82cc93b5a", +"9bdb7ee0aed097a8b93e25b242c331a9d2a2e6e9d6c906601cdd93829dbb73ad", +"b2bf524e651095a0c04adef422b2d979a079dc64a45b9e538679e2fb06d516a0", +"debf06297e7556e7bfec6250b4487d29c7f61e60499a72551b5817c836c5c025", +"0d48fd231ff39aa32ce44e5b27f306f04d5100c844ccd43911159ba3caa7512b", +"15955d365a90bb37480b64b5229f97d71d6a6a040e782ac55f05f7e5ad280ad7", +"75894976e962181a200c54977d41d71c5975c10de57834b38accbeff3be77663", +"211a931398dcd6dbc471ac35aa741b70563110c01cf56dd68bba9fc79d4cece3", +"bfec2adf933664b3c5a8e0cc35ce60516063bc76062a70137e6c8c22486e27ff", +"824c9b33d01799162eb0f5238337ccb6ef32a4a6999182c37316f4f29dfe5890", +"7e613d37f7b6279739020b93ed9bbca3b5e8846d7338ec21fbdf301448b54b18", +"301499c7699a782be3de7e139276cb43330d5a7a81e0346607cbdb171c828c80", +"6f4da6f46fe3eacd86f19b981cdab577dc969cc9cd695c3cb9be7539d7604133", +"797306d7b19b01ff951f5be13c88b2a25ac91fdc4e8f5b41453732edfa6b841f", +"30d2f111fa0c7156fcd876a1f35b973dfcb430d4a6880ab2d36e022d036681eb", +"1d15c5d0d8cb8c943e1f5938731b0639177a7735080ed3127896508b39343446", +"4bbe7a8d90616c41ee1b6a228a63988058787a283a787b6b8bd7b831e668c9ed", +"47eca767ef1806a13b3160dd02a4eab001b21b51b32bd7dc9b49384f168f9354", +"edb1f2ee720cf5e8ad49c8e980f82aa291e60ce2f0796a2f9362baad3dc05ab7", +"96047183f58322c63eef4f9568a9dd301e73ce14b879f2470d28e83298db2dda", +"e507c3d2a935b33265e3c93b1801b3189c2d6105fc81993815f137743616acf4", +"0990b528af48999f36843e379187dcc6a9b1dd19bb0cf93d8c739bc9aef003e6", +"26eee3889d28f7c059fcc72286d576c4a578ecddba3215e09b793728d0809e16", +"e4d5068088736e7997c79ac0e6e7db42193f6a5cb890bf09ee770f87413a4106", +"65b8d22960e35b4a7282d4d36fb546c16da9504447d91edd2f2d198b3a00a6c9", +"ad6536d46446803ff45ebfb20101d3a0b92f870e13678c403a70d630ebacc963", +"a124ed903969015f188a871a228ff33fea851f47d9438103548b75e29476de4d", +"168c200d11549d5dc869e4d6ae1776df9b0c826d7a8c91fa398a46c5c4751230", +"30a42d5139de6fba2ae666e9a20c65db45acc68326e67bae3e1271d3184818eb", +"c8dd475b5dcd5af984b2230be7503a49f7be4a8fda78de538be137db8d9d14bb", +"e6dadf2527a69744447b12927ca0a3161e819398a7bfa5f08bf3ed20c0b6d014", +"4b9a0a7685f4f86ab6dd1a837f082ee7522ec95752908a47245b19bfa040a60e", +"f09360d3782e0deb53ebcaa8532fa1db2de664c7e2476176fc40bce84481a8c6", +"bd22a4f9daff3309dc6d29574767ed3bb9c32d6b9781758a7de3a82e2bf88f80", +"fda214eb3db44178f59ce142b6d28aaffb02e7dd474c816acac16e398458a70f", +"70ea22e2818b39b81ded78b5a89750ff1f7bbe58b12ed762745d45e6b51f684c", +"0d3ad95c9fbad1d968e62976bafe29bf372a13c82e532bdc9285de7c57f92824", +"905c1f304ace286b3eba316810cd3e54ed43aca9162d8ab1749f15fd6305edcc", +"c41c2a03b284543a54b570bf9569b6900306586a300dcb937924c6c967603798", +"9819bf168db06e1a166be928b0fc4c567a6b683558731764a524d25bcb38c1ea", +"38bd56f192cf929341e20be9bed44c48eca5f8aaeaafea93364bc09a437924d3", +"32103cd2adb186b1334ec133321a219006fe06b72ecf0668cbcc93dbbe8b1dd9", +"b629fc203878de20b184a6036045ec7872cda4859638c7410e0f758769fa1d61", +"097df028f48a7b2f897296385a4c4c84c81c3a33903d4b65058ddf777af805fb", +"a09c50fde8178aca4bf3065baf8bfe56e7c8b9eebc12458ab6fd3b17e00c56eb", +"e2864c178680e26ff4467cf616436942b070fca737d344a6b2adf017319582e4", +"5d00dd5202fdc7917ffe5a96f0e73b457f01356c051703b25b7f772d4a127754", +"01ad7dc7dc1ca50cd21809e59e527dc03c06d77585d99d8f71d0e270859c268d", +"ef2593ae213d1310e52083d13e59b08fbf57536e512d2bfe378eb4c5695b982d", +"e7213208762ad15b3d801bb0027c01fcbcb0ae0cd083d2b364f44cbb64b7bc0e", +"b9ea7324bf46de83ff81f0876dd8e6e8485318165cf9c5073549dd8824c7c2d3", +"7e67c6d922bbdbd0b7c9f3ca720dfb0ecf19d77e649c14824159d855ae13de59", +"8e9f4091738d5be34fa6d24088859136ab8dd860fee9f329e40d86f2470ee608", +"c42a6175ee0cf75e5a3d5e9c6d5a3570f184cd6e09b181460883da288d690a79", +"d7ec092d39b738607f1ccc29160e33885b004eab002ce518930ca4f9bc2dc74e", +"0282c1df92347a6397870d0997f78e7cb6045ce4d692fc7fb73ccdde1ee2a76f", +"90b928d99accbd618929b56ef904478608ea4edfbe39d6d2b66855dc5d52277d", +"b0d76684b2bd7f8d1e901e8b4e451f90e2e57bd56b129db14fe6121938176381", +"79da1786f50eb8fc41670def568ba1f946e25a4629a1d5717954681fe914d0f6", +"63a54c3753afe149f27df7fab1d48d647fdbb687014902c207881bf01c008e11", +"df9ac28fd4ffceb40305617dacb13a58e45a20d4cce72ae242873efcf4fb3922", +"b795e0774b135700f245f165806180602165188ee68b15054892c60c5c31f555", +"c3417d0f4ab3a8b46641111a842b9eef2fef3df9ce6eb126eb1285502a7a8487", +"7ccd9daccf569a6016173298dfcf6e74a16b169b68599a853f19e516139daafd", +"bc265c40a3fdf4db0795ac17646e9119f262ba5ac45d2ef0b1fa3d723070606a", +"baa024b1df51e31141c2906e082b5a1bb04e1f4353ce4ffdaf7d106bca0c0dfc", +"d864bb808bace8cedc58f936ce4c15e7959cb2bff5d56fafab420eac83401ea8", +"01621278091be7d31c85b5bb3856a89644f52a6df9d7c2b258c2b71fc718cba1", +"cb1094b3c10ecf5e39ffed13e6cb3207ccfa19306ab09205fdb2902cbf534ca3", +"a8956e278741124dafe3bbc80935050f96dde750a99fac0df2f3f6797b704f78", +"682bae293ba9e865e42186455b07c1f9000480cf4f83e9a0639c4327bd557446", +"05a7ec394fd6145ab02fc44137462df9accd0ea88526914288d3ceeea5b710f5", +"1088dd5677b8658a313c3b4f9dec552cc1e08ba36260de456c4962fb201e16fc", +"27cc3035e74126bd5762c295a2cb384bf572d296e6936d3941e58e70abfae223", +"6f46cda8672a7d660e2f8cec69ea0c344a2e80e9b1dc1520753dd5b248f5e561", +"062e970a200653a49b73447a30badc489dd0d54996273222a8ac611089ee1553", +"9742a3b2264a7debe84eb60e2ea796b3a19518e7f8d2a5f57389aeb9ff12a855", +"93bba324688af65b727f0c9c08f0fb45e2a2389b269939b1414f4818b071c598", +"97bb4acc6c2e9f88b2043bbe547f8178def40a4d931122f9f4ddaee1e45f47b3", +"1050e3a2d559a9ed57292918255d3e8e5bde36210e524f5fb6bb627246e59c82", +"3b363c755d5101f861b8e980b36e5c70a6b4f9329b4132df65e3b72b37ad128f", +"33208adce88477a8d40953b322d9079be118dabb96e6bd913b36da29657e8eeb", +"1469a868f957169e33d763d509c6f147af3973927f940f1db81fc54e48a43c56", +"d5db2b239dec570de1be8c140aa7d143139203f37030623b4d73df9f0e8e113c", +"627320e8ddd3f373041722d98af6732ff47cbe0ba9e65c0994128f4f2885d45a", +"6324bd7b70a0b6b552c17aa397297d5f4ea3cd2765aa476eab903ff02f53ee1f", +"65922ca1f6ba733cb240e6113e5896d164119b631e6fa9ff0f9759d55043b234", +"39cf67812ec7a1101581c280442e72e0d2fffc9d0192e45e24938f8941024d9d", +"722fcd3407ad005077f53c3a2e5817050839f92bdc6d6a937c6c15643a502bde", +"491991bde2e974f4894470188e0743448be51f87a0b4566010c859ed57807744", +"7a22eb14427e9ea2e1bd93f49084f9895781f066c387de1495feb2844a4287cc", +"c58e7945c47e03c5c856e2c33c2dbc41e748e9b8fbba53cf187caf375d5e4dfe", +"69745335c4ba3b4c8b52f2dd68f4a1297cbd787e0e76f1993aca6e5003384a56", +"22f2e25cf8776a97a16851277bce97278dfbccad6c28df7cabc3cf8c89bcc3c6", +"cb4d13f1974f1d29b917136fc496acb3902f632dd88f8e53f1f5d3e18d220e56", +"e8e33d94345512e41a0ea57dc70e66d50afe7740f859e7b0814eaa5d8b050145", +"ff57e13ab35ad743d9c3bb4b7a08e5680309fe1e345eb8088c03e102b8374ecf", +"0e8dfbe308483b9b6397f3e773313be0dfd1533a2fa613bc09af5aaf872fa816", +"cddf84e3199014fbad13a3c1c06c3901934ada1afd5c1640bea40d5185e24bc8", +"b09e35107e2ba0e49bec515675b144d8870aa7f3e304384ef1356d0aadba40d0", +"6863d02cfa5ec2a3ea945f2c5d6dc3bf99e96e9f75ad5e2766409487214d49a9", +"583093930eaeaf2119529d11da64a39099ba6d031fc7ad49ffd7418b10a76137", +"1534f102dcaceeb0f4f988b5c43dcece858c159acec8234b7477ac696fa135b0", +"512a462521cf2221911e06dd58fe975ff1511460cf4d14d441e9a40b4cc203fe", +"762b6ee85b45ae0656e5c793a74fb582b1c9846a561045dcf472cf679f5f0422", +"5f3491ff9569af2f43ab0459b30be6ea7849400527841350ded83f9d1240c7cc", +"caff29e3a569914c4a73374a55922380f01c138fc78b3fdbc74d85939dd2675d", +"3a310f7008ebf59cf3f44eabb7cfbae1e71aa2bd2b9b0f2ecd956ece2022ef00", +"1703d4da8fdcd31927d898c9f4e32a323f1388f4b12efd043b1f1f56da28c850", +"9e7d2044fee9017dedec80a1c29b5c222132fb2e7c90cf783c1a50507983dfbf", +"95ccbece717ba641c01be95dc3fd6bea33e4d25a61980a03d1da84d9031fc90e", +"29206b40de3c7f641971509f8581c80d426b450e80792deb110490d3c047a405", +"382be757fa40774ff39680eedb1b520833bfbcc779896eae4e2e5bf892451e03", +"32a76a70b9fd923d295e5d4a9449d53f1477e2abc4fb3e522d9ce478df66823c", +"041793f35ee68f0f4e3c381bed41631b9941cf6d1cc01edd5b3e211d0a3104b8", +"baf642388f2d53c13cf59175d81ecb7a165bf15feff4ca17868bec727dd04fb6", +"f1159e6bf0904cc092d0ddf098dce395ef760f3ccd8a41c475d8ec42080fdbc8", +"61c0099b6d5f8911122dba3603ca631cf9358c991cf3eae1f6e494b01aa3a85a", +"9dc6882e862362bdeeef60fe5dbfd20ec25de0ee273ca77739697273751680fd", +"1106850d35ef8b6cddf6d95759462eb0ac8442d2b3cf104eebb1956773978005", +"8856c8742cb8f53f03bba5cfdbf6a0a871a6d5e5608d73a47aad69ec45ba1631", +"edbba55a9aac364dcfd6c8ef59b7e908b4c6d28c9a3d0ce7b65d2dad9edb9f3b", +"f32b5e08ed44312dd4dc87eebc7b57f1fd43cd74aeddee5424fda8caa4a1b072", +"f13eb362c038ee2cd7ec96e50f97ca9a948d087b69fe9da064d2aa519889e976", +"0c95e6180f3d708396e8ae45246fe1e6dc0a3dc12a63160ef2f8e526b759e88e", +"f312f724a6ff2263675e5e81b724e64f29f789192dd46ba0a1c7a97cc1e691d8", +"742b962caaa94390863397a28200251a5a2cba4c6baeb029c9457adf3fac47c3", +"c175f2bf219fddfab264ae8edfb28944051b2c0c6f8703db56477d54de00933e", +"658f5832cb37d4ecf4a99ed412cc51cfda06a6a20f75acf5c8042f34471e42aa", +"7d1e92c898a9b7048b451cb1fdd7b9be6c3d81f9f2845a01e45214752fb41865", +"0a3d459d2340c4e33b15fe951f8e4807e7ebeb835dc7c8b9b641690a9bb7dede", +"f2ac619424dac857cccd16688d6cd2b752e43b65847cca0f8deecc7ef70e6773", +"e535a478823cab214e6bdf50984e6fdfc7f21e7e15426314bdef6d469904c856", +"02f6cecbfa506f6c5f2e2822e9ae59e2bef44d5e3da9b708c2085932322f1680", +"0cd9e6d7361a77c8a8a1efd75a42ab76c93943081f7e0409bba3f035d44b0d0f", +"348daf413764a22a22c5a9338181784ca6498a40662b273abdea31683c3fd096", +"1c8d4957ef656ccad323563daec41544fe6264a1bda714580f3a9d7ba97d21e0", +"3511de76972a27609eb0f7099277ff2b8000edd74962a1469eb0add2705665c0", +"4489e42172f2bee0a973d59f82fc594f86995028a211f605794568dd66c2c45f", +"27b75409f34609b4ad986d7c762eafde72f2bcab268891c78064f770286f716d", +"465eeb4ce45d24c2fcbe8db96af45f65c062efb712eaae31ffb7c8eb5ac968f6", +"844f3b206a00629e64993e9d67c0e7b4bba364a8c2646b0f0cfef92e692c38df", +"cc183b54d325a7919c6d08cfaa203a259b4a691d8d01c94c97b44358f040cf9e", +"6669b64ddef2594a23dadaa795325938fc48c57a016cfe2a2422fd3e68ef35be", +"ac8c3b473609bcec27ac20edc391565ee43ecf6da585de417a55f4bdedff6c1e", +"90ac90affb81cd7d368cc6425498d4e235dcd925f9b56bb04959b2b3f4a9dfdb", +"bbe05c6d996b1eb7788dc1374e9291571e5a260c2811ee701362b5a069b5fc96", +"a2c0d000b692661825fc5675ecb1d21129e48507d85e9d77b78b6570609c13f5", +"b2c1d40662356cfc4ce5c6d3d5db286d631b4d6b7e140c25990b364b89ef331b", +"c968523bae049ccbb1107ed7ec99bd0974581602998ce7f099ea1ab7eb15b0a5", +"84a1b53d4e73c607e76165ad9a4d6d1d36729e83ed798dd1a2f7a0e70d53ae3b", +"6e4b5031cecf29745db54b4719a1a627046b28164edaf8571e722f62c649f111", +"1a73b1a7d94b3e65292bb543e9b89889f8113646152219b2d096809a87fc86bc", +"bf471eaac162f0bf317235a006d4126dc5ee52fb0d96eb1049616a88d198327b", +"0fa45a4d4c0e916541612d6ad5fa60602581d204e70746a6292961f7939a5b4a", +"9d85547186a4edcc9f06e5f7c356d972d6d156be94cf5e1bb11e0dcaea4f7e0b", +"a7885f0b559ef89848795212a6f80eb28c351cabd5f2209c93f473768d38fc80", +"7295c57f573ddc5fb21545d9b3b82cdbc973d73d54bdd4bcf974827262026b54", +"894d64b630d030d27ed1f01865fb97f2d191adf7eac737d8ff1e1bc245b9a046", +"b657832cd260d0f9d494967781b367ad8acf70a3abdd99ff490d7892692cb7ca", +"8580515e78d47699e827fe4e9f79d0066d8ed9821584be48b19c720027862ecc", +"4bffd2031c78ca73d8a41d3fac2449918d227074bdd96fcdcd53c5cca5cef8c8", +"828494c33dc8328558611acd7f69f86702c2fc1e6ff389ee2a6611611aa65219", +"5254df0af0385f267fbdda35528578788499beecf9092e10a8e74a656fc7ca70", +"3b613d5533cc0a8f092b7d94f6de89d0888e03c9ae11f4c31dfaa97766c29c7c", +"eeaa8016aca0c26808e8b409b3c2c04572e121f7fab8f02dcd6b39abc7894fa8", +"3f7cf5ca80d86836c05785d41c13f8ced0e417f985d936b91f439f2ba9490a96", +"540df4c106bb4f92491dce8739fdefc9a3c9a3659d288e7bda130f27967b57ef", +"a291e82e9c1329a5278f2ac095d9fe85bf0c7dc163bfbee1c10bc1be2ef56458", +"40e1be377760482af67cbf42373d168fd211c0b14920b8d16e10d66f532e6efe", +"d9b28950af2c4127a079a6f24fb4711325c672611329b2154f9b608d6ce14884", +"6feecd025d23b7c4b558a14d7a04f18babbec2b24fe612d9b04dd2c9b8f45446", +"188b882612eaade35690d5bf14c67a6b8d825ad88577956a22d687b501ccc280", +"c2151d35cc52cda5650f18e7b0a72987f063ca94d8b749c8fc82cc884007853f", +"e4928e1a5ab2a431e6bd2beda4f06e885e55524b9ff475de7f377663b124510b", +"c9773ff2f3dd0a53929155cb426d6837fa87c21993f9b4451233f659cb202134", +"44db51a701771031bd7e84aae5231a4b256b06a70b2a42238b9f837c7d75543b", +"d5c65929dded61c70b81339d2ceeb61eacf72f2aad7a6a708f4659d3f4544f98", +"a02d78d87954194327be783b13634e3a5353326f7b6cfba7ac6714859e934e4f", +"89859b53d727aba1309a7ef8b54e5b8b352ad6b02da034e41bf99f30957281f5", +"8ca7c70de07c7282176150bdca86ed7c501be6437e9a03e4da022354f1c32edf", +"64d1dcf47eeaf30b8878c1482b6ea5b7614168b95aef7459ce29ab2e667cf392", +"334210b39b4727e993e1fdff34f4dbd985900609ff12d9c9820c45d7630d71eb", +"73075f0ca16293e46dcc5c28b8fd764790ab98c9ba60c4f11b4c90e13cadc760", +"ecf54d6cc0a9d13e004f52b439dcdbe1ce4135ceef597d302946740f69196ad1", +"7c27bc0d5f7bb5ce209ece9d9ed6fb7ea45443fa4c505b30055e6046dd4f2d92", +"b8242941a8ade2c6da31720118a349870c2681f2cb327d2e9b7f5bf059f9436e", +"84a8ec26e786866d90c520e188e54fd7a2c92c3261731d7f03ac76d26325812c", +"3d49c1b7bb44f5ce022d8c617593a26b957cbeace2e9fa3f38c2eccc6abfb6b0", +"4b69511abe64ded37921d8f142c6aed005bb014c35773434361afb553f4add66", +"975258d2c45a91a46b28f7edb3bcdabc9ce7a5c646f0bc1e76ff66d558df8be2", +"d6a3a5a8c6fb82d1bfd28c7d4f09549552aabd0c599022985e1cf6612b80c976", +"0f77aa190ff3ab7ec9d563cfd66173fce4850f2a349c6073fd0cd9123755dfe4", +"41d088f979c18c112d2613fed5f85c9f4cd4055a23888e67059545c105256fba", +"ff02f7cf451deda817d588a6a427ee12742c9579fb79c2ae523fde28ae12d7c8", +"6d4e4e71a78872cde6ab7565db7acbf89d79df8adaab44f070fad17e31bb813f", +"4c1065f9f8f548a8ab83f2dd12ea108280bc743fe79beb28a8c530e275129cfa", +"272c6b9acdf9c14c70e06d2ccc564b9de022f8cec9730546711a5715996574c1", +"3ee5fe99b0a07bf297ba7e6ff937343ba69ea88f94a51c56238f72dabcb77b7f", +"dbca2ff00f47927d5b0a60a733f37e3064d5ae83f350f675477842d5d301bb5e", +"061b01691efc83a8ac9a6d474953f12e467ea3f8f61296ca8fd7447741c18727", +"49717c89152bf8d8a9383d4087a338ecdfc9a2578a88bce15cf01a0f2fee9bd6", +"95fc32a967fe9e5a002e20b6e4579fe8d30a63924cfcfd14455a02ed4f149e9d", +"b86099514fbfaa85015537ac44f35e7d6eea18ae264169941d226372e9fe30a5", +"1e9b7d9e8f87306a0c9eaab2b1527de6bd8fb328c25d51cba1ab55d2d5909c38", +"fd7a567bc2a68e76f39540ab8e01a9c9bf6668d3d77c5705c195429778f12546", +"9477e5d0c988e6ae558e9815ee1f08992619dcf71694e0b09faec7901697ad7d", +"a822ea37da29c27589161e7866bb2bda80e121f235f469bbfd99d15c5ff289bb", +"d88f50ef0357e84ef769a7e5dee65b43b6ed2576d99a9576091d6bece77b3ce3", +"5d7a0a6484c157beb3a9368adf53e5d84552ea765df9a1cc617bf6add4b56a65", +"6effc705a2ffd27c2d7142654daad0865ab21a83602faf10c74131c140231b06", +"ab1674aebde0ca57d52df24047cbf25600c6285db9bb31c6abcf8c0820202eb8", +"0adef9a74fc5ac5ab6c17499f377c262beb6c2c428635c2cd876d9ba5da147dc", +"8552656766a0b5d8442206500b4d4e3d35958b9501d3e19764741864b16a4853", +"97ce3a31481ae09a6419fc00419e0c5756d00cdcd4343f6dd628b4b16c12c7de", +"3f52bb5e8aa7479989bd18d242c8390a7556f479730c9dc5a82e908d362fafb5", +"77dedbbcae12e53bc1cf70bd381128216c1d8aa8b4f38865863ba940dd332c55", +"aea36dc5267a85fc7a05ca2ca533dbc344a785c99b065a104a077cd3d12a9a5d", +"093b9191e0d8c46978474717260810235f6a6446d1ff1d875c89e236695b4821", +"7a87350537842e38df99cb0589cac4e4c3ea36e8048626833f388d944f0a21a0", +"87578ff4c3a7291d57b752134e7befdcc448dcaf5d462e30c4d43bfc67e0c9e5", +"9fee53e7d1a25c5e6fd098f49cb41d336b5fe212a73b8277651a7ff868fab9d4", +"2dd773e71f63d9c03b8e1d9da398d525807d14e1a0f992488e47e686af48856d", +"2d655ed33b7c702bf578a1c9062fac0b947aaca69b96d79106bce6a3682013f2", +"7261a66e7e18481674d9a6c247a0ccc3b716581b94d8842aec8a7e90b8e343f3", +"530c85e2b4413d0d1102347326b9bccab495cf59700f0e3eaec04344bb632bf9", +"174568b0887e4da8e29ee0865f208791e813b1e5968c922bdaa6c7a49ed36e20", +"ec7b6d6d02fbbbebed36ca87f7ccabdda72cd9268bb47250eeb253a2f5873943", +"cd12ba33baeb21153b476fc10aa9100f795644e4d378f3151d4424e628d25a08", +"b736562196ac80da2ac359dc7bc862ad5a702a51ba0c1e20c462ce7d7fd583d2", +"e55f034c664e69dd74629fae18eba0daa29ccff1ead0bb59ebb26d7b4dd1728f", +"ac53bc2fdbd27e03dca1daaaefda9e512cf52f2676d1e0598e31b83d747809a8", +"8bcd754972982d42431a429967c073e722767d0e8863a1a1fc9449fa26017cca", +"c062bec1e20bb8d362bd817fdc306fe7e0362271494d3447927f32b33740c581", +"5f285b59450ec1118b9d4392fc2cc1da7cdcb87a6399e8e02dff98ea4558861e", +"0715854b6c69d0607e2fd47f5d3bc3c5441a3b76a0cfb4202ec00e12e744e7a9", +"b71ebce0f3251587174fe1ead0e8e29c8f4f0ec6fbbc853b09aad8e5f38ede08", +"09111a08e65d372bbff427cc08312cdb5f38e29170211d642f4efe657b27a345", +"5fe0768a47090dba2ecbb66cf4c8d405f992b0e2a24c03cf480f77a70cd8efd0", +"6c4dc995a2df993a5f62f4279138dc6ec4d578c07d9e46dc39a20c2fb52297c5", +"1b280fa1b034b298a1e3487d0da13f3d3be3b387547245e2ce78d20ca9c945cc", +"47215f35b93326f066c3a523c972d8fb6988c9cfb03b3cca44cb3f4fad9a558e", +"f90dabcb1ec8980623df239c6ad6a325fc79736a572eeba369c61d1e8036d1c8", +"f8ca767edccd21e8cf56aa31cf258e40c26060be7e1816719ce13866059b74ca", +"43c1206686871994f42d1ac390303c9a56406b21568bfe7c8005eb4a1afb21ae", +"fa74194315b653cc88fd6cafaa488589c3059f3467aae5b9ea5608d645a8fd0c", +"66299b177b6709e2ff572d06abb742105a9d8d7cc1fe1a01b4374e4f14120ea7", +"1985b7c612cbb1e0ef96a07c3618274fa21a43a0813e692a5f95d46864a09b4e", +"962f24a97903d32c37d901cf86bccffa592d3917922e8868282684cefe819796", +"5331ff0a363d439bb81d4dd948dfee657c3c574802b6121688f18a8adf22374f", +"fb011851aea139806392b4f9f2d6233669111d12987fa9139c21f47f8e1241d9", +"80518854fc4b50d1fe2f2589ce5cfa4ecf1c42d8ca0bcd1b872aca5ca0e52cbb", +"5ac88651c3681fefac99864fa0db4e591942d6afeda35e3409304543d45da26d", +"44a0b3c6d936fa99beb853fa979eb14f13e8a86969412f614771d389d992c377", +"d5e4a2aa0f8a881166d20ad60e6fb4a8cd471aae9869b3991cec63c22f32e255", +"c83cfe6b93b0d690a1a97143cf64f69f9ca7316f43b254c6fbc03012158d6d67", +"6b06c622528eafb37ba442496ad59bf2113b66b8128be686c8412a267b672ccf", +"5c07819d89f3132f6216cf62009d95cfda05bc4345330647adee20b3abf343b2", +"96199a009886066775754e23d565d952640e6954ae259f0c222841b511a4e821", +"bd2703919524d3285dd9fa0bdf3dd506282b61d67caa9fec217a81b3e7cea347", +"ff683683b0d0f61a24df7fb771d7c6727e2185c486115bd65b08f353168335ff", +"3db774ef45be39bad54ce04d356bcb52704c5aac795d6b3a03da53a45f938fa7", +"7d22db901c94e209bf46f3ec680b1055cf681fd74025a272990a3416cfc37728", +"70f716da7a64617f58126090a3cbd619c851f059728b159b5a4bde0e0e93bcd8", +"48b16434770b5a2873c290e50e3bd6f59a90217815523abc5ec8033b5c5004ec", +"6b97b5d7c07b48bba28090ee1af7a119c123dc8d0af284ce68ad1e7be5154e00", +"94ebb121cdaae52da30d0caa2c586d6e9e0bcdfc2161737e4ff35e2997744cac", +"7dd522ef376dd0e96306abc1be6e1ec4eec4695fa380f30a7ce452cb358f85dc", +"f848e6676ceba21d1228b9d94675757599498be931c92de90b8ec3be3a694cb7", +"e8e634e2b85ced512776e0577b5fc901e3e9a3f28339bc9ded9df7b3c33b8c97", +"5159517cfe58976df87b1031e572f6e5f0bb0d4ac1453451375c7949e1092e59", +"94122dac0ac444750c48d6980f3b350a639df8c1479f493441c5f294f79e31ff", +"d26ab68c89c72d2e4e8653b4c41d2a07949969ec707bc3fff1ffbcf74b717b49", +"2eb26c12c45001dc956e57cffefa80cf29d3985319b460bd9ed288017d6d7d96", +"c7fd24dfce405b792fd01c7897294c37bd4bd4e93286a1f1d9e7eff0f0d07db1", +"bc31c241cfe6fb430e10543380b875140c194efc6f895c47be57a3f2f9af01cb", +"c20de979128acb6246d620475355ef5d3f678a171e81c67defb87a55292d6a32", +"b6ee205092cca81398420d74124bd62d493d6514240fabeadece7eeedce133d2", +"b7a717a08d7bb771d800c19fda05a4e7ac9bc9acd6c3ebae94156b8bd6166344", +"3385275944595e1ce1c1aa1234b92cbe6b1f529015460aacbe2975a03075fd1a", +"f8454f608795d1d233fe53ed86b361dbd19aa63c591ef9a2684ad2cf8252d104", +"14c09cdf4e60a8b5bbf1405c178485b55d345b9f28d77277339f1e37af83e92c", +"99458a028d3bc747d22976330757a9e25873cad1ff3410030297182b0a55a74a", +"9a91ae818df762adf0d15ac5af73ce85de3c50d2fadbe7b394049252e28a4b7b", +"d7bc5cac9c5f87c2094e726d6ca12ca65fd7be8ad560d8960a305d64bf6f340e", +"63dc2cfe61115832959586019584aabbd7050eb934a9630dc579d0c6b44663a7", +"e04deb7afd733211ebaee692afe550fbcb5b612d404b314440bfc7d6c9f51e60", +"700017d5f5fb86ace5f72d4411c59c8ae37f70366434857ffe84e524a31b014d", +"b990b47243d3ab10d3ca2600afcf32f83fb9b6d0a9c170cb3792f75b9e84ebb9", +"7b7592aff0fbab3503d45c895c58a279d4f15cc78076fa27f52562096b7e28a8", +"edcfb88d8cf82c494e673af0fa740c067639abb90e34a6c206d25c5d4d9ec064", +"a3b9a1b7885ba69d6c0501f5a232646fefc3a8e108e82a956fbaca56778c04e5", +"143fedad2d2e011040ffcde8c32baa7af50167b63c3a893673aa4c18f17e38df", +"92d15c970cf7adc0cdf4fb1df38307c93f979bc85ac655d7b07fb874779559f2", +"96320c590c0b5965db5e386ea08a4a41b09283ca798fe26afdd92035313ac9c0", +"076c1e52bd9ddef4e553688ce99ecaea7a0f89c0e2b651ac839802102931da2a", +"ea5dea997456f8ecfc98116acbea8941285522120791482583b12d4836f7c8a3", +"56b228382870330054babbf20a11f0d13e3ef43fd0b4fe19bab3462c3553a4aa", +"28a74895e7c837be1b90a2f652df11994bc9eef27d81ad3dcba9e554b7f5fa78", +"07b47665c77f8824b7d254ec23f9557cc602ecc448f6f441875f1070e631616f", +"54a684587e818c815dbc557e8652456ce4805e48a34688ca3b63c9bb62114588", +"b8ad0ab6e67b2bc498007a7983282ebd1cb737c29939a9add22133a033d8e74a", +"bcd9fa09cd7075fef51357b7bb06e83409467d22baf0a8d51277c9138342cd68", +"1b01e39790ba05ff0429ba79b632ba44e451a812645b1ab3b117589f587c8cde", +"ecae889953f3e3dfab71cfa94be824cbe1aa4582455e33e7eb9b762d891a88e8", +"d948bf014eba7d2db637a5b7e22123c7597ceee18c7640df8d9acafcbfb7c773", +"37c3f862fe36d6b9497664672ed80352d5015e572ab061ce3acc467cb16a0261", +"df335381e8b04effac9f6dcc57a09a66c4c9e09b8264d022630634f465b5c35d", +"0b7edce16e4ba271bd363a15540898ce5e659aee52545fe890a55a6e71a25984", +"37b980b4b755193baf6a3e2f34374b7525ca87f2fbb552cdaf5930b334695139", +"541983582a3fb39f422a7c9357ebd2cd19806dd5f65cbbd82403e95c2baff8de", +"9bd10504173663388f7ddffb61ece4f9e02358043de1fdfe461789469766c885", +"33f3419df0c07e31190bf104dbf0fb5e26afe4426ac7b07aef8eb511707fdba8", +"263bbb12bfae9f0417e4c268999d34f458eee70724231d05e481a93bb4dd73a7", +"18fd6863252aaa55eb365c73afe5183e32329c3928bbb7176147c8410c478963", +"09b6166c4643184817811d12c0d5993df05786da29993f9b6a7050a14bceb186", +"377dfaabfa788335f8bb205f7026ea6435b3832bff4bb24cffbe5d06fe8301ed", +"a4aab845c099b1285b6a49620d63c54aaf6c7c8683841db00c929c85761d35ee", +"ec18ec0272757fb66c78b34833346e473c3f8982e8d1e5f11c792647530fdcac", +"f7d205301b8f350f70ecfa4fc373264552997567ec3a384784ab551fb9e797d8", +"fa6ce6472e5b1e17736e3d1d21e31e282cc87c853076f988595583c171f78203", +"180fc03657338cc3d16140554d431d0df85407b8e58b479f565bfbf7370c0b23", +"56dd96c9de110d282e57702bb0c521132337051ad094fbb0db61f936a0eb1035", +"a975f8cafc64c3aa3afbf3d6252497e2ca542e0c972a7b9acafeb276751a9b2c", +"cf8294c52a29c751358402fa5de834a22365d15d28d089ff08e87e310742e08b", +"4a930a5f30e9f9d2605df8e9bf446931a9f7535b97c6b5390e30d28c052b180f", +"37ac36f3845cbbe549baf9e55cc48a7f32ae1428ec24e3a96c1486a81d0588f1", +"25df661abc284c400557b489a24295cfb46cc4122b890fdf4b2d7f254e3a1777", +"a757f8cd0f37cb51e980f5bc282c9d1ab1c43defc7f3fbb40e5b385a217f9d54", +"08faa0ba959ba6c5c724e883b81312532c0e90ff275687a157a27238908e6dbe", +"e9467294d53fb9ae4b82ab5ea704aed0ec989b7fdaeef89134bcab30ed3bfa8e", +"3d5a16cf6e5d60b8ec02306c1c94d5364dbff093419a8312bf505c1054d3412b", +"7f657ffc72bbe5f6bffc9bf31349317a112e6aaf37fa2c7c222bbc76599a89b3", +"6e84008da1582804ff67c02031cdd97fd022fa91d62dcfc38569e32cd0687253", +"72f895e5fed463a0ffa07b85141b7ca5c7394ee00de6b09c841ea01c39983dc2", +"7576449f84212614c8e1ce3d93dda056eb996d1bd77277db89303cec0d2f3cd6", +"7d3fb1a2704a337bbc7b96611a5d59c3a1df4f0c0c23f9432c5f50bbc3abe82b", +"dbb9134b15077990e3b4fcee8555428bfc2f078883e6ddb501efcca6a7af9dfe", +"affd3b84f4baa6856aaf530adf7576099186064109101df2bb794723e4a608ad", +"ddf2ed9160bca4243c312c8326977dcfd999608576e0231fd3daaafd9183f2a3", +"1bdafc9721476bc3a0ab401b062decb70960ecdda0af8aa56381ba9c18b4d78a", +"9a2748b5e9a68bae4b941476297d30eefa33c549ff8a9be6a0897751513bc969", +"03f5043a2ccf6f9598b3ea1f8c47548c79f97189afe37cdefcadf1bc99ecf00a", +"50c4dac1c4f00ebf780e3974894ca4d4981e473ab4f57e5974bc71d4e0e94c5d", +"80f766a92f0d0231ddaf8793f114142d5c7b9e4c0ff83af3d68abee31d7a97bc", +"a9eb9fca36a1dfa2a6f57afb1277212d4f4b94f848858be1c827240441a7c0eb", +"973c608326bf0b025dae1c79ffa4ed0c8fea3d2f0031c7d2670f8baa9718c625", +"2032684cc9136ea9e179a2fad06809e16f5522b5666fab7adbf0566cee482eee", +"534c82fadcc43b12e6d3378ae798047cf909000b3017a24b769903f5b7daa807", +"52f2754cce304036e7ce4fe00cd5689e0001a6cb0deb08989427c41b7ea13b06", +"1b8aa001ece75ec29c44840a5516093d6f0e449ebacb9c85eb0082f6ae0b356f", +"c3102ee4267e930ca38cc9a266e22aa8637d74bb08af30beb38f324fa9a43bc9", +"844295300e1521beaf7288a32d1d57876150a22937f1b167f3f5bd39dc4f8b18", +"1ebf246ac44ef2482a18656625005144be425436bed2c033c4b2061ac5e5ae63", +"2be8a70eb8c3b0cb472dfc5f9d1422aa1ca681b7623fe410a9f7027a9ca95d69", +"ca779b21a33c8500c6a6909bca3e6a073cfa80e561fa788e58e90d53eba40308", +"1cb082e64e2de5ca287017c87247313fd6cb4226282ae836c859ff3779532224", +"619d03514d3fdf5f9fc11da9bab723cf8a5cbcdb7375a75bd8fa4145af2ebac8", +"5834b0b8fa1f0855948902bd9acb59235f08e99c50e669cd21df1b9f800a2854", +"0f72a4a65d4e319055313bc032f98f7425beb91affe1c18c9d55f209ae562af7", +"69714ee42a6485f35f49ba5370cd8db1b4b760884b9d461c5e0c19244ea6492e", +"6fdf8a2a5808bea8f76052e5a16beaae469af4f5c8b746a20aaf4b92a79bc0ed", +"f8365d9c95af851983ac88acfe156a7a7e222e11d83be83137cb8c86659200e9", +"8b88e9902112622df0e4b01a51b741c0d81017810d0c981a6cf0651237efabb9", +"94cbc33f793c55a52282b934ef6689190b550e67179f2975d45bc0dd11bd7829", +"a1b10a3936672468789a594c2180d470aba330f9eec77df6d15cad95991306d9", +"7a6f439af3bf16b6b4af6603b27efc7b149bf8854b7b897aabfc072127eac2da", +"b99984400edbfa07ec94e18c151a03c3a727371a7be544ec72fda38dbe6716ac", +"770839b3739921fddb892843075623048cf29125e09cc8962d1c8284f159d6bc", +"cc9135bbe37f7a09a256e92411589c960f80f826185519ecc551090eed32f110", +"9c54969821914d00a6b1bbaa887aa505e5b9e198ce805d69893a8c5825713248", +"6f7d353fb1c3af2694b54e268e9207ea59d1b699f56fc38fb0b4f410c1b6f36b", +"b71cbb7e6caafe3484fce6077343cc68ad515e9bf998adaa2470bc8e31712352", +"d1c35c4705e9805f1506294742a146ced9df1640d115e32efeb4ab0026bfff8d", +"779bd37f9c733e8c3db8360081dc5036ed7d45a2c768cf984a0ef3ede67c205f", +"57b8fcabb61b38cd629195cf9942dad532e8a23b3b43a5efd1ffe446f39accff", +"1a64a5e51b9dd78c91c57ca0fc99736563dec32d9243e4b0560369006fa5f016", +"0b37866d7a31e8a058d64fca597d96616203fd9e0b7763c67a8e6239d49cc6c7", +"7d0060e4c98cb562b358daca31f837c090e239742f6dd7fa04afeefd6ac3257e", +"a15b8655f56da3154c8420bbd3f38b03abae9ca95ae22238b8906bcb85d6e545", +"df232b152bcf40d35f5cc78ba3a4199a13de9225979f9935e98b269f67a92cb5", +"dc88c021ff75fb9fa6d6bf783262a6f9e48d8701a64958e0ea5c86f5be805731", +"81ab2552c405a18b74476c855fda484683ed8d5e7a4665b4518afae7c4aab8fa", +"a7b4b862a1e920f1f36577f85198558e75c6b7beea6915d6369d1aee5b354a09", +"1ebfe0aecbd30ff546cc33081cde3abc7bc8320ba6cf824c189d31a5911b58be", +"019588081108b5f38ab101e75cb412be6fb88a37f48a7a70a2b34edcc9c54b4e", +"6beaffbffaa2f826aaaa1ce4f3c99321444ae21f269c04e31ef1264ffd4b75a2", +"a4001d94a9233f037712b7f35de31b84134acf74f411059e9188bc8f2cf3b768", +"2df1220ca7328d412a5c4bfa6e4393420d88cd8a17ecba308fbe514588505cde", +"01e42f20850a067823a1094492188bc0c0f72f9cd34bcebec2272f66c75f73f4", +"c12b398e0a7e895b3def77e9f0dc32a8ccb1ba6c145f24503b563388ca307980", +"4d4c5faf379a48385b5a549568ddf8519546a8de709971e54ac45adffa854f3b", +"c5ffcfed21367f36ae6e0f2498ad2fbc8b468f3549973edbe9061e73b2591256", +"520e95ef798cba4365ec1092e07a7a1c1056b99fe019a2a3f267a36689ecafdd", +"a7d278a98efe2fe74f359aaaabab203850153ccaad8e36e851ee1898496b46ac", +"a709db0195c1a72fca9a793249079634975325f5e219bca6527dec1a064e1ba3", +"0e225b7b4a92ca1af8e51d0445f57c60275d1592a78af747321edfac8b2379ba", +"8d4b0c071ded6acfe348b76e44043232a893842416e33431172d041ad4be5690", +"ab730d42a6fd3fe234157f7d60c05ac073d086ccc096aaed6f398bb8338f7ddc", +"39fb5473e9dd678b14535aae8bfe73858d8d3deae4a85f90409fd01e63184e09", +"c7053257b8e58e97e15004238014c2ac5c29133469df69fd8895251a131127dc", +"999ce914759938bba3a53879c0f20ca0942032b55763d4c326d823f93f7232d6", +"4612ff1e5bacffbd072c47d4b2d047cbc087591b59702ad18c53b8761b30b897", +"4e5b5161da036730dbd6c2a648e13d1cee9f4a683132323197f81d9d5a01d5b2", +"e54edbd3902c01f154112001a51ac5c6b1847bc8e80279abb2f51fc1f827c5e6", +"2bb98aa0de91ce3d7e99dfaaefd830d26d7bc37cb954bf56b67f58cce9c353f0", +"6d285e739a38ad7615e2d478486aca552737d05feab9224878f6e514af47131c", +"2c0eceac915a1715c183ed69fdd59561dc7a0d475b4077a0b4d63ff84cc3f72c", +"c4e05aefc9e6796d590124e5aab6e67e38b3f0432333ef266825535cc2d73493", +"2bffbaf9362b834f3f4c6530da5347aa7b84177ad8238ab7b4a1905486a322bc", +"985188879264416ee63df9e29283e2476fff6d2ebe95d16342e6e43f0d60f186", +"e5d023499f6206102e6a4e4ab2462f50bb9dc93e67962bbb5c5e5a2f1686f9db", +"9b40ce147ad05140a8b82d7939e58090b793cb73a53dcf728e4c1df3275cb6d6", +"4562c14eee738de09fbddbd313289cd51f11ad81951be73a6c49a7fd2c10fe3e", +"99a57d71ce63f17ac4bc0822b89fe57bf1d90f9c73368f5cd734eccde24a5f7f", +"4af214321b0237fdf4eccdcf17e33e316bdc199961794d6d7e531769021b56ee", +"e7954d745123cf5e6706f09c277de720a0c8aa43aac1802a53d0921aab03848f", +"282d2ab432809d13bb11d779255b36d3af692d9e50499a4e2d03c9d6afebc545", +"dd68f82054de2f7de9ce8e6c1e6b125b4e1ffd4c7d801c8923439025a5d6d3a4", +"3ddabf5bafd0799b9cbd932c6134233df96c2ddf04bb60dd11aa4f9faf7a75fa", +"362769bcb7846c3278c5c3a012a34e73d3ea6b75c16f0eaa43ac2767725c459d", +"948a54b78666fd2fe4c4879e38aaa030de0d7a5e9a13bc1da318393326abda42", +"dab2352f68395c0974aa2470beca4a662b3b1081285133ce6904f398d6fb1ae8", +"7f246e0126b79821c8f055ea427acaa37ef150d6bbdb984f6179bba05182a9bd", +"93b570cb1d91ecd4eeb63db2d7cd76dd9063d5748f20e3e05d9a6c680c066133", +"3df9ec618a7ab2c9489f9c9ec2517878633c8b6327050e88277006a6998239ce", +"81340faf6d6c4cc188b195e617789565d9642e69f9b65c89d1f04c9ba3a5d4cc", +"c3851d80e6332e77d6f2c5c9a45ccc7483d6ca0e45bb16485ed1a77d7a845a71", +"f444ad6d6b25c75c0340173dc7a14d073b3128bbc290560280aa5cdef40fed18", +"acf008ff5fd841b0ffb4587153e2f6570f7bcaa47cb6b5751233421a70bf9103", +"90ce05c0ebe5f8698c015ba486096f15a63cfc153ad3331ef538fb7f266e1329", +"93b81be2eeef5fe7cfa547fbf16829fd729cd7ee1e75b4908cf1e07ad14d4d59", +"64c98ea331ed0f68bf69c6501624cbb4dbf7f0dee1e0fd2f8a8ea169bc3d850b", +"7805fdbfbe9e6f6f20cc9bdf2a35faa76c2bcb37e6dd379a1b1a91b37deb63e0", +"713a3edea7e49bd9f8bdb8c356972ebab14d3646db7e9791a213982e91dc91d0", +"d05141eeeb467a622d8ad66f4c06137667a000503049acb5632d2bccc6ba7956", +"f2a13c19986eb0b926519033942411aad2244706cd01d2ed0b191bda10a79207", +"0338fdd3cdab503d68a28e4cd4372abbd93a1ffd413e05b01a6ab98289480c1a", +"59eafb23fa79b9dab706f1d0f3ed31b7d5018082284e1a53f56af977881e77a9", +"3015de26169861a5980b6de97181d58ca656a05aa4f19b5c53f673ac746ec82b", +"62f6ef8e35629e538f5febdeecddae88b1dd5f165ebd5303d4ae43781d912dd3", +"0134826c69b6db8c246fc24dcef8f0bb0754b188abec58d89e7999c478ed4a50", +"adc8fc3a27610d7ac94d808eed9b7f815cf594240204e26733d2410f8573f904", +"da9b42321867933ea4d5d3829dc6d26e577a11d1c3bb08972b1f036129ce0962", +"9b470b7033ec2ef0e9d4d264c5be22c205d0a528d75a6c03ff2af481e0b33107", +"0c8beb18af4b5a1a61a26b1e16641a2d3552f3308ffdc8ba07533390a394a6f4", +"ac564c8d297fc6035f032e8a4a58b0e7ba5fca3c91e70765703e6708b33d582e", +"2d995e8e79109d184e85f42d7875676026c2cb25d48dd5abcc90595338fb5149", +"429b50c6e459512838b71bb6682b547df3fcf0752f7dfa0d005f40e2582d2bf3", +"7c3b0efd27343e1cc131a2171ecf841e665d88b52b6ca2932b7f792bae305f01", +"c1aca44f3e3c6d52281dca6743d726d7b851e5556943ed12280ddf865222f75f", +"cffa83e00069787e0d116f6d219d75e6cb08bcae727789955914828b42cec577", +"cebdc4941806c8ecc36c379533379ca8f218bfb62c30db64ceeeb82bdf14fe88", +"917312b205bc24c9fbbfeb42dbd55a5395ffe42ff0107eb521b09a95f3376f2b", +"90229739e9b58541785f51c111148ddf29842281907d1f5365e187bdfcf8c409", +"d8accf0140a68afbfd9a31988798775b83219728c085c5971015b82e3cc2d83f", +"89023584833268c72ef5dd1ef665bbd1e8a0cf69ad02d129ab7b167410ca3d99", +"81a838f897e3a2bdcd4629db143bcd7d7f3997b332905a1bebaf225b9aa54441", +"429ae4628927a9a5c16230a8fd6f0e5f652396aa94de3ec9b2b67eac09a0a6cc", +"7c65f762a6ff57cccf6fa22bf064d135f7b6f0c6aac77b7c07911429cc9bbcd4", +"7d5e994b2b6bd78a0405feefc8436a1563b968b4ddfeb1408220fabf8cc49c0e", +"cb85ef47a1c2f4924cc6515a56cdcde23b47981a49d5120577146e8872b94db8", +"17d2d60b683a96819f4cd1861b0cbf8d89e80b1ef21ab44df4e771a96b540eb4", +"b6bf3d7f8f848e05bcea458fc0f78cebdac333e837aa0edf4a016504468efc20", +"8d4415f547b96c3d024c313c0c34be7932d36dae133461921d07d007047efb10", +"48517c8f97ac845039d9fa739ea982a47563d134c59cbbd1dfae2d91b78be1a3", +"de96dc6d65b10125aa25782f4562226d6d15f4f3273f37c1a99c9621ef2f0f2c", +"d15bdf5d7798a4d842cd7abc494484c5060ba11aac6dae658232c01d76cf32a5", +"fa5daefd99efb53faf9ece241768d6f13c264d9478763626f88fbaeebdeac3ba", +"ac13a1b348aa4e1bc8078c31f30874e5030560144a09b7a6c2b0eff4b01ddd6d", +"f039b89e14fc759425e8c3762d48b4fea7f9d01cf4ccaf2275479ac0ab05bef4", +"1fef56c29dfa3033ef3f0b44c2572c279aaa25171f9a77923b07f4eb2145a51a", +"bb141dcdd14d89f9a00cde6002444d5cdb467d63e3b78fbf0bbfeb5dfe2a10c3", +"91c6d2c8760a90ac611e770005db3c1cdda6e2be72ec6ca8a6419b28d73b4b56", +"8d53024b5340e46556051eae86e17f94f80283692ab6ecf7a21e485133736e9b", +"18646b57ceb8c3f7f57fa6bb95ab7d70e4b3265d60ec4bf5ee9a870ae3d91cb2", +"66a9a530fec3883cd464158c51418e8ce0ddb8332f17733515c877ac896583f6", +"f261ca135c20e9a10dc4959e52e36dcd44e9bfab35a66eb0ddeb7d493e35d2c3", +"c6f95309ae789aa0acfdddcdbb9dd967e6ae59fee45025e726957534649e9a61", +"cc0bd316966fc89f56fdb2e189b6291c32d283b37753f98543ea1adf1e42395b", +"72c36e036e34dbfbbaaef1ec1a6ee029fe2384b491a030474a14a0241872d7b3", +"3989993a95ce9a49383a5e4b1a568a1153d0c6863f086335884796f82aaeb37a", +"4b50faedbd2da5db18f31278e514212a9b0c2f4afb6fcf49c33810f4b4cf712d", +"f12a5a9cb7cfbb24de1951d225b6e052cc926361ef890eb28d6a986416398e45", +"a254fa3cf6f2a002fcd7c8d7b7d42b073aceca69e969a5b0467d4d294d8c6ab5", +"f5ba1c0bc4941433c992bfb928588e7d104edaf9e80b216ccd10fec231baec28", +"4c3dbb76801abc0cc744a5938752c81050f96cc5a52695722f82ff6aea6aed40", +"a79108e6371f378c52bbd930eaa5f594669d31abcf58176ab582bcf0d74ab652", +"cfe35b7eb9a0f0c94267d7e206f538b850ba365097a93cf1e41044e30a0d601f", +"1573ec0d9c693d68a43c66dd144d89faa036364a9dd96c565bab73e909b3aff1", +"f67e904fff8901b685fd3884b1c9c5e214bdf7cb903464556a1aab8d14577270", +"4c961e363ec78cf19c98b3fed46508f5643361fccc5682623fc143caf668ae45", +"f6dbb611a227aa8c976dae2e65ac8c8a8881fcdba2d30351e9209fc3c947f617", +"25769fc7a569aed0a28781f9b21e4a672e414a4fc56282363a784b9c6911c4c2", +"93ccef3a07aaa0b6896c9b9230d2451cba537525fcc9a7964cb546c4b1cf7f19", +"32333244950be2f4b97411aa2e521ca2f2026825527189114ffada72d856b6e7", +"ac2ddde6ed8962c249e4355335ce8ec1481c1447813f2e18ccebe52ea774be09", +"3444ed237a27bbe8b13689b3cd6ff8f758cdf45bd1f930db115153da54adc49c", +"066ecb7173be7b0504b13655960a6e548572647c3c2a016ca484f198dbd3b55f", +"bec5e63a06f890773de1db6b7ce388d588dcd42d1701f2edd0346a0bb0c6fd5f", +"d6bceafcdbce5963e84b4daeb34f96f130ac08af40ae3a3f4da8ea451998a4f6", +"ea636ce4d7bcf68a9781e1dc0da61f3f1d54424ebc78eab1437f85300c0c9b3e", +"e53ee61f7ff01098ebfa819ebd06d0cc76c8f7b79b0dc1382fdb35375b5f2be6", +"ee4c7f9350d3afc86a8393067845f7a0121f8e2726caed7add44b7d981289287", +"359dfa5ef6d68cc4b9793c66164fab87b0d966758209486e4656dedbfa0401e6", +"da92a57ccec105a5371db14422ca22646396c428f4548c4676c2934bf8ce6bf7", +"ded4aae22b045bfa6deeb82da843efe78b28ffa06d936e174acd3980c9722ae7", +"c79fd5f8d4b01262ac673160cb32f6f637776682c83854af7059946f1815390f", +"e532475b7f95c8cf5e25dad22baa2b987d89c43a5de50f7793fdcf19030e564c", +"3f1fb816b5aeab8c283402e2a2557944041daedc3cf56858e21b12c1b3ed1bdc", +"acc3b0bac7b5caee72177e82ef621e0f6bf43f17710bb740cf8cb58c054cfcd0", +"17e8a80951f5b676fff4f496c6688905c4151002ef6cdcb4c195ca1ffe696604", +"5799dfe8f070bfc1f9e34b9d96626d34246bf6b0bf5bab3a785747607c257621", +"a771fef338522b82d43e04209c33eb7fca2c334215f34612d41cc3c94577c01c", +"5a619554c8c43fa7522766829bd6f0d22dd82b87a767f2de84283ef25384051a", +"385e81e1ce9261c3a12ff8b8e7dd10db7fe0090809a4761541e59a8af1942a8e", +"a6c9247a760bdd862af1157fef8e4dcac2e33d7f60e394b149f4e22620486d80", +"0d15be900e19b304b3a036428ef9abc5e5ca72666d08f6f67d3b38b41118b695", +"9679c2b2e05b13e317709a23bfd5429762b3ef92f6d3d84217444dd1a99b8217", +"5bc5992d69d5842e119fe594c175638081880bc4631796379bd2ccd3b7124675", +"1373183a485d08103392c00712d0b96627fc2ff23ac52fc5979ce27be2e4262b", +"f4393dd7bb31a5c249d87b0cc93ca5d0fc9c6b28597bdfaeea8cb116386ba424", +"95745c745ffa8fb927dd0e8db59f22874e87e97857c6aa8a5ed22a063a645fa1", +"071a6345eae0212ab342367bc187e64868f47c4618fcd6432931f6aaaad106c9", +"2615452f9bc4efa77ad62ea21c7ca25fe387f18f290747ccfb9ebc876f208e5c", +"be109c11056ad49cc686bc9e581a926942a51912f71c817fc0dff0451c3acc4e", +"a739d1bb89c0af34ab5538ada2d0c74a086fb18f370a3b6103782ef715ce73f8", +"e6afba12b462921cbe7c47280a7058bef1b510e2714bbd2108ab4d371a035da6", +"42e0de9c8549b60740834a9e70762958527810eb07dd91f35a0edea14865bb13", +"2c5c3ab08e620d56c47ad8c1e815c556fd85b51389ba55d7787bcfa860ee9a56", +"20a892c9e456b4a15e8fcd621cebe3fe0cee5985b9f17d75c92de6d69bb1107c", +"a0f0172e9a9003259b87fac6a5f364ba6e183fb5c606606458c632912021fa90", +"6141c7b3b3a3eee139e608a0bad9ada5f122610fe5bcefc0aaa6a6aa54d368f5", +"86bc1cf5454368b92ce2a667adf2d0296fff7fc9350c11bd1c56c893efd6384b", +"7762396939fda1cffc72c273232794bb9e8fea91af248528c408ec39a1436be6", +"17f8ff08714a92852a199283e23de6afa58470e4039a2ac3a4e73189d6e03577", +"e22b0b6b09b53e1180a3a65cfc910ac05bea47370337ffb40547e191be4b7eae", +"81d3d2ceb730d93f2117e2f6f5a80f16dbd18e753acf200a9b90d782901c00d4", +"0f38c3747879991e918ef90a978eb3e266e9d20fbe54fc9f4fe75237a75c9f1c", +"2034c6bb3c243296a40d47dabeacb28dcdee77e99b2f4a18edb273aa87bfd658", +"ece239ce85e1173f39960b09751b00bbe089d59353ae7f3666f5ea22879dfc1a", +"f92e67b6c3f112fd0dfecb85e9833121e57240c04f864c40a0c5e86eeccb1c56", +"ab6e37a7502da756e41d25a8b2a4981d082277286e1847d5c1c493df2fed18fa", +"0308d1cc3a7d206064280f3d1fb6ee833e9c02d9a7667c6749025d6032ef5935", +"0f17db0f0057ae61cf7ea6096d8a50f8f873dcbcd943091e2d5afdea02978041", +"0136934215c61edd7c1d698d2467152c288551103977a9139c858136554bd9db", +"062a618c336ab945580126a14aa4d07ec336ccc8f3e32339cb1cd0a0d1dd7674", +"be5cf487b33fc52bd59f03f2bc4eac4d817c356ad67dcd11f77114d1ee651e46", +"d62452a654340aac6f62cd9302bc8007a370aa332153e29b4df6ceb58b2d6fee", +"5e06a4087bfdfc9f9bbb94972a37f82d3acc52a5daa93e22172a1f2ffde28cea", +"de8e9646fcfe4f5a8deaf5e4a893ccef06c87c791631f04728285f701a0c6285", +"5c2aca136c5efec8ad1cdfa3d34a11a6fd12dbed930872b936ece7993d32cba7", +"ff62d5f62620cf352bf58ac18af4051dff9d20d796c3a906f446bd307830d447", +"0c67d43ee7ad19c26a3d24c8267c37bb698f884c6a7fbc91713e5610a88882ef", +"d62329aa2dcada088cd7f7f81f7dd6af8d2d6e507154674e9e0ac0b42c63eeb1", +"382ccc97cb3b301af8f77c21953f2e9d3bc627cd7f3e56b661368e825f375f45", +"9c0a2646970cd04dfef6b08018d14bd6d35d30ccb1dc3995b723ae956dd547ec", +"994d6c7b74cc585b2e168678362e5357f49bcd4c8d054bde8f4d444552617733", +"6c7c2d84cd618bc265e01477211bf6e7a93868e48acbb0f69870612b3971ee81", +"48da683177c7dd82392d21f0a13ad7556ae07eb9fd03870df7968412be97f37f", +"03798eae0c99aaadd0504d9964339f486d447f5efc82a95e2d2a72377772efd9", +"251ed73de6e73758a85cc8b6b3bf11fbeb220e9e12cdd314c3878557b32b55f6", +"0aa93f6345036b0ee9e58f30106a531dae0a385a42e9a9d57551aaee5d228fab", +"6fdd1ebe2fde94875747641fd73e5d120675f45f8656a4768d76d8071108b799", +"095fba50cf84cdd4f4d3dba6ae9ec9b06148eb46bef19029e1471a19a0459984", +"f35573f978c9b65fb0e0dfae9e93d48e5b931afb492da21d2bc5a38c2330450c", +"654292a190d016beac84ff2d287b046ce2810ed5617568896c1302e255a8b68f", +"a8a715af3adfbab012abf3a899e3040296c21b90b36572646e6d95c0b4c9632e", +"413b1f0f3dd7a4e73e7cc8031780de1bb3456a270e7be673d1ff54142d6eda81", +"9a18d2facd7a9d1187516d0ad7ae7c8ac4df60cce82868edb8d728a3fdbc8b0e", +"53d512987f7992387f8b29a6b4636948641542b38b0f8be161d7e8e99f636a8e", +"d770bbf2ee6a120cdbe2c40ded56a58054f4e335d9274f64126a00ed936fe6d5", +"a78893790d028f88634b1ada5beccd99c48cf02174ba417baa29361f74fb4be4", +"a82505e7ae52e50a737b37ffbfbee34d060c2af17087334e9995eabe91502de6", +"9ac8b8d081c5f3e696b957d4444ec5718562984cf3b051916a042c584c2a5713", +"36c38abb403a17c054db8e41f4ce0da11175bf5003dcb7e9c314824118ac92fe", +"d272e643fd93b298e4bf387de53b6f01db1870060f87b68660648f2f060b4c9e", +"ba46bff34d438e1a086e32d996069be3151e15ed116a894b59a915580c82ccf1", +"2a3d2a896b82b2e058056f273e4a0dc1233a7369dd89d55509a4799fe993e26a", +"4bf775e63669aa919b49cea40588033c1b83520cf0b417580b102fc58052f1fc", +"42b566a26aa07764c9b1c5d20b51a8b2213ff4c5976bd4080f603ff03f0831e6", +"d3d6d41a49ea4235fb2d381a5c176bbc90820136ea813069a0cf45ffe7c14c6b", +"9439909b5870e894f40bb73f77152f831e64a7b83c63152824a5b0bd00025e4e", +"de5fd62a2998fd691f57c1c0ff7bfa691611c443283314df0d2220d3e1961db7", +"d308506c751775ee7b5d2d217656153457fcad5ecc22fdad49ae3cccb48d98bd", +"8f40974a37ce9865fb4560d2be69716014c511403b8d6c16096f5982200d80bc", +"9dd8225a7b801eceb465b2129fac7bffaf4903d8e0a115604876b74d54fdd7fb", +"671a182b83120722530b56d525e0d8bb04d28396dd252cd2d0bfb43f0bbaf077", +"aef0b594cbd2922ff54dd151d293bbdd002b222dbf1bf04b5ef27c50aaa79743", +"df4fdcfb0b552a9d82a5f06dcd765e3e9fcf928468d642410543728b3c5cb75b", +"6b7835f814a3822e47e9414eb746405dcc06451d955592bec2d23c4585ec48c5", +"2983d81de17147d288466652bb37f4e8828f19f2d79605eeae626955f700f310", +"5686b63857f901f8c0d39194e6d4cc1decdc9f0c58efdc5ec8bfc61ff500e973", +"f46df0ff842bac970fc821478578c83a8fee0f0c1cc08ca9bcf050595dec0305", +"1f8fcdaf407627592cc49c95cd70762f2343bc123f6e82572a63c7f0bc7bc0bf", +"30218d208db065c6e34400ae5fff1c644dc209f9a8f1017186b84fa4f7d5dbd5", +"aa7c5a84bd25fea1c58eb671e89eceaf8066fae2dd26f6e20af0ba8da3cceb40", +"6c354d1c4a23c8e947e7ab849942de1c4ec0cc158d14801487d857fce67dafdd", +"d29d21fae667e384ea891c751b42487777c44df2c8f1eba14f5b2704db92c06b", +"8111eaa217065485499e6dbbd2a6387e360e84947e891c9146395ba4494af0d9", +"997e1901d513430719c2c96643cfbf5cd32e80eba160122cab14f36c7cc0e4dc", +"a8f76e26aa307bbbff3c70927cee68fa7e9311a28a52c09ae4e1ac45ccb7193e", +"c08573554a6e1a3a7c8c41a8a3f5f90a99fc2061af8d515048562a8dc0409c13", +"485f43bdb5c7a297f6e38c25b41202903662d8d94140c56423f39fff21b3d617", +"be05212b699116a23743768bc986984f0b345ad60805f20bd5fd51ebb5756707", +"9d12b4c27c1476326e4101fb5c888710d0ba1927d85cb2395fd94ac9b7988942", +"6bb240ce52d08ba33d8f6e033bac5ea89d58aadadcceb9d7299ae04b21f16cfc", +"e6994ca594a1fc4a2cd1052cde7558b0f7211f251f2ca4db34a79e3649e3f7f7", +"d35da651b66d557e1fc4540329d2ee2ae5c929202cfc4aecb81087e81f074b7e", +"27d930caeb6c4b854959673fce7f0722a69b12b862476577fbe21c143bf51dc2", +"56d8ff8caede8645437d2bde4f732340d265d7192120700bddd8cd1c62940a7e", +"7c13e6548467a8d892d890f944bead7cc60823059213005706a0f5af1fd4fb72", +"87b443922d7d60bb7f0253071619d1e31912c44aa70f50875e962484b33c55ba", +"1785a6f4c9a4b06a1bc60f31ae38528d64c1394bc0a9663d6d5b195f82fe937a", +"7cacc89280b539ef236077903455fdacf72d9f32021867522d2ed4bc22370fba", +"cba25c4eaa6cd851ed89f2bafb07197887d62d8a88ca360b72f3d50f75e671ac", +"3bbdfdc97a8cd5d11f48fedd9751af994f1a8c0cb965b4143d94bd8cd6ee1bfc", +"3b6d05446036a99cd9467893dc2041ce7c31c7977753fa0ce9372ff2e94e5db8", +"a6e29228dd2332906303f863d6b3fda19db1a3c8a282c0800148d87bf5e19cfe", +"d602b052b18af6e2d715d5ab39fbba754a02fa138f7059139caea465ab37fbad", +"b4c84381e21e3c92542fcd12f78a0f8256d7c872db269cb263d7afb9ace6eb31", +"7e92ec8befc0680e150c340fcbd9a4402cd31ce7effa8c7a43c4bacb2039636a", +"3f3644ded2186553edb1e031a4ebd6ee27f570a8d78f8820be60667467fd377e", +"cd122821ac60bfbe6414372ea6222629aea084875647824ef2962fbe9e9e008c", +"82f7ec6dfbf9bb507d69da194bc0f1c77cfff635f3fc96cd9be3f155907459a7", +"d5f107067034a696854dd2ddad3e6e67015717c15153c343c21d567f98f43acc", +"a9b291759ccc9dfe70af766569141c00ea4c9d8a74a4527ccd073fc5c3739645", +"f7c12e1155b546ace8c60a216549a244600347454d4dae4a5fdc119276c4bda4", +"961c9e2f015e4bcf06e6cf2e04b3deeb495cc92776085af8760d77ba61af452d", +"1ab9a612073725423d6626148e72b4b69d00088bdd12340889a7efddbc8620d5", +"65df43b14305f2a7de1846e988d4a197d493e98444e396031ed2fc7efcbd2137", +"a931c3ea165e5ff36b2aa55befb144943b9a4dd51fd1509f6763fced48572065", +"479c1133b5aaac5d490bf31f80258ed0dd6bdd46d893ed9012610eb34739cff4", +"8fd183b7d03d39d857bd78ee60f7c9b8f06bcea59e90b2b9e328692e18aa8c8c", +"f9681766393202342b75ba5ad585411bf0b20edcbd7a161814d24871630c735d", +"07aa11bdcc83c04d7e5607c67bc8c9049fd86a29b4a92787c03501b4f16913f5", +"607e4f56d43f87ad7d505fe36506bb95f2114946cbdabf5df41b70930673285a", +"c9da35738b4336623de4f3edca6a0e1380a18f8bd91179ec6e82c8d165f430e7", +"7c00b0a48b98a763b7461ca9add215f32c48a03fb04d87788354eea4938dda3e", +"41f47f0caa2fb5f85cbea544e1d9517d9acab02ad06fea3dbc50bfe59f973abb", +"143a4c141a9d1a44c03625e031b769a03725a7edf5bdd629965975cfed341662", +"e8d0a97c507d251e14581f56f820d9f0bb3d0ad5bb7f8f4431deb023912f9fa4", +"696b1ef20e0d1a3c772caa39e4e26efe123d84ad032c8fec193b55b25a7ebd1a", +"70e5d8c79a75430bc68c228f2345cfcd8e882871cf54c42243028066f9b415b0", +"2e5dce6430ff0854db8fa0cb34c8eb4daf12ca8e51da8de9a14a02711ec96c92", +"0c33fbd02b4ab8141772c350600961765848e72069e4c0a9397e55b4beb9a1a6", +"b8c5da06edea54f81a2afe3263f3c22b0aedefc1e342c0188695ce2f45daf614", +"333460e67a985a8e352f6d95d01dedba8e555e621072883660a7ae954d0d90f8", +"271ad913edbd9b72f70b19cc03acca245969ab09e0a5d9f122d7a0f11d2595c3", +"1bcae69ef7bc9d3b637a2ea4080f01a9932e6d20290adb87596d49531a6b223b", +"35b76ece18e1c4d156d970f86877f34ebdae6592c60b8df068881d841bed1be6", +"f4cbe9e3ea34801d098ffe73de049bb5ddfa41087dadc91b938aad25578c66c0", +"b0a843e543e8120de7b2590dbcfb18f391bd53d40b64d9a4868265c615304c4b", +"1f8abcfeb56ae49d6d893de20e722e6aab7fc5dd9208e8a52792af13685dc02a", +"01b9735631df4c2953541fc16906217f26a1af5c23e7a2eef345e516a756bf34", +"2bbb0da3493a7f541593a8678fcf807622809ba6fc08cc6828e526d1bfa11764", +"77e3f135a992585dabf6208d1664d40e725b5bb0e591ca4885df36ce43a63b6e", +"b86bf2d7648253e0d2ded5e81843f190064fd37aced6ead38f8611498faab564", +"c9fef5d273a99298a4a4a6551691ea33afb6698d028af6dd08b22a013b1d8c81", +"e2d72004f8f59fd296b6f915f850206a2bd6d0f520559a39961833a8cf7022fd", +"7d7be68058fb28b05626b6ac91b5c40e12f7fb36ebce6ce5fc3fd787f97e8fed", +"2495ca7686ac191a02aadbdeb4901982126d859600fa3be9d0c22edb1b6a846a", +"e221c764530fd802a3c3923104b6c453386591c771233e037be70640c049acf5", +"40bc5a05eeddb5c94c36259691bb84faef976b9388da999113ddb505e18bb949", +"631391da68f3054973cbbc970d0b9f303e0b697241e2653f3bddcdf81d92a044", +"bf0877b0b5cd8c434347cb54af95df00b677ffba6dc77641552a1edf7cfb93b9", +"27a4868e920a81de9bfe71f5c0dd625e1542098c2f35269b13a310362372465c", +"52a95ed95ab08025506427a516088ad925cdb2832a920a0b89409f1ce8e2affd", +"e28d980c5e2b70e373ae0dc6016738d13e2dcd62275c5814d040fa8bd3c75519", +"3e47c8baddf83d9b56559e7740f2c765fe88d7edd7e17cf96c10bc670c832553", +"6ee3df8af2c0f3641ac5b73f89660e8048678621b77efd0e36f849e610e3d543", +"41a407203331ce35f1f61c1ef3f885e8b8e6eb7c2e044b7d58c214dfce100871", +"4cdda6f7076a26b5dd63f5eb3542fe151b7266a8f79620d968b33a1a2694e2f3", +"30c57c6d43ebc35341624552437562ace4b86cab7ba45e1bc36f0af32792cc73", +"31e4606b9275007ebbee6bccf4c94d75392ead4a24d574385d72410db6cd2bef", +"00a774ae7d8526f3d055e916b8a336a6f37f7d5ebfbcf60955658d7a7a5161b6", +"b8fe4fa529796a4180c3c2f5520f12898938f6fe29d41f6b5051dd47f7d937fc", +"b195fcd096c5ea15d17956cb36aa9bd8196c4cbcf7e2582681ca953147fd1458", +"81f7a995c91656eef2e125c83bd92467fe53bfa56008296ac6214acce6adfb13", +"cb54cd5bc4513e2b21afc21d445315b1211b0e329fe645d932e9a0edd66a22b0", +"3d8a2b299b2137c12c5f3fd2be200ea7b666adef80bcea350ab8168a6eed70cd", +"6cf710f89856dfa735b21fba7beb95be18ff73a71f11064c48f16b8e4e69d07b", +"58dddb56aac9a1afaf34e570f74a3a4b67ef07b43604b976e9b94652f325119c", +"b4fd7cb58329bf5a95fec2b7a9cc81f909b644b02102e5f69ee94bb1c4721e77", +"40a69e4b7f733b8e60cdbe2e7a5b1fbeef864f1ad7dc821ac9210414fbd6c485", +"01c3f0d92f56bad0fa7297e363253ce91afd9fea6f4b220431b1a3006541882c", +"abec8a200db7e423df225cc8837f3de9bd5d5a4713958de796d1dc34bcf6f5bc", +"ffc7dcd86f34c56205296b41a02e29cfe8ab520ce8b1d430b98431f8642f4f61", +"cf5d48ddf3576981a23ef8800f573bf1ea59c3fc0a094d68005e0f78929e14f2", +"371ffc72b9d103e6be0835075059f2246a5e360130fcb374917f2ceb53f86e8c", +"c41f3826d749c13b906bb99b28914185fe596ee62a1acceace8abc134564a8b8", +"a60bf888d26553072158cead9c12e1ae52943ea178cc2221ff164502dd580c63", +"ab58465bbfc39244fe92a6b13ef6479fcdeec6cb118038659af63a8218a53cd1", +"f7717f073832af434ed4623dc485c325694e53a1eb443739276518c78ad32248", +"105fdca940fd8da347b538e2bae21160953bec0833c76e8ea58ad208927cdc92", +"8c2371bbe86b475b781d99302dece9305b546ad411d6522b233de74f55c760d6", +"1b033479aee1b28f7d7af5ff029d9d7f8f2b81cc9fa3459c489bc0ec215b0e6a", +"c52e5c7afd23d74de3a860f57c60cbe4f63c75cc388db596364492be46297f24", +"6335370d643d0f002fe28bd102a404936650671e312ad59268477368b0452413", +"a9f1f2279711f476c30e2e1382a0e99358c662a48fbd1774d7e5e7f8ea27d98d", +"09830c9a1b0779420f9eddcaa0c9a03303ea47612716e1d2bdcf9982915f3dca", +"6f27338a30abe4dd888d4df232014612ba94900e9795715a2eb989e1216a9af5", +"06567a37846cadd42c2abc9269f0565933bfe83b3af21a8355719a5aa3336ae8", +"3f119e27900f4648319011d21d0ce403317ddd16c73b7c7df0d2c3bca31ccd8e", +"749721101012951998888e444d7424a8231cae2ea75bbf5784c95bdfd12810f2", +"0f7b545a2d34c345ffc02b40b4a188ebfbbe286237d0d6a5dab7bfa4d368a4f3", +"9a4ae0eb8c3ec350068989cade49b369f6dc1274623ef6a297cc5d94bb662cf8", +"b4a2d2fdb5071f5de093f84a76e22c65382c4d3fa053511da9013ba2fe7efa40", +"87a039f6ea334b61bac246684c75fe201746851a72d944ba75757b7f858bffc6", +"d68bd08a99e269525734001e181e9e48f4f0dfec105ad341dd22111bae6b754c", +"930d10248d6ef210cecbcc5e26358f1c1eb44a7e56c7514472c548427c239b84", +"75d81280564d4e691410457a46379f6d1219c41cf7862cf9019d6d9889caa6d8", +"de7a0328bcb32ffadfeda9edbe576ec37cd2b8ba64d88f7190f061475f360901", +"2107bc0caa700c8c1d42a31f3b3bc34eec1785bf6043de2615ca436dc26bd39d", +"2bd47d0c8947dc44d858226c4def55dda2e2b3475e0969d34cbdf3830f1a9eda", +"ddd24fb80dcb3012150fc7395a32df3422b64e2e0e6463e3bc97001cff04008f", +"0ee403e1573c2c6566df09638c2e3b753b6756cfa89b60dafbb0ab85280baabc", +"cc1c07e470f3277b2eb3e3cfb2b899e0583a28c1436b1297224e2e7792f285cc", +"2c7e4ecc9ad44f86f71cd6d5c0d9aa6a10d8470e81c8b3f404ba650daa355c7c", +"2e73f335705e3aacd04d78e220612b54f7a815975cc5f3315a13ccbde8345948", +"a8b277ce1965ab563f7d330f59ab57351d10c192b2393cae30e435a4f9ff067b", +"4f3836c12fd1c9fd8d4522b0916dde72494b89cd0908dd4d91bdebecd98c1be2", +"5c6c6337984a961e21770ae4fb3c3d60cdd6c44749f66fa913cc0afba0bcfee6", +"4a0f2bee9de5229fa2d4cfb36577a38146eb1ca3dc0744ff22c4c26ec3fceb50", +"25210853c824cb8a26f3cad66c473d4f8a38b2975f0c167b2f7f56f938f69c82", +"d91fcd5be049bbbca831c9ada8515ac1cefeed7972a8d0a919f561c3f13d5d6e", +"86af58bf6dbf37da19ee089635c0a841b659689a0ec64e77a26534a9954e8f97", +"1ad85e67ca2e0d36ce4d44cc442a73a8c067aac3bc15a1aa339b081f78bdb1dc", +"0a44ec3c4847fe94c14129b74f978753a2757565bb3bd583f10bacd5b2f2a96a", +"6b72c32e64104eccc67e3071708798f73f8cc05b336840fa3f4cbc7395d8202e", +"03f5bd0a3da53ea4de7a317e4f05af850daa05345927a1433120acfaf85b2e24", +"fd5b4a0788d30aaa3b96687ff694dd96e87a2538cd2f097a4a40bee1b8106427", +"7039d8c207981a4d315b8ad821aeaa8b26ce2b6ab0c5c3e1bff8538928986374", +"6c4358ef84bd789ae8ab0cdaab2125df4e0d9508b732a36e4ab64bcf645a30be", +"cc9f6cf8ac765e680de75ef675706f617b3f55f449cea8ce7f1d3e7156d56e2b", +"285b3940945e27957a0a4908248c7811edad770f81244aca667826cae47150ea", +"2f8f55f4f5922146fcd74d16b4f59fd5b6d294af58fd472345d751d4124ad70a", +"c6f00b54b8d523c69e7b84c34d1538f11c2ef2a7f8acf9ea715850f339357bc7", +"f3ba1728d320bef6dfaa1cf383bc2f9684fa3944fa0827cffbecd1d1c8019bda", +"c1da39d750b6deba3f4a062a736ae34d9d035ebd8ee66e9ad3a7edbc1ff5fa3f", +"9207dae1e01abe402e5e0b37b418bbd46fac4c047aa1c0ec712e3a548681913f", +"b51132fd2d4b2bb2d5f74773431b0b47418c7b1658783bbed7369195b46156af", +"1b6dd3b5c41b2d5d99fdb2d0fbf7dcb9f29142f37f79399f585935da05c5de55", +"70e28057c83920ad5535b0020f167c8871ee118910e3591568833c73023ea714", +"5f6f13ec777a59b17791c8292a15a8b0ac48865206eb224c6edcad18b198bea7", +"e24f4c83a83ae873e010947e53c25e2435c5543db85c444150df4d6e81d561da", +"ba45936a67c752a9e5eaa26fe974e0cd661b31226140a6fc4a932aedf4c149d8", +"5be1e8040e796d1eed386eb3269596a00b93b39c37754105e108b861d66c45f6", +"9eaa5046c4614fb6a93a8ade544f20fe63e627642a85b3c4ecb938bf31cfe87b", +"16109de89780244733fd991c3ffac164dc2cc57f0c8c9387bf6fad5a8773d5e5", +"14bc088716ada47bf6504bf4ace6554da7c642b2dba3de2c38914f2534468026", +"e8deb84ee2cd32d09fbe5c7a921de89550763fb7ec5f03a24f3fbc91e8bc49b6", +"d24ab60da59846b69b88344fb976625f96d0c10c6c0c924b8c9fd4aaf01b32b2", +"fd791d73340ea96b303db2309d37091973a15acb75185901de5cd3998ae73085", +"107c20cabee5663797edda4695aec83bb01609c0977b5a59ad40720a3d99d5b9", +"f0e5c494e741482d5fda5eee920f93c5a9d3adb33cf2c11a0ba4759196a4dea8", +"4e3e235645095375b07ddb912745f1a674a51d9d7d83db61c7d07cec47ebee81", +"d26ec68f4a8de230c260a37f6e0ca64d37003e7463f3394ad6856f2f532fcba6", +"3774e9bcd930e166d1131939f623b3f5178948b14e5fa53dc5dbce7e1c6b3258", +"09489613d4dee865265da94f0c652a7aca4e9369fcb70ed5d8e73845a0d03fd9", +"3248ac02c1d9f2787d2101424399eace8d0ad08aaec3d2be7d1fcde3a25b63eb", +"fe6d83bd87053e0ddab426b5f1035dbf1daa40f29639d3e35e678513706077b7", +"5c396fea426678021575ba600d4a91f1fda1b916e116f1af3ff0b0b716c302e0", +"908d538ee1cb3b270730965424c15ee1085c16fc8b54d1e92e91d870a8184ade", +"7ab188a5b0c7741fbb3e2c0b1f8afdaccddd93e11e3fb1ecde1c8783ccc84933", +"0a930f20c108af089e864f6290a3c9b8db6dab092b3a77583b34bb4ff4b35c1e", +"0d24f819e32fc965da611cb3196f49d3184f722086a397c6d1fba1a7f7a2e65c", +"53bef52f912acf44b45bd8d0ade66755f325cd7ac00e72795ec9dfbab4c1b896", +"3c7aecd4d799a79df76a9b0db90550762e7d6a57f500d992d7aecf4757fc05a1", +"9be36ae28afbeb3d896958f72f9c02fd3460e96dc525cc22466ef40bdd4d8d8e", +"2ec94f116f46ac31b75644a9f2a146f370442c25342b256759bd732999dc565d", +"442f0a0d6f857ff5eb31f6e71794000d20ec84c4edbcabd43b2aba9d06ace8f7", +"e56f84a239477a1c74e5980f1076d86d47609447354dddebd8ca00337b2e5cc7", +"72f6bf66f69cd8e44aeabba9c0a328002e8e10f78ae4387e0f71363c04c75bc0", +"22cd12a8c47e310d4a8180412a53273f222c7aa1bf6042e49b4ecf491b5072aa", +"3d8fd84ddcf56602623a0a633b6ab8ed4c6c9584faf9aea9bde5817367533cb9", +"6d3a46ee9cda6b3390fbb4b754f5206abe4cc008080dc4e9d2c3c3726b03c857", +"a27dd55dbd463c35d42c94c3fe19ea0990e6bded6bfbafe13131c8b44fca20f4", +"e09d104e718c87e434ddd69f348027b4cca04a464948a467daf1d3a2820ef525", +"bfba6117cb85a5fbf4d5acc2e1f4033e9a062906b9e081a1e0f075d28d54770b", +"a44de9f3762171be541e68effc8dd58cef9c18ced0414c21d7633b2a22af3921", +"e3c0fc5c04d6dd20a50e1ce0339296dff9d50c639e76e1fddd5adf002956b565", +"e76fc7b7910c28fd8d9e354d8fe60fef8008e34ec617c2ca0ebd58b6c844ab3d", +"fc56257a21eec97161f442b1c3a71e3e0e5514bc511e0a5cc2f01573f3bf2b05", +"04ec682351566181c37baaf2084820dc35b4622e47df28cf2382572ca22757aa", +"dcaf25c1d6a99fdecb771a6751a53177a89b37a09a7ca2c8fc5362eda3150381", +"6f080d9992ec6c76ebe5f624853f64cc3c616d0b10fe20fdb6780c8fe47192ba", +"d2ffa451687b3da05eb0bd2dc2e5d44849e9f76a76763bbe7801054c4a6e4e21", +"a3ca6eac9fcdaa02a95fcfd2814ceb70f52de8bdb9c608b7ac71d6da87bb61f8", +"58ef8b9c98a90dbfb748342fd3591ab6bacadbcba941118f5c3e5f9c976f953f", +"f9eee10b06baec791480b42f74ad28053ae3cc34b4e0db319d18edfde13825cd", +"cfcf171ba5f1209b7d081bc5aeaa77c84fdef58715cf2b7e935879220fa1c88d", +"be6bbf79c48261a3c54c47d19003601598f219a985ab2ff54020efb18d53323a", +"2303588dfaab0e6e3c5d916b5ecb9ae48577b89fe0647dd34cb360d5d4c961e5", +"201519f518e33e1dc9c6aa4cc07fb18544b51cb98670b89af182059998524f12", +"5b007c0789fa9c076dcba9b7f21ea284eb1ad9119184c29babb4bee605a27f57", +"8cd2e9f1726cf923447878ba18c0fbdd1c1698710acdfe5b6dd052cc3bd193fe", +"996a3002ec6d59a20bc8be46f078107e9c7047f7237cf3b64e9b877c8aa5b276", +"868e263a7fc0bb0b98f009ac13ebc0ed6d4d1a91837fdc7bd1f40a64d5764ad8", +"9bcd070fded5d3b8a1f44d1b11af7c7c6a7f63f06fd84018f46158c2e07b9ce4", +"66222e5924f15dcd202f8f1323ee76fc088ef3073351f7d72bb8e24dd4df90a1", +"46ef4f691cf3d9191a10873662683002974810af0b2ae065dccf84d8496219e4", +"394eb6a12f3c23e030bad0f66cafc2935ba71fbe8988ae16caeb23ed401ea750", +"be899c34cdd26133f829ada10f1ec4d87544991e649e923fc40917f9d91c5d06", +"386467b33c0f3148f7531e0ae7348e68b26840be0bed4c51f4ec9a7bf73efcf9", +"082c545a2c78588c3fa89819c223a151dcb9c19b05ea0736945781cfc506cb19", +"497b8e5c841531203de45e56a8a6a3df4158d75e6530cb700f61ab9fb8dbaf97", +"448dfab8487cc1b41fde048cc85bb6eaf5039f0ab52616e1ca67346594b31797", +"32b34bf06121fd0546809d7d061aa8d157fc9c47b9908c45da54b0a555fe4144", +"52877b88e301846c75239ae993dd50b87014dd56309feda15dcb7f42e6ebf472", +"58d3f7e1e6a188456640c063c50d2be90afe2a07457e5288724d718d56f52d0b", +"11cc05625f4645043575deab804056c7a517475eabf6719193d7f4d9000a6f47", +"6aa04316582f1adf0b965f0fd8abb4c5c06f5d3dbda2e3a7b017a0d810fcc061", +"e574b31645ff906cdd8c7231b52ae3b7c57ef35fb42b00d6efc454b898945322", +"47386fd5d68abffed7116ad1426a515e30ce38ef022950f34f1fec130c613f0a", +"a42d55dbc476be0a02b09e0f7a1c1d149b67b4dfeaac0965990da1cdaa8d0deb", +"45f39c7dbeb0486af2ca4e1c3310f6a52fc340d4f2739c1d794119b47249f970", +"be80cad661628511f178dd4f0617a6b8ace86b45ae9801b3db9c18b6dde80b37", +"892ac99de9efffb8790ee35510fcc34f87980d445eb4ac1cddd42dc01f93effa", +"afb56bfc7554c61e6f72e65c4528f4b9ef6d93266ef1f29ed8cb5a25043f268d", +"570fd7be2192b237e9f1a8b9a82e7f48b5f81fbe8c971d7a5b063b550f8eb77f", +"3aed4d5c352758c9ee43fda617923c7a06a3c9a23936505d88325727c4ead7ef", +"b12afba595f181efa46b9dcf029ddc9fbbaf364c421bd782af5d8b7c463296ee", +"2683acb13bbdf8d1cc0a97f75a8a50118e004efc7f226d181f798e240597cf9b", +"55a0cc076db97ae2c41de5b43899c3e315ab9228cf030128a35b2e1ed20b9349", +"917945d603efdcb0c708802ec4567fe503434a758347254b653b8c7d4827f60b", +"29c563858081e8b6c8143ce5611fdd6ea5a523bea18cff94f8bc3c0310c8707a", +"0ed73856b5da22b2f0dd71763e15e232b85abbf98822b22643bb66f0f8134476", +"dcc9d1893280cb656eff20f3725e7cb8bb787b5f94abbac07c067913eef011f2", +"7d25666f06424e2d9bf6a108d65400a4581fb9e9be54c92f71baf6c2d2f90f55", +"1fd049ca0dd0678ac5a944470686453e3e46c8bdb6dc45c89e845b0347a082af", +"72fdc391efa4f4d94c7125c3578dce48024050832eba3dbeb6bf143f9a494ce4", +"c1a46710b7a46e288fe4320da8203285b3bc86bcc2611c0066c916272b324a05", +"88522ba2f65a5510672a5710b91b6bdbc6d1b9cf776d6f7091fbc538fb6226fd", +"8369c7f07b9e909c1152679abc04e036e59a26eb7312b1f6fe2b8a050555d087", +"99806ff619938f6910df77d8861169fb71c676466f21fc697b37613d513c5665", +"921a3d090a8f862d0071602ea7bcfac8aade06bb4d682d312133cff1a027327d", +"0309b100056950f36b19f90dc647a27e65083bba6a92a1155e1e26c0a4f86bda", +"a5deda9df42f0b10233d7bb88827df41b63b275eac9ad37df77b42a5f875f068", +"299ba984ce97a846192061b4ac52147f3dd5a71a91483df0ad04e37ffdbb4b8b", +"59b4dacaea63b77f53faecbecf57d19d296bfe9ee61cb90221e2605d5c7f0c7d", +"0b16db8e6c08f5ca6f2cd5de51281320bf5bb910ac6a1238f271152b9ade85e4", +"e344457a5b6c0f0dc4bfe08575ce068ec628d60163c5209e3059c3a60a176f6f", +"b3c16e152733eca6f7052b51376fa3be6f6a97aaaafff75f2609cfd651a9a594", +"e4da38d2727bb850780d37ae7967bf70794099f3e98a53f8f41c7267b68e8e6a", +"02b5b24943bf3ab0659643c1eaad8a905aa8b60c74b717bb07993dbc0275a6e3", +"c0834ac98980e228860ef954e563f21b749a407594c4c165e27fd2264e6dc8be", +"da1489c42411d7e46e813c6498f082279c6d76eee66684e2f3e47d2af5ca2f73", +"6ee23c818d67787c3e7c8ef85853b5a993dbd85c30a55e7778cb872c4ecbe5da", +"2552b834a020946f43f4cb85530e79aca258a4aa1d30897a04ac817665e030f4", +"70c44c5e7d3229d900b74729c9e5d390ef86357773df55d31e1ef70795616403", +"89a3698e7a773840e0ea749bc429101a3a3706277b9d5282722cfd05278c4a65", +"56e1cb2c0df95384a7077ecd6cfcd896a2fda34f4c65cfc080ef4669b69ebdb3", +"11efe081ba7a9b427f1385c783783afce90adc901a8d7545628fb19907b8e5e5", +"7eb18862afa358b74bb004473a46db2098290aeb616bbc42e57e45f8e65a4e29", +"4c6e1339f6ec894ac4685f4ef934180d376b1870005fb5c6298261353d46cbac", +"49e4a02544dce995dbbe1279685bd14ce709ecb2e18579ca7e00a4bd47d2c1cb", +"4293449d34fdd8ecd449a35f09e9a49a29f81b0a67eedce32219cb1cdb9d5a45", +"6bd9271fb8a7bd49a7efa90f373bd394e2e4594ed6c9053f8346d8169f4897af", +"862937f5972ad3bbd69475cc04dc7c9eedc5bf48dd225ebb513fbf697b4d1b5f", +"76ed01d9d0522d1eff45b9971419667f443beff4a5aa003c890938f1c11915a5", +"dbfc078e49a3d0fffb8f5c0154d67e61df267083df69800aeef10c7ffa470959", +"51938cbaecae5085f08b88a3856c8798f92e964fbf917ca22061e48fe54cbaaa", +"36d6c6ebe044e5bd11c9d4968b412ddc7f7b3861091089418e41a3ae4294c4c7", +"bc510629bec5411ffc3c7093492e24f954cba0539a9546ce19aa4e382c68fc2e", +"817909d72da92a8b27e9a3aeccf697cfe0eb2c0521173f19ee05ca2f21b34356", +"7710dc7e96e4ab0ace1d56ced45f568f3bfb46f735f2bd6e34d2f286640014f0", +"8b1e85f304b485451d76fb227ef09a4b2d921926ca34b2e8159a453a1fb50c16", +"3d97c183e1b32341ec2d93fad38fa0409b7606ce794cdb68adc6cbb53c2ed73f", +"b79b09fb572d54eddcbaf25894462b79499fac515adee8d810df922895b35227", +"cc4d47024e9564d4838be1e428ae2bd6cdaee5ee9cc39757c9f4a0722563d8b9", +"0cd6bd62e97825f463e21cf2cac000f58e3544368300d796ee08f7a6de957dd4", +"15242d7a16fae68fc9878f21470f7c4e12eb0f8b074ca3bf8e453feae6ce09ca", +"f4637d5fe3c34c72cca183e3850bf96d2bc0d69bc2943c70dbd9eac34f039b24", +"e99cf8bba2fdf8ce10eae783454521c68c5265525f9c77bc40de4a690820486d", +"44d974b070ae95c982f4a9cdc72639afe6b876b7b72ebdd33dd58c8080166665", +"eeba0bc4b12d6cecbb4854f89eb3822477e72690682267f85aa2afd34f13dccc", +"bcdbefd6d3f9bb6c3b6af601d1214facf3610e99a18b703403d1ebfd14aff2f4", +"e820ffe1e2a62a41c1dfc0bbd53a644e72eb2c5a70626e51e2487b9bac893646", +"d88c1ed915ddef523210e81f065e497c043d032dbc137cb64db8b69cd6ac73c8", +"a68efc2d414d79d0a9b6109cbbaccf508a14fa3fab9a1f41ec637e5124ee9ea6", +"3fbb7c4e1aefc516b20d5332458d3c8acce8d43de26d84ee7bdd0d8a3c0f1004", +"4211fd895ea240a1a3c0ecbfd591b2b4627459d34c358fb3b10284eddf8e605f", +"8db8d1a0db74f1b6688942f8fd627e8072bba21cf4848648b8e3efb238baaf86", +"25436f14237738a6be29bc161d7d6c077ac08c39f28686817e3867e1bd4ce765", +"c778381db5e7b73b2ab3b7fa09325ced469943334ae4e84cd3e561c1afb72ae4", +"5825095ea9ca0a8436e2a7ac856c79278677bc0e9cfffb55c33b5bccb31af272", +"8f8fd499397a381b3e8c53f6073c80e1600143d189528e774d222dfeb871aacb", +"d49b262b12e089d00ae2ced3c5674ae9df1c5067e35404239ba5a73fcc4ca1ae", +"caa02641010f59fb8593e2ed0f770a89e3387029f63c0a9c4ead1af94fdeefe4", +"91a6443f6a736190951fca5f8b9f3e53a5a43155aa46a9ffa876eb19e6e81769", +"ca87e98d9e805bfff3dc6551e69e3053ed19fa7210356edfb92e575a2af5a501", +"b8e31441067386e00dcf4228c349d0b330b0a763229ffe12967783c3e7fa68f7", +"446d0001b6ddeacea49c54e00d736faf87563da2ed2f138d47069ad62ea60ae9", +"4d5a445575df12be0fe25e5ddf5fabf17eac6fdcfbd717cf0a6938d6064b5783", +"55df19ac1383afdd44259c75d24c9110096df6296c25c324a182c7cd8245bfa2", +"a71c80c2b4a5d6d1915e353c15f15010819f7d3ecca93df6d4cb0716cc527335", +"1e6a9729a9afa2bebc71992dc2a75cc86333a7d64282f225ee9249bf7c97d483", +"40d41a5ae61f72bca0c87c5c54212b2af41bbdcdc300a768fa551135eca38cf0", +"0fbfd137a64949ea2564fdc4dd3fcdedcf5fd35566495cf6a3d88b4e56a0b992", +"ceb86ea9ec7653638ba175b0fedc9290463fa65aefb7e776eb79de1ea44b4eb8", +"d41d8cd8e328447860e92ae464e47a58f51c8ea2d8bdaa238d91b64fc88f3946", +"1b8e645157cc0a244667ee97c0132758e9127fcd9ccf89b22d3cc538b48b746c", +"60e41a641ccb5ea996b71d29c6c4ed83646050c3c5f2cd9dcf7dfeb50ba6ce96", +"12a32fbb6f1d8785290fca81c9479b351fa8df646dfa4072fc91e5cec339de14", +"4bc8a6bbd9f01b7c20fb59adc81352ff087ff7dbc52efb3cc55f188edebf47d8", +"944982f057f7a2c88b54ebf5956f587388d1350ae295b4df480ac39cc2aef7e3", +"1f2b56a48220656aaa156a87b8b8b563aa13fe8d241a36fcc0285b4474cb7dc9", +"8189f519ddf096e77bcb5f864956514dd8e5d13c222d2f81655eff239dba0c47", +"3e69b5bde7413bccdfebf6f0062d25d1dae155121945b172324ad520ac0018f6", +"2689762a4abf0cfbb31949d364eed11bb0151c37e1f13be424b70bc0ccbeb664", +"1ddae77e15cdeb0eb24af3eb70eb0e4e69ef93f16ed26869b716569b6adf1b8c", +"59e09faf72796071cef91de471f2b1e50945b798ee76085efc08aad6834b9265", +"8a38113689cff9e66c956ca46c15116e328797f826235ffdf6fab271b223059f", +"0982735aa361b160387df91ce0616bc8507e56bc435d17c7faa03aa47bd69c26", +"068980d8db8c5d9db05926148567c49aad76b6a45c21f65c6bbb258d2b5c0dd2", +"fc5841a8db23fc57de8918aebe5680a4850f92249a837855c2d72c14767f929f", +"7a4c4d3611c1b2d8cc0fcf5b889c64fd95d2d63d65e3ee8786ccd7988cd8d5a7", +"507290b2ef0a2a0eb91cd35d6d7900a3aed5d079679853238e1c41526ddbae5b", +"490d513d1f9a0af03cf0645239ed7be9809ba80c1648716aa1a632d0b30106e1", +"386e88189bcd8bb3baa7cb96b82ae5f7684b8ca4ae28fc81cf285f27516696a8", +"3c9ff8b37e02d2adefacad4c319231203f6549f75dd0bd605d432a6a47d57922", +"fd2a9951cff7a9c0c80433c2c420a19eb628ae7bf1d63a13a00c64798144473a", +"a4321af89ddc566b7c0cfdd53202ad13795809ceada3a7df41dc1f45d912517e", +"e0cdef7bf9e2f2903dd780aa53e54ca3004cdcee04dabe063be898ce8f541fea", +"6b6f00940c2048e1087a6e3954929f5bae1be3e74af0e4d67c16c7645a588105", +"a86b42c3fe5e4d6a2e35d49a64d539edab0b85aa327c66928ed6a6b902167ee8", +"9d83a58b181ba41c9c448ce17f3523faf7aae2e4d7903256190ee6d0d8e49ac5", +"147e170895a5ec27a86805396aaf3d02b2854217910236a5954293a6e87bb807", +"7781e60cf6ad4305526988985fc285e0d7b1d3fc2b77245ecdde0b9348574acf", +"45c480318996cfd129644fd6273ead5366176269e947087e18473b7336c2f2e7", +"afd30f1579fd265793c001559136529ff3e446c7a2d608b135baa48eacb81765", +"f594229743e1ca9d4a8b2a876957cf8f01e0f371efd614f13c310fb678c15c3e", +"8084ec51fa72904745f19e3f6b71486980b8ebb020085e064cf74eb21a294b12", +"5186b4db47ef4f531dad78d576b9fe473379346f280deffdf9989fc0b579a806", +"eab8db7690440407dc4799da14c672db575b138b74f7f398a52a6d84ea6d94d5", +"1c4642de078f3c01134e4abe61feb606c06dc112322521c830b88d251144dee1", +"cfb81afba1a930f28497bf4b6168868612c220b0eb356ef17243f0f522af2b8d", +"b2360a80475d21fe84ee6f25489fc57176a5f39c562468a7dc1984a51e7a3a1a", +"9e1ed016a3b9fb8be4ac50b5a43cb8d13116140d634121595a926a9279f819df", +"9a740dad1f480f954c7f9e0f06763aa9a2ede00e155f671c44c5ae0e2766b1ed", +"110671802652146bf22e441f041496b2a73501baadb4e3f422aa12d0d9cd03f7", +"09f0e624f1642ac6fe58cb277e7abeb5d12d48f28e6c24accdf31962334d872d", +"0c261e98d0e2ad51db8153439d58fccc34cb886cf462262407c56b2596925f27", +"6e8cc4b84cab9df3ab3e1a1a0412b7f2e8a41cf7637961a2c33edbcfead9d894", +"7e4343f658dccd29fcd344bacd9a1d4668daa639bae1bdcfd59329fb0443ae6f", +"748d8d69757b70d87904b1744e8885b38f48729547cfcdacce5a649533ddf281", +"bc26dbc2db2983ff0f029be6e3e1015ef4c56081bcf4d4162cd1c17c7a2521ca", +"da48379cb93e3cf8695259e32de3627841c5658e470b543f016758d27a351fe7", +"8db84f0ea2a6ebf26bbab4bbe685cd6bea94cb0d03af70dea5486ef8bfa40d5c", +"f37a9bb55b4da0a3cf77c36c860d750513091e3367263149f433686fe9e12211", +"f29d78e11b4969f83bc4c3d1a9242187a38afd67181fcabcf366bae5d464c6e6", +"b5ba4a68fccf0d5cd7e33baeb1a973d353318e534bb55c39f50455bebe7e7222", +"d590fc1919a3cc857b7d5ced21ff20f22dde9dc996cbb33d404d69bd8b26405c", +"6fe869b17d83953157671c1a6025ee6f326d7f8f3f7f1bb08140c6a018ec6c60", +"770f8d4518c1adb99b37fec51bd5eded8413fc69dd55edee09d5269400ff424b", +"a703509738df47bf2ebd5b0d202ba6fd1f33ebad70458c49d2f9b9a293d525c6", +"3cea4868138144a2f12b22c46ccaa6c83225bf3aaf54ec671aa5cbd3c33a7916", +"144b35b0c2efb1532f3d05b39c2a11faaef5983210c9b233be3459bd6c03ac45", +"4d28f92400c2ea1b5d126774c239312cd7e0710b8868bda36478149bedd6b643", +"249cc5a1c6053da8586dc854dd4d9adde1b0a7186991022bff7fd1867dfde4ec", +"8a160b1143d1c2d99e83c011c8faf7474e6b796a7fd3d28544ec086ac9d29854", +"1a5885d1f39100f12e8646d096426335f8e76fed0cf63dad0f95642f9b3cbbb1", +"37dab158f1d82a096b37a38698be42466cf4c342a9bd336a3399b19e591c7878", +"474b515155bdd00ef7f4e4e025825e6f9fb8fc0a26e24d1043d071807a847a68", +"9cf4b1c1f59ee56148b5254972270169a10855bf63d6db39e784777541ce700a", +"6ab48f7739ebf60ecc92f9e794b19afde47e11e4148fe324fdec500d8d397c8c", +"2dade2b3345262bf1acca853438039d2bc114e9f65b3105cc2255a29a912c9cc", +"0286c685b75e4a1baa2b73e10af3b4d347c74c348c582ec347928c207f2cdaa4", +"1d2a7c42e84b846d2126d2959676c5b65b276cc20c1bd12577e2004eb612912b", +"3af7826badec8313fdfbde8db845f94197a117b63c61bc083db1889cf512d375", +"1f59003c3ddbc2fb750592acce2ee0a1cad08007603899b32e77afa21167e071", +"a157a3732db62961e517cea0ac66d06dc22361d0ed59b281ca89f4e96eb28407", +"8678fcd81de7971e0daa5bf9e4bb772a22d545f936d5cbf2d337521372f7aa1e", +"5589729fb1e1643a57acdee33090e0dc320bdc66939d0ee001532cc4f0e5bf15", +"d131366b958ffdeb318f2c394068d6e8ae648fb2b996192f15caea73cad9ec4d", +"0b6dc730ba740591cbbdf981a1d8784b7da041773785dd31fe2fcdedf0f10121", +"65c18d1223a4bb411c77ce269850f1ed098737f4327fa196a2755e6c00ab6de5", +"c4dd77115e93f4c078de197d0e60f8d9897978e0484fbab085ba3e9720ad2f94", +"c8a181085631b7452a36ccdf88980cd581d83ecf1575e5307544ac8f4032836c", +"93cfbd9bf9359b18fa3b3ec84c33f08704998a205530275f903e1c5eb9f5f590", +"c2c976499bf3dff472da9252e990e49a00c66d2277eaeda63d792d4d24863638", +"849081d472ea95c0b3891a8ec3ab3191e25b595807b272092a324b09c26fb0da", +"7bae7db2d1505e63a559802cfac67142e16301a1b4c1f928a00545217357a5eb", +"98425e9199000c0dfb177e3f0d718923763ebd4807e772eca895bc6768b72a35", +"95a91c1545c8511270f94ba25a3dc83a1dc96fe30e8aeb431d035dc37172d01f", +"2a271beb48576c7348841af80e8e112bc91cc4c2e2dca1366d1220a50ff1fff6", +"af1694a1d915072daf69e306dca383f40dd1e999a258952301b1990d19802f5a", +"29b4f96ec2fb7f49cb1f234994f924906bee5d551d353c3612ff306e880a67c8", +"f109b76ac6f1ef52e2d0ca3e1c408a74643729cf41e294f87f1bbb285055296b", +"f7a6a7d4f3cca0b9a28de0745d567c0f76a0d197e6e0e9cf038f88d894dbc42e", +"929ef760d05cc55d231f090a215e1a783abc00a9115d1708e3fd1aacccd5d5be", +"e01ae1fe04a583d8d32589980d1e40414a1c402faa905bc9b493749df4c07d1a", +"99d1c8f1a8c0f0d45fea4c457ee07241c4d7f62e1c96113873ecb1ddda7f8523", +"1a24dac1b6a224ded2160bc2fa2d6fcf6bc939b6733456bcc533cd7379205eb0", +"6fd2ce6b7c4040b52535a22d19ef2cc0ae21ddadce8c80b50fb1fac3ef58115b", +"47190569c5c18bc7851f52af1634de62110c9bf7d9d00aa32f9ccb749d96b427", +"fff86c6e887363c826f0902ec5695a5fd9474bf3a094f75f3f2f63c35bfbc7dc", +"93cf1f595ac8d95cbb7924e7999268e25fde9125e6e3b2cb547c9b74b121bceb", +"e804db92ce9e9b3ab55b0baa9190ebfd73156d970111fc592e90cb76282d5486", +"4411171fbbe196094e60c70589d8212c8a03f8678d1283d428f0ec49bacd9233", +"6bcb51e4a8c45b1a599ffd146f7ca52c54578be305dad1ff96efe4503433fdf0", +"88a3f0651f201d9e9973a45412f002b085e499c52c5394710ae8591ea1ca21c2", +"e4fe49fac453c21f32723fcc5ac066553ceb0d6eccc03794a0985e3ac31c5702", +"1240d97948c9949909704dbb5386b5086dc7e25b72116ca698c9d11afbb4ea1a", +"02d976efd3f618743e82e3415d68de1e6578cf4a152f9f53d1bce5911c92f98d", +"63ea086abb5c4fc1ce57eab83104666136b29f41c7b454d12fe6eadaa365371a", +"f8783e0f2d218808b07c3ddb3e1f2408511cbd13e4135c2a0f09cfcd6c14fb42", +"64a63faef6e151450f59aa47a30a8730a9570ac95705b980e3eae79e884e415a", +"d6b5c60b4283893dc26fe1f38acf5eff2c9bdafdfdfa9513f1ae468707a9ecd2", +"f7cf41b782135d80d7add1da44b87a2f05f7b8c3122748bcd4db8a4abbe8c191", +"e9c3e6b0047d3cd30819dbb495a29603cca9ffb00edfd960550efdb913282bd0", +"e7862066cab5d95d5353146a2a7b0ea692ca9f27c661dcc4034fec17b72587f0", +"f17e20b36bc5ce9720d0d939c4362743ba45c4555833b23f035063acdcd096be", +"e0826a868c5d4bd27d7f408748f41a0a3418866b71a3d1b7b2fb01093a1934f4", +"7778c8aad8abce70ceb03aea8b2964a9436c1aa04d0157e2324e579fe25175b6", +"73b6bf9db002cfd702a7c796dc91bb1384aec9f89c89d5180317e06f80056164", +"4e4b14c044f82356e5255cf403e06d94180546af8606e5bd1af4738d3f0ab7c3", +"e01d7bad8125f1a466528e407302551465f8bc50693f609645047afdc5700e78", +"34c982d6d53d4d56f5491e9ea3902a0e2294ce117a3f8d4bbc89c219aea39af7", +"88d8be2a4a31a7912d67db02fe61abf8cf800d7a01837f9b9c8836baeda2119b", +"026868242be9f0f172ab5035877b16ed769026be654bc9f157a739dd974c55d2", +"fdb32dcbdbcf7306bc0c7b8b8c35ecdbfbfbddf5236fb740e7a6fd540dba4504", +"5e4264597a493f7a3ec9c66f6d4f820d0b3b5d645f7a6d94051c72266db25bae", +"0ad7544dc295ccb5691ce57cf65270113f051f63b28524a0c2b48f7b1aabca61", +"c8a0ff2a90148713a01f45b568befc40925be949135db70d745a8fa00caaac6e", +"3e134001f1061aa4ff4c2ced44683ea97fe1d5fa9f092fa57b2b001ae2291ca1", +"189874bda89ed07d465fba5564c1365d859219f7491beb207d728b28519e0b0e", +"07aef103947cdc614bc11b138fffb9dcae3f20329a465734f5ab55e7441b245b", +"7fd9b54a030996c69367f9f7f10e398b583da4edcfc8a4e9318fbd4dd38f08b3", +"0b93c2ed78f1dbf73f81785e7f038fca002485ed514adabfcb7ee77464125fe5", +"b7b6a764b613f988859564876ee937b80b0f7745ea0622ef9de9c114c7e18c24", +"d8b337d44fb9ea37edf9eacd121ecc3926edb3d9818328d771433cba4e143088", +"5f46924f2e23189d6f9481e1f718bb72376577e3746509c9405f49a37497493b", +"4d5ce2c71e1cb7ae27a5c55822d9a29958fe6b6b45718394ec039b34469eb237", +"09477bacf10f3e97d2cacda589866dcb82bdad25ff8f3285d3980fbed3455fa9", +"86799ca49832bff5fffde205f976160115c9c5b8d8de9f53309565a02421ae41", +"1a01d7ed8ef48f1aca8a48404d6dfaaf1e5a6c5bdab4cab7f34a2d8279132a28", +"44e965ae771ad762dfe5aa9ed1b08a3e251ad1da7cec58c5388708637c13fbf2", +"f5e70e311d6e2e0157e887f52ddff6fb507f0940f6a4453a10a6a5fdf62aac53", +"2c84b04e33252926973923a8e4ee6cc7164a2752f391b32eac744ce36fb72e4b", +"981a0499b6befed33c25c5923614a78588c96d0dd0d0c772357ca188862656fb", +"1fd68cf514a437e11facb3b452ff45e5849881698a81ad9c24424c8a20ed48f1", +"b5982a49c140f41d4734209385d1c5bfd892c5567af814663c311f3077455697", +"b1d7afe54877157c952ec152fbc2089cf9219aab504d7b60f952280a149fd3a2", +"dafff17ea8fc52bf2c4a2e56557e881ca6ca9ed52be64c3f1c8b7fa6ff50d9ff", +"ed932efa0f7d26437f8dddfcb70e940cd9c42cdc4e852de4d7c12800128c7aa1", +"4c40f67d061c8401dae61babb0253d1435797c36ba278a8ce56d1cedbff8f557", +"4288427bcfbc9d2e5fed3b53c3d3322bef1e42652d7cf68f24a5a66bbe201728", +"fc9a3e445f2e8b1de2e0dcb45970be22b1b3e47203a86367e35eaa08f6a717da", +"2bffaf82efe6cb217c85174f22429ee1204449802bf7e190cb93bdb52509191b", +"7392268b291489de1b32569f4180960fae7099549e605b087f5cd64e026f2ee0", +"b2aefa3c6fe457b56881232ecdd6a8c13dfc04bd3d578c6fb76360ba106feb67", +"8bfb5bd6a1da16e1be0f8089c9d524ea3b5c8507c82b80242ee129d35a14658d", +"ab621def2efc9bc8f434edbb12d3ad0a791f72de8ef6d5edcfc67699d4fdc3ec", +"87dfe5e41a920bf420e7c5a1be09f054ce246b80336ad29b4ecc69c0d8dba8ce", +"8f607366d3e27a776f5b8db05c5381d909fc979bc823ca772d8a0549ac796ef8", +"61a73732df9777893a609f874bd1eda318f740312a60e49cc39abce30a996d82", +"04a81673915824914d8cfe04a6e6b3089f7a5ee7cfe1ec22da44f0d5370985f8", +"ab9e776ed77f60367eb4d7d522e15f2725e35032e1dc101763e29a9a639099aa", +"a20943d92f63d1b1dd9332d14acf8372e859477eaa1d9afd84552906edd4f3ac", +"cdb48ce1419f0ff4844347c039219f81ee34c21a8e14f5d1d8b2459114c11ea1", +"b76d3edf32e3871747d039bc486a6a8fbdd05aaaff427818905b74f6f5cd9be7", +"27cd79255fb2b51a07f435b836a15f15dbcc866055893047c1d7114ec371c181", +"2a5778f11542cefda2022698c4cb3c57274dba93510beff2cfe08b0e1bce88dc", +"223331431eca79c16086dedca44d6b40bf07dae7f92213599b1be57b4c59fc60", +"733132833664924471599ddeaf84aeab175f30ecc02c4b84413e09513f8ce8e7", +"df0246dca43cfe7a60c113e160d75cd93cb17a4abca3ce0eeb9f40048ec1d536", +"f9a4ca1fe666995e38a11042f528b963f349935d96c650785a0007ac9eb0dbd7", +"e6964bd21c9013c19ac29bfc3331334d3be4acb9758b9023ba1f672f6d9760cc", +"ba2ba50158a358a6d360e3f362846e0a6c871c026f5ffbe658bb1f0b83e0630b", +"716eafabafcb12efac2e273a98d150686bed49a733ad8ea635a1961165199ceb", +"d2ef411b8867a0c06ec1ee1fc060468072068b66cad2f37d07e1f19d3bbfc774", +"1d158dbfce3f931d82dc844bf0d5cbe44fce6ce61676990f45e143e53a684e95", +"f2c4aeb351f5329685c54e80796cde77547d0eb0b500e07a02477b6219b026fe", +"ae1a8abe5890fc0080bd4d3f53cc329bb58b73ee896ec5d2719f237cbec366dd", +"cc1abd47dde8a8e01490a0dd0f3a1ac57710c479c5f8aafd26d42c4f3f57d81c", +"7cfa5734a9409676983d459e2fd089585748e1b06d3b2576c6c69b5eb327b113", +"ca7ed12cb676e78db3a67a02991e3214d3df66e41e9d54124644d6485c5c93fa", +"1fe3d42c34d70c8f6192be9f4771061c6fde1d55a086e18184a1f8af5c0fd87c", +"b8c09ea6f8a25268b17c71c286d0325f72a63662dd71089f8819888291d4f35d", +"063739d777ae4310d036c6461c1a5b3d731c78b26dba37c06c0c6c5fa3cfde63", +"2d2975ed4c87f3cf63c27f27925560606f198a8c1a05b78ed50379134b0b37d4", +"bbed24a6ed41fe7bf757dda08b5691a851b833785dfaba2b869b448322040fb5", +"d7a2823d5ce0ab893b34ef1114ff55dcb7dd5cb77f634a4bbbdacd23dbbd0009", +"89c7aec3ae309dc64a7cafdfc32f31bd94ff3e43193f880b1582bfe1cc4cb402", +"4741b89ac8e91fd42bb655b2372f13752cb12a1c200c2366b552563f25c17c76", +"da62a7c0a5f6d2d7388e55381c6c923c01b690c5c77895022127c0426e2cc14e", +"44efce3d64e036498b309f08f7559999cee8f31dbf4a6d7f3b21dac914b69c59", +"ca1a207ee4fecac84d454812d858cede75ded3bb5c534a5d10d6cd22c68176cd", +"4ae8009bf4a46e119f91820f418940fe19fdcf7cc1bd2ab5569d08be57a8284b", +"c9a782f519986779d6f37be3d9309bda4cdfe58e12709c1b4da5e60217ea18a8", +"7618a3869f63171e9a5bd0faf2220385dedd10925e643bd5b59844cee442e169", +"4ce7091f3009fbb79ae6a32f6c800b775967dfc6098e6d7791d623a9deef46d5", +"7a110c127b59d231154223725d537ee13e658ff7141ef113a0b223cc675de9d9", +"86e4d5734655fd4b1935f6bb42111f64abdc89f1aac862db984d70ee0a42f8e3", +"a2114fbcffe22e5d81286e5c658cee6138c663cbb367fea74b44d18cc6afa790", +"772b3d57c5cca6c978150ab6309aae771bc322646272ae0a2d302c69a72c0fc8", +"3e6b78c462057f8c89c2e54adbc2be1f11536e5889d97b15d465a2b708cf7033", +"aae960e7625d443cbdf267ad62806ba2658d957b12fbc2a24ee9e29f87f07984", +"36792858fdc8f06037ff810c96d9b27de081bf5b2586bed7490ef219b04f265e", +"a2288459fb85ba2b6c78eafb25d788b1bcf5f0602efdbb61190ddcd5b59aa603", +"abaddf4531d841f38543b3acc2ff01b48f9f5d00b4ec9c2d253349570d09ac72", +"ce2e97ae718f6c6b5cb537b99b511ab73f5577cb52ace204ed57c5971143b441", +"51591ff11f01bce595cc72478a78306b38371cb4875c6b0d33686ca1a8a52393", +"6cfc2c7b402ec7b9ece8568cec36478f1ae1c0ce43d279ff1b4b9b1365a912d6", +"0ebfb0ad538f162558c170e95e72f357dc1c8f4b90ba12b54ffe397fb7124c9c", +"dec9cdc6785865de9f5cb82e625f6601af7440d5f539443791d661bc2cc3e5bd", +"b90b1cccb6996b94c914e9aaa672393aa17309a77b77b1500fe92487b6c0da77", +"7431a716e6eefd4b4d5a3b7536f1208974e64bc785c8e66a8de6b7db5e13df8c", +"f080e33f6c5dae599da7e10b8b2c14b328a9af3f1092f2699373c8de5a4c0e74", +"68890c9b9dc3efedb81b3d2c5d4a66c8adfc0348bc567c724a3a27d34505af72", +"7a5c400f951400f682eafec5246ea878f7a806d4102e3e20839ebc819a63db1e", +"3130e1de4fa47a4bb3976388e0ad80d5d2e95ebf1de12a2a3484322aaed6df20", +"133e0b0bd67b98f2ef8fc13d36b768ccc316fb51710a9c8f8cdebd709d4db09e", +"385dc1a11ac5736563c40767d983e5402e2ea85032def60def5e9d0476d48b00", +"687778870d188a7974bb41feb6bd0c670e5340b2e2f6378395c1b9ed92e174a9", +"73a53b0b15aba77f314eee518590c837559a0393b52a09389124f1e0fa6b2e5f", +"099c975d37f1c9422e16db0e4d2903717508a550bc54ad27be2713d8d74e276a", +"64b0cc9f781ce022975807938574ebcf9e45d07074b1c93e3b29e1261c5ab6f7", +"e97e360e51cfa9ee2fa7720c8d660faf5246a4fbb0c8d52c84e0af76b7be02c6", +"729e77471e3a745d5bc1b956b26f7c2e678823154997325d97d4d76ca1deea04", +"47fc0375071435f151eafa8db1fcba9c8fcf2e1aba49c9d2a1395f157d7f2d32", +"3a8106f7e4d9a3d49a07a8263766db69393f8041e380c584fae19aae653b3e5c", +"96cd7e92a6d6ff3011a2147c17562cd2de5c98e619e2b08865ac8df3b7d4180d", +"18849eb8b958a693100b2a6821b4f82b8b341c38c9f8d5cd50fa2a3f89ea7bcf", +"c780011059e6fdaa7e892bcdfbb0cfb884e704d549c3394eb1cb2b753240df92", +"3adb97c9c9ec4c8acf7d9c0991716c7eb25437b9cafbfb5b57feecb400ed7c23", +"45f41535db18427bbebbab0a124ccaa4b688f553052001d08945ed8738b03039", +"ac96b39e1dc5f539032ab0ea137f9c9f923a9f04d4195174171e692a9d3222a0", +"aeb6933a55bb73b1624ef6bec607e3c393b12649b549b20cd19718c6ce11e940", +"7872988f8759f214d9f93e81b7b6b494247637a325a0d891625c3947cac82013", +"f45b7b27d906506410048cdd20fc71cd0abf45af739bb3986ee7dc28e2cd1502", +"aaad6079a9ebc6031ce76d89f170c8fc76a825d9a318696dcd028d5070929db3", +"d23c777d0f6b79ea799659345844b116cb06f43b5d8b88af330a10cccb374417", +"701cca74fa1dae963b0949dc583e4970bad2f27f5618a09e700b763f822819bc", +"f6184c9f5a5c919956ac39d4ad05118dd219446b056dd0dcae51730bbaacff4e", +"34af8bd6d8ffcce846918a73f22f497645eadb0ac708b912c8a189ea959693a2", +"7dce1fa10c9300c532ad61a3b7c57f5193220f2559215e2fc5b7f64c77503d27", +"c21efd335d60ee1ca25cb7fc5581e77c2a595310af20beffda320daec33049c1", +"67d917f49aeb5d9985b9f7bbb376185e9f208e3062b77cc9a2545ed27f779b75", +"16eff1df7b6dbbf538c53d2862c848da9693fe29166753ea1f32d25544b6d86b", +"01ce47fddb3920c763b17ab0a5aa17ba4ead8d651a4b97461dfbc5afa0958c3a", +"82fa56e84ec7e396f97806f0617814b2da473e6475be1d93cefa5216ebb473bf", +"10e5640392798bc6067e23dde163b66165b9e22663886bb43e295fa2886e60b6", +"3494abd89d456a9228a060af6b000ef9591b473b895c83929b1088915dc42423", +"1ec0199e01f4d9300bc2367af6f1a2bfe40f32239239842bfb98970f5fb599d5", +"9871c8c4b70f3c8a2167a56dcb70132d058f77460bcf36d8cd154622c92cc38d", +"82fb88c35a2dee24d11109ecfa65b9e3ad6e10d37f16538c3a87cecf69d5ceec", +"a12847e0b9ac4476d831d8fe2d93af81c832eb575ca3b24634748156a3275fd0", +"c6a868e0d22945bf46ecb0df469c11a30485da01f6dea71669adbb95a9b3f8bc", +"423c61f1ec58b9400a373e1052c26debdd5d55eb0057900c3dcf41f54762486c", +"7cb995e1362c2bb21640c886e36641c907c14ccafd84f307a1f74b0eeb782870", +"79c0002d3dfbce81fa061affe5bca46e46768bc26ea3b8d1249107cee4669148", +"f74f06dd877c776a93b14c325337f439279b23ecb982581b0ce6c7c7a294daad", +"af3df9868b72a494f47c4160447f86cedcc7c184d75b0744e222c242d73fc595", +"3b87cba950eb3c08a2e394aaecbb304f4bee5125bd2fb099449c0803629be6b7", +"abf53391b8ae0772fa91df7a3889b853db596251518ef385632332bbcefad1a8", +"41137f7c7f91592ee93d10da9ed9b848ed65e4f5ac948fc2d806f915bde5cc4f", +"5eb86e817c483c9cab68faf228a9cb1d7566ba573f32d5d5e01b2f2ca42d79cb", +"1b8effb13d53c175ba61d78f654c503580690904e429f29c4a8f1df551c86a90", +"424f6d87bfb338e5f0f6a26cc7c025c2b7b2eb4dd5f8d7c46d4a7b3b5251c60a", +"710ffa6ffd1296d9957c943841616cd3eab1dbd853c4727b01e5c1b0c2049bf6", +"20559e49311ac1bc57e553216cb7ee53865d4b2f4416d26e29cad3914d54100f", +"02c5e3497f4e7139a92581f1f81c10b76cef2d2f0b5afaafb4503ae3a876964f", +"ba4a2aee3d05fb1ee2977a42f6fddc33c9e39f1fcb442226ccea1e6ca561b617", +"aaa1909229eaf73f7c2be639c7d3e8c774664c64a61a2741bf327bf00c81bcd3", +"2b7222221ef1f49b38f0b7d5ce04a312dd4c56492ef4ca45fdaa57f419a4f2d4", +"69c001ce6b95e03c09ccb66016c70d8113439d1c0cf5b1d478591146ca454255", +"e1a9432c61579368568dc1b476cb15c0f57f2605b53ddaad08240190553ae1a9", +"53cec7ba2dc54fa46466c0d6c55a5869851b5a8617e4e21962f219b1956c0faa", +"f1e96c924b3fec22370fff308268f39acd3a0891cbb1d6f810a28d7e583e0f8a", +"b4d5b929508742e1b1d80ec182af65aa9b3f824752c719f7f868ed122ce9a671", +"ff781f5ccdff7a07a3edb2bb9cc262183bd0750b67d1cbd8233425299482974c", +"a7f3c0d3c532c18eca2aa656b6046703db5e81e8ec3983ef955877b7345374a5", +"a932b9c6eb071075818a2dd348f426a1d5250c1543094dbde1359da691bdf401", +"abed83079f3d5a5477d8156a4d94ea455cdb64eddbd93059658c42befa801079", +"8f145561b188d4c37ff7f3d9fed9469dba2db8fd2d0f47589f2134c5270fbf25", +"364b923ec382e6623c6fc551ac6a1f26d5078dfc4324805a981e3d788ef6ed03", +"cb67ec18fb933e4d9887fcfa1944fc7aab5cbf0b6d67326432569810418b9ee1", +"9f0ef938670382983762a7859641054243c89257fdb5b2307784ff8e9bb55451", +"b73076a77722f691ef78f4198061dddf6572f95eaf2c45b7a724b1808bd307bd", +"089badebf6c4de916f77b325f5ed8adbe11617bf6cf635cfffdb6a25de264e1a", +"037da7389842040606552c1047bd8f583b426010206cfe52b8f1ba70e4885b7c", +"c47454f6454564cf2c6e624cec87e29f75666a814712dcaaad1d723702b1dec1", +"df8b130db7321fbb9346d6b53d7b58e9f6d06afc77de17b3757a1471ea507869", +"8c0078b31019640c61301a28dbe9931d449ae82cdc72b16f25afe678d0f7fa35", +"e19b7b9958d675ad8c6cc33d0f5c2705f6817486c17896637fa499d413ce0233", +"1efcbe883b2eeb422c2312211214fcceb255ab0e3db174399706930871e32029", +"9d82a47201944ba330a23ca21d957d6bfab1bdb9b2b7e3d1e239c3811cd0c5c6", +"ffaa91cb56f697e8c9b9134405a1f9f8d6b430617fd05ce436318ed362becbda", +"9e37bbcf0fe8b4c94fdfbffd4c2f2ff6b6978279edb71fa37d024193bdd320f1", +"3c5a36956bfcc019c207fafa4aaad947e796d932793ed299e7f4d0b5452eed9f", +"ecb4f12693b70c6f51d52ab3d4642308462da25786bcbb3d0fa44ad759dd47e4", +"7aa52b08c68ca8da95b9936bdc052619e283a834622c2eee9e6cb0aa8e97f511", +"170aa3a9bde72a9a8bf76578bc5734df72a6899667ba324ba546214057ccb8ec", +"655edd87e978182f97c9db2386d48aa284c08c17833255e1716eb0229ee953e7", +"dc630a523ddd69bba1395e8ae1e5c5e09b3bb840b732ec77cb58e0a409f723ba", +"7d10f22d2c115ac2962dd092cf457a40025c75be8fb7116c9b6a95fd46ee9342", +"1b70043130898636433abd0135213ed6891a07ebdcf405b093bfbab82bdcb692", +"ee6845b61efdf0e7cb4aef0998b116e57e294c72b72c4629846b743311d3030c", +"a951b9a6204acb6a254c328942688b006c73daf7af93d7dfe1d4c784f8a9a51e", +"3d125f02d0fdc7ed4ba7ebfa4a4133604afb514eda30723ef39a3ec9b156b69d", +"99bf982da397947eb0999292e909c17c78d884e18d3e59ac03fd2aa7f0241c7e", +"c10a00c78a678ee38cf9a38060533d7a732ce4f31565f1b211800e7e58aa0f76", +"ed6773eb8f90c35ef22f222299edfa17bcfce15a88c48f63884adf771ae49431", +"2c4cbc2059a994f5a18ba3625e5100783fcefc82b5216fe338df0c2068487d62", +"3376310225ef4ceb43aad95add0a3b5cb88685b58062394ae6bd9b0501b3abe3", +"7ecc08350e64b4952b562f739f3e33e50b13b457ddecf1615322a3a191afcba9", +"c1afa9a8884350ec36f1ad5ebb15201ccddea7f17c0240d75e49e301cef3ced0", +"da34d3d8cc0268ab89016204d52f4d556510de4f136f27e67b01cd9f782adb47", +"ad01dab939f86eda47450d5e9c751ae42d3c4b07d66edfe21c86bb0c72aab84b", +"379e4ea29f4caabcf6f4337936f1c6ec8bc87109ffa667ea4eaad64db0930a2f", +"39a3a148ecf1fb9d02871e03e747f24dbe7da6b48297f6cb2729753250a5311f", +"2ec22427d614f7ecedd1dd2efdd02591e2e8e4dfcae8e213f77dac362707bae7", +"c2d781875f992eac370ce0faa9e606d8d0a53930f3d30e5ebc6edda406597d4e", +"7c5d8479a6c7924c1ca56ed949fcd1943a83c100bbf1282690997008779fbfb5", +"e6168843909f84c7a2cf6dccdc0a0b4670a83d6b440a5434d2e01aa9af41f61d", +"2c53b059277db425a59d965ced106eab6de497c3a341d3e3fad9e5e335ca7d30", +"4e2c428da8f881ba91fb99ac07a39cf5cc407ab52a91e541e0f1ea6c26518235", +"cc064089fac9282110c20ff014c0bb53299c77a5c93456c3d6671882cc7af79b", +"e5dded042c0ecab6afeefcfc21246ddf69fbe3a6ece49dec01f3f047f03c77a2", +"89d5d85cc435eac6031a322cdf84c458bbc958ed4a911b062027a1d26be5f545", +"7fa7ebfbfe8d79a68ff506c27ded2f25d329e1edc76b2cf6cab5ae8b23a8fb56", +"5f3e7071a8423a64f19af140991705a8284eab5ee0f514bc006f823a49d9b5f5", +"45de832c5d70ec379aada04dd165d1c548ec9279d3fc35c5af4bbd22e78935c3", +"8755051ccdeb2ff4a537fbc1c37dadcc949777914f7f14af2ea9f2c5abaa6cf8", +"e2cd5463424e7c2ffd6cec1c81621d20bfe1b7741ff1548574ac299b2e65eba1", +"a7375c8bd1d8b89f4b7cf6587583358f2f94e865d05e5ba61fce8e00af28dd5b", +"ffd7a21be0788abde5889973eb6239cc6d1c9fcb2ac34cf0143c5d758e67648b", +"a03704213d72284c28655a19d4613f8eb7b49d5ddda331a934eef5db3ce9f3f4", +"2debbf2d7ebdbe2714509a0b18f37c5265e8dde83ffc3b3b7d541467b87647b2", +"a3db1339103bfc259b5ef11a65b5670b8954b2d6d8196c4e6f1804575e39d976", +"13e501ef704a398aef7fd17163161176b18b647a36515f02d7e71b90f39403ad", +"3451a4ede401fbd7e6a506362d7160802e28bd923a05efd1cb46c888e526e789", +"e04bbaeef4aa47a881cfaa7bb09f1d7258481bb001098737da165b47855932b4", +"12a933900479150cf873a4e77ed04015b46aad15374c4882d070914739d92639", +"f67d7fba4561f0a4e49260a060d37b1e10d10a6440e463c0c2d69cbc7ec0ecb9", +"c36406c941e49729f9be80bc68bbb34a61c9d7d5f341b8102e1f006c94f25891", +"2375de1b55ac345a646e0658802a72455225aa7b400e09b93b2201a77fe3a40e", +"96b15da74aa86e8993ed0512bcc901d58ad37f961691131e910c129770699cdc", +"b800170293eb62d82d971a494d723b31615691d78b379a35589feeafeed9f095", +"39901295aed33206b618dbe15dc11caa383b499ef9b8144c69f393558ec81744", +"d77af91f7c5fd7197936600222767c29ce3a09d72882e0feff808a01095adafb", +"ae9e7012bea2ea6fdcd7baa078503fd03afd96d22dcb7f0a88c1655f8288ac7d", +"7ce54050e29c43204e51b0660a21668683024edabcc8e0ccd985d9373a70d3aa", +"14cdb7403bdab40b2512c5b43f510b763fee6d8efc3f3d52efbd43f862c3e270", +"94f50caa126f720468e8e0d3b6058451c9924c80a6b3169e4923d974c511ea79", +"58c2fcbb1ca9582314979cec3d314e5acb494e49ea42c55967121f85ef009252", +"5aeba9bb553914e2d06b4f5230354b04167c01c41129c4dbdcb23f5643277ffa", +"b1d5739621347e154c3b42144cf92353b6589079ba94d8d2ef2afe23d9832c89", +"92df326014cd3bbe09b8394ba3cd6ba71b96826610015a9df9080dca02519bce", +"a26cbd16957e7bf2003f239c83c1a0446b65d05d1f4b537f062839887e424403", +"c2e57ba56c8760b7c53a839ed577635c21f3de2d601a48d034d6b55240b7ab26", +"2f7a3c0554e22377ef66013dd3e2617b04ce637c94e22ddf203657e7d0ba6640", +"fb1759f381a5f5eebcd590efc472e47c63b72ad8e5a503a9eb5fa47e6edf1aa3", +"6998f17a59cd28354f6822549a49bd4ffd884b0164165a38a796b167edbfbaaf", +"80603641d4ecc43492a3195dc8d2efc326c9d2eb52718bc012246c5f139ac214", +"e072d1dc0f1806d2149fd9e567e4ba6e0408b76e62e46b38b3543bdc3880c438", +"a3f2555889532f1bf1e71502605a316895b2f51314ec7ed6606df58cb1d2b5e5", +"c85059dcf5e45271e991ef2150c08e202cfe5d27a6bb81ab74f40dbd2e041af8", +"09c19bc4d2bdaf8014eda9d5f4f688a1ec7945337028c3553238a8f20c0a5650", +"ccaa887b0b663a0e894790351accbfa84c871a5bc66c2fd8e4d87813baa7694c", +"d1eb125bda8110e193cfc13d2d25bbe9eb65dbc425b43c9a46fc6cb2b287d577", +"adb31aedf7ae61b6ca25b063495951bd41b40039fc521bff42df6611dd906123", +"38d5382e11cf3ad6476b7451cd501ebee0539453d2808f0cb76f5637da63074f", +"e4446d9caa9e7af06db07bf83e81b96c8e783c48d0653664f47043e0387c8be2", +"f54cd625fc692556a3f8aee08387a851c153b8aad36ac440eb54e7c62d9ab800", +"b96b044f5062673e302050fd72f7a6e18bddfb71997dc5ee39a2d7af72afe432", +"98fa8d47107c569cc19af8bba0c6d8addb0b86489a8b6feb4b4e71906d0400d3", +"7f053ea1bb326fcb1a68239eb0d312feef64dd166cfb75928ed8abb294827d9e", +"0c41adc94ce8806bbbb55b7461f9fa56de5895d32e0572dc9a5818d0984d1b18", +"5c87e98e4541a0b55e3d0f1cb1831b93e003cf35063ade92502e6379b53c0fbd", +"71af4d6665d594e5892fedea2be709001cb1998ab8363c81c0ab01638690f8ec", +"f4b01f3682ac126ad39c860d4841452140261bd810b18409cecde695aa96ad79", +"4b03ad1a2c54efd56f1f3e9dda5196d9f4ed39b79a4284aa2218c569744ab9c8", +"595a068b62fc00db961dfec8e2d6f7d4c7f5aeb4820db83ccb13636a6ff01f14", +"61bc90870c0728011212fea446e91e3da2cec07f896190c10751856344e947ca", +"64b1c8c202afc370fe9098d1f355e180afdcd39eebcdd6ffa10f16b31fca19d2", +"0f5a9d1884ad420d48bd6a767c5a5cf16db2431d2c728407191a02d4ea466017", +"ebde966b607e1781e6009096cee041690482c3cbf99c775a1eb06fd31c52914b", +"a216f4df946f27235301b2e3febbdc989dd99071da245026f359bd7e4b59cdc9", +"7a23806a80c95a7b346dbef40b9426409b2ccc7b6ef300cd2e21fb2681b136b8", +"88156ea569541b40115cd983d9f4f91ee92160f8d5e7c33aee0179f8a84649cb", +"28a93e8d422c208429df5443c3f3f536c44cc93677a99759b55c03a62711e468", +"9b03b5b3a641f8c91bd69ef384abbd7fc1ea758f6cf7971dce353ed8240326e7", +"9db249840dbb2fdd6c569003737bd907d36b84a5c900484d16801fe467998ca7", +"70ed9ccbc66071fbf415b88502031577444c7aa7d53e82a0361dafb776112062", +"7cdf92543f32609c353e765b012db568d05c31144d6c4964b4a713ad7bd0dcd8", +"fbc12bc889fb3d8150fde2311baabfee3fb762da5021d852fcffe7081899f0e0", +"0b77740efdea460b07d98b09d54593f1f51742d7e089287d10fbd28f99cebe61", +"1972942d34f8ab8bcf1cf32ef039a82f3c64c3705d88839b1407584d203c2b49", +"26ca0a1968cf4bc9bb0c0283bab5e8555f3f8d2a537302f24b48db8615ece41d", +"ed47d52b140e51ff466a4a193d4d4d27946479b48f9db2802c3840709108d0dc", +"e7bf7acbc5660c6c536de5420c69f61c47390046cface36c0b2fead9e5758aec", +"874debde408dd3f22bfb8145218c00df618dc688bef3f5c04348a0fb9d86b357", +"019d8f9a54caf54df2d99cd42daf2a003a6306d2800b50ac23d6ba7caadfa41b", +"8d1efe2d4cb191ebd7f08a76188772d230fefc009344eaf7844309f980dc491b", +"52a86c7cf89fed89735ce8da32b0c6048e263b9d5ae5a85ab19a5a00c77ba59b", +"df189bace51935d6b17035b1dc15ced2354cee565fdcdb115c6acfe479b491de", +"e9f45c66b784104a90ae947dfc601c6551eee7809138ae9c937d682c6195e8cf", +"f02383579b57ed826bf8c736bcf9580ddaf8bcda9ffed72854aa4d46ab6301c5", +"17200dad22c5279c536776f132ea3048d7cde91f74e35fdef34a6598d10ddbb5", +"32ead108423bcd3a15975e0627f73a67f8c69136cfd89ca3f7776b17252b1c2c", +"53dfc35d305bb6de8636daf26cf167f1fa83649d4ff7da82ad68b3e137284818", +"d1842809448df121ea395159cff656150fdff6dd54428874f7712eda13897523", +"908f32cf4db4d6f57cb4648627b7e0fdd2fb570ee190febb1e55c5dc951ec361", +"5cb021f8f6d2d87b5adbcda6b7a94f94384c24430503f54e6e93eee092d5d78d", +"acc8330415fc7314119d5f3e968db4ba1cd94fd7621baf51903d4bcbbbafa8df", +"35600d15fb97f1b630ca7b5edd18169949458ff4d2b7b5eb092e6fcb0ee5833f", +"16718bad1cbc744cf045988158af9e2ffa02eded398cc08b201f6a39a8a20f51", +"4028ec7f3f6e3757d7a34ebed1900a1ffbfe037cb412584d8b9fda6885839933", +"46249da9e3ad06203a617269ee6ef4713629e9b46a8c004a576b761952735a99", +"d2c58b18c7f724fc92031de6869dfce8787b6330e4ebc9c01856b977eb6f33c5", +"6d30507262e858d7416dd6378d8af77c1948f0b411bce1a268dcd34300f4ec9b", +"31f78a8895f98e876174a147a55c5414eacf7cd3bfd890d414e1d1da8c27fb39", +"0824dab906585a541bdf7c03c89216cf208f294fccfadef718940be43e8fdb61", +"f69f1db424770f4c0968895cce8e672dc25e987d402138ba4887b083ab821247", +"29ec90f84476b22068b63ed8d82f2ee8597b9d59af418f52ed137fca3a606807", +"a440afab4e4bc037b920bc3b3d63814413a0bdbaf2632a90abdb1d3b34d857e1", +"b166435d26e62b7e9bdaa5ba45c861e810d8db63cbafd9f2ac02514dea4d426d", +"46af53efeedb1cd366dc8d031baef9f893b2f27745ba12096c8d18288486ffeb", +"934c00af139fb1784417969cb765dd19640814270530dd0a21ed88450f041fb0", +"2aa77a090823b10d01e1023ff86c2577f23af0b4e10abf5a2baf069a606707c7", +"1dc51c1c5b11d02c49077676cf1f40d5574add32f18ba6aa4b5fbcda80a25fd0", +"767b85b0f4eb55d9a6d1459a63dfbe482298357d737b9e295b18b22f05951948", +"e516e6427dcaab45ccc0704c8046b878a11bdc0a86967ee40f01320b161a8d58", +"cb21bea98d630ebf1f192080d22727136237b009f85c86b9bb6f69d982e34fac", +"463618c9d140a1647c57ec49ede38ba1f9448c766c4311f4de107460e2876e74", +"033286c61659610c3606af2b93007e24ede4d64e0b0f5df99ef836b5d75d6426", +"00b0f05b0b9fa3dea1e7c74211c95467975ae8efa02f03eeecef8febdac9ad0e", +"5636a91797234a50a1c7c0c7c650a58d4f9e249020ca552a0adc3df7f1d457ae", +"5ef398cc07eb167b28560e33409c7c5492069ac38f5b0c6ae24331e74c26bda0", +"a109a097255499032f8e99b116ddd7ca99f096996446baf129f6c2df2f3f751c", +"4ab3cc4296fee78153d60d2884323a84260157db0b83a72309272f109ad9dd32", +"6d9938d20e092e89c5c8b10d9cd9f1eae24a3b780babeb0492ff0f7fc63fdac9", +"1b45f9cf8a3b4964102558caddc29ba967b36d1b13c22dd89d1f37bbf7b836c2", +"ba6406526c97985b6e4de37ddc8ab5469297ebc782aaacbca90421e5008673d0", +"2697c2453e57e8373bf02c4761e948c720858356ec239a22f28f58474de6e462", +"a64b0b9c82cb7c840d9dbf9a9d3689e566f737b09c2404eca0fcf25de22d6953", +"9909e06dc6a732e3cde85c291081a1c5957b2c9a39859abb6ca9d68c1e00ecf3", +"b125909927019728e88900edd66d5e1bdc83917b127452b64647e5d996f19a55", +"bf1090e4a89922d92224e63b2ed79b24268cf92be4a21bc6ea7bc3afc180926f", +"39b85cfc73a13323d33eb4f68685ab96143792cda55ed6aa7cdc60651da24d1e", +"f9fd902fda72b31b3a87ae2c62e4b6ee5d64aa1a1c827d78497eeb3c2e2ac8e4", +"bb3b87601a2cb367d2c70ce98552d7ceae1190be52de6afa7d6fca55d1a98e63", +"706a96519fb104c3f86b25f0f121d2e25a2c1b449dda83d91dfa32bbe6b58262", +"4985beca84b1272a67b7c2785a2d7441cece3fd6c3d7fc934e45bdf3f9825e70", +"922f47d7498e7bab40d66731e843bce78a8bbb212299a4f9b7fb8bdbc51f7d36", +"693d2d94c81610cc0811148dc54c9261ba75a5bde19a33339caf987d2106d1d2", +"7ed8b2c5407db118cd72363d965c5a09d194c3de5787a94f57d2bdb493e6cab0", +"fc392a3a4373eb3b5ec958c0638d48947cf11b0c4632da94e0e36a8993fd5dd9", +"f52db83345c2e8bbb2b5f448c574cc8e3da572f59981cafce3310b16c3c4e9ef", +"f4513f9d2e051a3dee4e4da71034c72b99d62a81a97d46aadd9be63b41ac6098", +"babb445aeb40a48ad9ce7c1e6fdf741c3d5f8ebcc5695a66d9ffeb1db4a505a8", +"8d1db6e351895158ab17d482ce481af1230b851648088dd4d291e5f0a3a7c5ac", +"dde0c65a169213903d064ca3615f34b0fe92dbf1e380d54c80e2c52df69c9c9b", +"a25fcfd3660c494a147ad37247ebf38f51ef48dcbc9fc0e60cec35256d899ba5", +"92848ce1d291c3c1aa0e5aa5b21491bfb2c5af723c176b29af98b4e4a57391ad", +"82a82a6b4dba188469243523e4a6fd024e405d66d0aeb4589663081e40a5993f", +"6ac22da0da8430f8aabe88cd3af122c4272279a792bf50bcae6961f2eb687b5d", +"afba1f2ecb46524d0dd327765370cc48131eb91803ca3e902f979fc09f5e8532", +"50506e2ebe7aa2375ac714360d57e55a16b734ec4340d738effffd641710de32", +"55e2c2b106554f42fbe3ab601b74fac94d6ea07834df928426233bbf558713b7", +"9152662afc5b0b642a85dbfddec329585aa5e65c32ff9e4e9b25e98bf241819f", +"d5cac87f4c205e89c74b3f28c0d88537a676b23bb8b32523dc58807d1446ff5f", +"abd99216aa8777cf7315f35a658adb310f4b08d46fe20571a70bc0ee91935cf0", +"1e9ede7d8d7a3d03fd080758fe7f95d77199018874f730e9f4362b262ca51855", +"15b5d8e2df59db952a68379e922a3e4c13226bbdf3266d86501a22fed8c2b1e9", +"42a334097eb92c051963ac3cf3e6e10ce648f5b2b7d3c5cb99af759ce6db8f0e", +"11aa75f3659d8dfab95b5edfc753604392f4e66772da292c681118092938ad01", +"8242101f9bd09461d4a8071eff46bd5c2853e3e4b1109ebc295703fc45bce8d3", +"0988270f512a54e00c7978aa2fe33b512f6db3744433a19e21d22c95fa7d29e6", +"5f5f41a7d1783b044ca77c75d8c6a9104a83c9c9aa9d1c5f59af8f27f3427208", +"1b892e00e8ad5ebdab8793540bd4d1dac3133b0f092e789110cc1b4b99d9ab4f", +"3678f3a22111b83e3b1fe13f3549d633c4a11042822fd741d81f0a539b892048", +"ff9358cee4f94e0b4813c4424b7fd7e5931dbe4fb447623d5a0416f01b013b52", +"2af29573578cf55118c8c6d3bac95887ef7c20a0cd0b782e608b80b92aff0ea1", +"aa86e65a5cc711b52f383d0ad8bf9bcd32d229018267c32fe82393c55a72cf64", +"d44cdde2e095ccc3d4aeb91baf6bd1cd9a9a7c0b8405f9ef036c8a7290a79af2", +"45d0b9fff907dbb9193e876877c836ea2d04ecf358f9b2ce920c78d6bd0ad61d", +"70744b835941646b11baee6a2eee6fc8d7679081647860804f51166eb9081942", +"6c82dfa980ef40dbfb69ea169f040cc084beef3a159214e2bc667b4b868b2d5f", +"ea491e45833a13d64e46e6d210e32d8a7f115e0fe8d1b2e387e6586d3048c1e4", +"c01d1cbe86db25b5b1dfcf66b1e3147485b0bb1ab0e256bf40a642469943160e", +"04032ea20e8b0594766e67a6517a247e059bcd547748fd6cd2b659c59a138f66", +"95bf117e466468f468056019c713372d9696d4e9f3a08fa75f3c1a285cf40991", +"4a29f904c36e1c49c937ffb0c5f1505488b88aa9735044406c3758165691ea48", +"92196088312828a405f41d398b7e49afaf42f46a93b710fdb83fb50d52ce70ff", +"3e6aba35d3e71b8ba7185968d56d53b60bd58159dc138b45a662aa748047eb9e", +"1f450281b03acac4ccd60e98d9573abcb4176f3126b9503a6de7706421700da3", +"1c7c9b58f1fd2dd2e3b0b7c383d8c855a479279ee0dcb52397dbf842eb4fdc52", +"d9292752dc327b8d445d4079a43ed7eacaaf3cda2cfe41ae22ef1ccfc1d1abb3", +"cf5defde2692c92c514841b231404db099ab1278e1a3b6bca120917f6a99c323", +"4d0f0aebc30fe56f95e4c601e35c4a97c190f22ae0f79b8d487e6007e0fa654c", +"3e9751556d4d6409c8ff03b3e580b5bb476feb83d9016bf6fdc9965489753c1c", +"78dcba76bcdd404f4597c6be549455acb4b12597f0675427f4da3a48ac5d7af2", +"533f8756be1d7b4e7b5a0eebb7a07b7a629a67a05b7427185cb3669a2d4078d3", +"c47329c7d747c80316273bb726e6ee87bae8fa3cf0ad22dcfd6d9af692459d07", +"9c184685c9b3dd107f3da78a1e45f11aedfa94c260ecfd6a72149ddc84973de1", +"c9e3b833d06ea15893a606063aed85321d878535f3c9587d94bbe53d7444fa1c", +"9ae37daf14317e36669e2f0cfaa1ff789c70b64f5b076c208a8bf692495bb5d9", +"44c7a128bf929ce2e116ae56559cde6ce518afef6096da9274493b2ca1e6b056", +"5950cc57aaaa91a79644f7f510339823a4b602bfba9c6640b8ff07c1a41fc06b", +"b5e9ec2fac3a9dd54e2dab079130f16466ac56cbedaa3a5e0340ae6cbdddb034", +"6c964e0afc359e9c0e71fc0805e9ab587b6b3db8773aebf1b42be95a5f9c277d", +"eb641b987daad963c609477823bed50abe7b3f5ab3ba5c4110f3f4651f300fb6", +"7a126647dc3d67f451311abbadc653150d192284744132903c0391cdf53fb775", +"91488054f9ad3cb2298acb82f976ef320e14bd14c84a882f87c3e8a84923683f", +"ffa24ec2bdb879d45b685b1e8f53d814c802cd7711d003a25b9b4628669896b8", +"77023fe69dbb31dae4bdbafde42f92d45a26370becdcd5701522928b6a73f5fe", +"346d1a913ee4cbcb0311fa8dc77e65baaf8d95249b77581e707cf6cb18e46c55", +"37afac7462c2baf81f8c19a65d5b79a8fdfda9e0e701e7a8b40e2721f1fd3e07", +"7ec2c819b82041ecee3ec8d1202e252febbca1906c8e56a90b5bc8168cd28f4f", +"2d56ac5bc7b4e799049eaa473b2dd1d07a84fa9f3b10a19f6084d8d61742145e", +"a78878c08a2d9b3ec98c3f24a6f12f006ddd218631b08b2ca87efcfb709725d3", +"e81eb803a8cb7d1cfe577b1171da0ad82ed088df7f5c8345c2d0dc2704ed14d1", +"61c9985c654a31f24ed4c1d1c7a46ec48ffe1abe40bced8ad31ff2c52f45baff", +"23042ae487a09a15e2ec0ae325959f5a302a30ceb1f998dfd82c4b7426ba0c6b", +"4268e9770e9de360f66a68388e3ad516b99fb6ecb0398fcddb404f2314c93852", +"2f7ac1a687be06cb1f44770fc24089fb71cccfd89d600324cccdee8185bb35ec", +"905323ac61e70424d7817d6eec5ed5bc69eb0c4864fcea199bfe43e92dfde7cd", +"e32529bef02a6380aff567bbbcb923f8bb39a351b541987695b03b04a3f20149", +"1fced5a3187b2195eb80c815082a7543aaff483f45b142261b9f3a2a73122911", +"6ffd58c67b58ca264e778fda96dac9eedf2c0ae7dc23f185d036f8ccb8e3624d", +"29af127ae40c801dbd710e50ad16ea39133ca90ddfa5a858bdc8c816f8c838b3", +"f6c8576710217355c82c12e3a474a0ab7424845fd97461db0a865a763e7bcdd5", +"356e79b2aa8bc3449492dca3e3767d52abe4493c93fa378740de2ec43ea077e2", +"5e6d959647e53ca568496ab87e3fea7b9a364f3544379f5b360dd6d9b575d9f3", +"f0153f258f594281958e6b0425f94ac679f55d1fd30ec729f9544dfe176bec84", +"d3c0bfd35cabdc4492bbca066b8e398e9da274d148848391346a8d87d2f1e2b3", +"cbe50735372af177bf2a1c866cee5426271e7bc77eda37e11a6521b05e5f6865", +"8a9748e009f07add863ddc530e2b9b6714ce11f5a889c4a3ccf517125b34a16d", +"4398b9f050eee9aab47ae58c18b0c2e0b9e3a9a64466a2953131f797b688a328", +"9ddd6f17198fbab21a0ba18c7cf2b5f37511a2193b018c9746618dadc42cbf90", +"32935c77dbb2bb8fa90923b3500e6cce6a8ea765a51e4deef1da48f03c90d14f", +"ded2e8a42e0043841b46b598628ee65a04121860fa485c0f64b915d5fa128073", +"c3d23b0ee4ea1b74d55784e9c3dabe3e9d50c5bc9d132bee4b01638c11e592b1", +"e9eea491bc7cf722cf834d2adf26c51b2228940280a52aba8c5a604b8cf03ad1", +"878e637261705bc7591c6aab1f8146879e79fa7b3e64ded2f0d9a804bb09cf7c", +"1074b8e6979809cc6cc15a4d5436400b0e81ea3830b0bf74c7cf7c5ac7663580", +"0cae13058bca582af29dcd1812ae3d396e08155c89553316f25f59f8705636ac", +"4efc6f1069d00fc91bb47e7f88aa0479ecdf5817c9cbe4571f98b3081bb83472", +"9faaa6c7b786c447618edfb1c76a8e36600cc3621dfc1ba1fb63a9c8330db494", +"27290145da0856d54f5852cdc3b37ef6b6a9f5de2b199dae290fdd0dd3356e3c", +"29a21710d9356373534d3b5ac515c0f52a2d9ae58068781fecbeee46bd5e3130", +"10267868aef4a7cf60ac5da26c132d7faa38346273b4f87279f02df87c52116c", +"e41776595c5c14ab808f4815ea159be866830fa2d3cbd757104dfd642bee62a0", +"f96c824868d59d38926f486bc58ea697f57e73a694294e304397000bca02e08e", +"3979a345203a44098bd71d93255d3d4de64b89c48e27c80bcd5568746ff5c93e", +"01982254408efcdeb6f632f3bd79589e62e450950facdb43648172fc8938dafb", +"cae34367610dc3eb123c6140d31ad2ec7799189906f7dd7b7e4905dc5afcf2eb", +"33230041003c45c856ae716ffd13401bb29d282db14e1f235166ac245ddad050", +"23ce8649e9e38fd125b51774ea90513317920715a5e6e7c3272da7c500bf0add", +"43720b29d4ac9786351c6c6dbf88baaf671e28fa7da379d36b257dd5f9ac5069", +"51ea531c1bb769bdd220aeada686bb321a8fbe646fb1f22662c7352c3220e3da", +"eef1f2d9d4f7cbaec3e50738aa2f25f75bcbe7ae80885129cf26dd61fcd2d684", +"d7cb93fd68148815dba1ff84bc5e9dd32d4f61fce2798842e4755ac338458c94", +"22e45dbd549f8f8d94068210c562e9165f53a8842ed2a4957fcbab7b45be9968", +"a20e63183a8375d5b785fc02e41c870d278f778be1c8adcce7e595f8fedf7e9a", +"f2f986f2cea963a6073ef33a4f72f8fdd751b3e5e7c8281e951966adf89c82ed", +"19266c77718d86cd45872a374537a63c047d323f8cc28e7b0c492e4e7499af7e", +"2d34e1c97c3de2aa059c6a2d22ca918e20e263d35bfa39b96e9cb098f556ecd1", +"cdf24342c9dd0f2672adc780816350025db36560f4e5aa9226a993627ea46261", +"bd2f7cafba11581f495da52700e5eef89b7136337e466ab06ee9572320c03a74", +"e6e4f77c612c736e2ddb7312b14bd86fcf867c9d8e822d27160ff152f5555005", +"384e3d84c43f538e3f28b747f3386d7e4e3591ac5e94746f0857d6566d885bf1", +"b26fb04a5fa4f8b18a8e283a15a471230e1e5fffd30fd8dc4f57a0a1722cb270", +"b4b6a0b149383bd9077463c2b4bc98360f81dd8f964b330f32637c86d0076140", +"61b708d7322033b5847791aaad03324de04d662f34bcc1f969e2ddddbe9bbe9b", +"09ffc2f53d494ac1c4547e3d5339755df7e939702184656f89abe509e8349dc0", +"b662580d2161840218833e521143aa2bf181c842fa37f98be1bc5b9137068e3b", +"630eec291e2f1c8503652dbf174d8dfc44e714c1bf8651ff39f74dded66568f1", +"f5c76bdac030ae05b9946295c24ad60e22e0f608f0b9114cc7e60a282725ab9c", +"5c864253d4db2ed14814ac43d6b7849f32d392129618bdeb25258251c34cf608", +"0e80eecc36770e23196cab5bf081313c03d86581de394871627a6ae67c2870a8", +"2454b8f91192d97c939e99e0b1cd37e90a64f56fabaccbd198253eedba3e1091", +"75726c9e06ff5229870b213f5c422d3696a468829755369e5069470b9277b1e9", +"efc8be24a1dcd56c7056f6c0467745a061a562f504afd54cbf5834c988df37e0", +"dfa7c5036f4e1ae08af93144585ec6b2923e8fcd01b070e516c57bb771b92798", +"eb4b8630b93508da9c197c0fffde911f871b8a2c4abcca773c96878686de8300", +"8661b0fc561f587920a4c2992644901276f61c2ac42c161ff26bec3eb80b957c", +"aa45fa3b7621aaaa3e40ddc3c88eab869155a457055c7f9094c3d9ed60472483", +"c3bfa8e25e9f13b4a18c51bbc503c68bfe6aabde36526e4f53e65cc48f73ca4c", +"b365eb897a61f17d275b315cebe5cf4e21f45f14c608f74592a9f2ad98b5a6e6", +"77f58446b8391f5384a9ce81ca896012cdc3a8ea32cb6c86fbcb0e475a451543", +"95c8dd4f9d2ccfc2396a9452fb60758da5f4cfe120b3047407073b96fc770eb6", +"fb879acfc730286f50655151ac90157edd412e5f3b906bd58751923cdee29a10", +"d7d0f5173c2d2cd92f96978b6e462baf37b55e24bc46cf485497e3b5b8c59811", +"49410cccc13d6a0dbab3fde04836773d185bb4d46a332d32965eedd2ab8eebaa", +"e4df7987e36658d0584a8abcbea9602f5365952057dbd6d7f6684eb391e37a2f", +"6078eef3848c81892832f2951993372d3ad12c9e35dc53de6b373878ec3fcaf5", +"229079a281a5e0e51b805d92bad8b4b7a4473d7c39c784835d051ef45b61ec06", +"ee4ad1afce4141b78ac310def0f73e2c0320332e5475fccb9ddf5adb1170204a", +"fe96a9c818802897b59fe9068c14f2e0b8a090adac46f02a49b2583b0940415b", +"24a5c4e494dab9094973184060483c1a2b85d1ce1b9a82a626eac5c34b121d8f", +"83b976a2e4bb1c5950d6c9eddc8383e4b6a07fc920590c6a3d2421f6b21ba2df", +"898317d173769f2943bac260acabc0b66bcb802bd92d2efe8c5728f81a646e3d", +"c0fee26f156fc702d9942c4e5fe236b20d0e51d51458c75c60c4b19e5e724602", +"2450b64e81119ab6f2367db9b69a532d9d89fcedcd7b3e46300a48eb09454300", +"7c8017c2accb94036cbd125f954ca4a7f79fac0a2eff6bcb30ddcf04551a03df", +"ef9640ac8e20476b1c639c96712db7b10a3dd21a2bc2c5a6e4fe6b030b730698", +"e8d99b12954b784440332038f60eea00d25a4295e04fb3c42c4648406db356a5", +"1770d01208c35e476911614aed6a6edb5e1d33a8195c804dd64a742cb5ed275e", +"95f6023b7272b594abc5470d97aafeb2acac563ff67a2bf172d6ce3a8aaa09da", +"1c80a0993cc265142aafff11cef0ea2f556fb8b4ba5f17e2146250f7456d91dc", +"3ea6c4f518e19828a0ae70f6e0f4b70d420f16826f32e50804173d5e1091409e", +"f68836385548e317c07cd7a6119d3035bc62623bb94556bd63fc603db50ba5f5", +"d8b802b8d6bebe65b61029903df259dfa474d9fab2b8cf5d22ace39fa4f41380", +"8d9af69afd20f913bb8ff0757e7d7b7de2e46919694d64fec0b8520fa6978f24", +"62491dd0196237a61927cb70a00e4a920dacbf38ddc00065dade6ea0fd275384", +"6365364f4c4e357c5ba1b2a4d9c23868991580807cfdc76e032fe476e4213874", +"0206e2b63011ec2d38f063d7e0e0e158e2d40539e7d791bf2f00edbd3442157e", +"5dc659b609edcab9535f1c186163417e536aef43dd6136cc648b51468c523cce", +"3355ef857f72f53f5082f9d0b31a5e0e809818d4669b6c60d4af3366f1b1b3be", +"e4f76b7583b35ff2eddd640dcf9a84320248cad0775b538f06d813170a21d128", +"bff686f7200e3bf74ccf2475c22ea87ef78cf0cd7d99a88ebe59bed7ef749cf4", +"097e9143da8b6e2a53da13a68a496750cb3bc103275e81bb758af55c3dee6737", +"354e5019378c4e7d96cd82ad379398911d92650d69e13ca7982dd821a1c5655e", +"73f4557fe7b4a9304d2ff22c7cfcb18cadb318c4e045add1cda991b7740ff1db", +"bc827de503e7ef42c38cfce0dc5a2ee6e0436f66f2ec94f0712da1917d7e6c81", +"8653e98db1837b29bc6f84fa0fe01f1e9428b5c140f359b1d14a5726aedd9690", +"c4fd1b966a3c46a7cc5b80675449f27d4de132a7a3bda5c822703477778e409e", +"2626f494d095b5653ccbd6b7f521a3d564441082de5dd1bd8d92ea35af0c9fa8", +"fa4ed7dbe4fdf792f8dcca408c0297a573af541806fedb9dd746e35de0e31fde", +"b88e1c041fc748ca4fafcf5f31e00c20467f1130462d9e01604d20d00aa6fe8a", +"c252664af40915dc4a2d7c2bd13f05c28a58b2d93e13ef373f673765330c8060", +"7d42eefa6d5fc90c3009d953d78f80714730bf3f894f37277aa5b0788e73019a", +"a3f469286adc9e2496736c4f2e9dcd444c691e4d365b8b40eecbf848c0507c04", +"70cce59c04b588cbb607edd684e154a4e6b9511d869821a8128fd7709793d2cc", +"ca45c36ac8c33c37e89f3e8aa9f68b957f14a7a68a1a79831fa3972ff00d73eb", +"581df4ce752444482c85e1ced92256b86e81fc8ccce8350ee1115f091033832a", +"5413fc25a705ac68cc05dd7add5b553d923b2ec3210637c804cb13e2e6e3c464", +"2d28b45d3a0450088195ef3f52abc9c00e728564ba07c43e18f07461ab487082", +"26ff3f3d32f48fb9bc841fad5460bdb5ce1e909dd95c8b1fdad4a63a37fc802e", +"2a71c8abbe494f5a7b77e210023aba67ee3ad72873d80d9106a6a904cb08110d", +"72eb92c9bbb160cd2b5bdb9254476d3ee17847fed304be1d5ea53546c7485cbe", +"fe67f3c3c7bd2b518863e241d77de378b0c6577ef9492284a5948db8370d7053", +"2fe0f577c3bdd5ae66ab06b97092334b740ab016de1fb629e2c7a94d4d967e32", +"768c2cb98f4b4bcecaedf48cf9f53e88e8194cb8e05fb0743be5b95ae88aaa13", +"3ce143ff5efb9178304f7bbb821bcc3754e850c72c636802b0ef94cc738959c6", +"cd3b5c55b63d01e4068f26347162a787584f8d0fecfa90eda0949d6c86fbb19a", +"e8a88097a902212e22fe263e797ccdf899247600dfa1cf69ace87ca22bd494dc", +"3e2165c8477f786601591cdb1c222505ebf3c1438e6090888c170607129ef816", +"71cfec4e1b3e4b85b9edd0394aaf76a2411653b83d6c8acdaac93520e2ca1243", +"c68f558f43992d5d342c75ab608b1d7aac0c54c906dd8f828082b78dbbb87512", +"77f0adf765cd00039dfef63f8133c9cbedc92c2fca3e300758bb994cc0aa4e51", +"3a36ede5152f03d80f70f27bb36d61989c33194c3c391f49483cdf07030391ee", +"d33374cb9f0cff75b590e0ea05525668e79195e414becdb3bfc1b32912950c02", +"5b87326e1ea8b276f28801c5f4d249ca530fcc257c6944305f68e4e052972643", +"7086214d6aee284e4e81e3274851dc4199911dc9c2c345f3f283605a0e3d6d1a", +"49b0f32284b4ea19cf6cad9f0d3336fa09faaf2ccb056465c83a22e1d054caa3", +"a06d6161b35fa45a83ba870e8a28e50f073c1daab08f8df4be360fd1f75062ce", +"93fea2b08da012d3d82e96662143485a6bfd3f882cc188f9b8a0e16ae06281f0", +"7ca940f27837c872dd94a4223ca3c1e0e0b30e9c3db060b3ea0595120b43050e", +"ea0ebe99a38de98b49d75c9c438797157a9d0da29f88e6a2c4a7e014a8d74c97", +"a6c0c0743ca6b9304d2dc9772a77638d8ca66d9a594fea27fd1bdcc430ab51aa", +"b69f49ad2a4904de92d1ed2b1eaadf09b036f7215d9a4886ecbcb3d2a994c8fb", +"f0acd6ef6f67edd92fd0e560919e44661216d8c961fd201c50ea512ca102635e", +"37db30b8a78dd4309a27d25a319635721f55a7ec178d8df7d3d6f26a5f8f534f", +"95afe2c38a58919edd94178ede7b3be71146b831563c69f9dc0809bfc9bfdc72", +"0ae7d4b1e807c7333a8a16ececf9de80380961a4c9140f7d00ad5e5987b97b2f", +"10d44e89c9be89253117ce4af117c8b2b2cf3bb228f964e2b23e84c51044da5a", +"4c1969f11518cd2bfa7e245222ed4d32cdd571e57eac2cf4cd8111ac1ce325f6", +"80b7f682d606dd3f60d93c608f9e7f0c0f568e2cab763ed32527b6e57bf4d067", +"cbff0c62306d38365f89069a578b553738597922a25d1929bff075db37d5247e", +"e27d1189d0d90cc8cfb3720f1465695df6d7a5ce9b79502ce0778c1fb98c7e02", +"43961438ddd1335268fe9cff04aa1615d06ab9239f931006d4ff8d4a9f17bc12", +"b86320ac11f1170996deb6ea4613a1bb9854e6ee49d58c1e1d147124c6e77545", +"4eda172383e35a0de992d0e91084fbd92b037b24064d88eb97a2347e9a5c50aa", +"5c504b123c666712553a5a8095b30f0d0bb463606e1b773652f70d1cc18b3d87", +"e7cf4cf38ff099de63b7ddd97ed77fefd48703a11b18dee3ea145bc774e8b9c5", +"cfd473c7b1f15e558e1f4d211811e718174dd57a6d9a045342579bb707a52b94", +"93b14882192a4c7a1a3a98118255807c1b7831433401651df0a911c64115fdbe", +"7a71af3b5fc9aab773eb25cde6e21b344608ec7f1db3664e1aec04e97ca440e7", +"4d1fd6e8f8537006c8e379a233c565d5960b3150888c3a2617a1d3abc2a7ea65", +"a20cbe51cc30a0cf6a82ce091b031b2107147194e8279dc548882c4b44364027", +"a59a2b64cfffdced679d468fe7e1e38013af804c6b17db6a70f0ec611801d30e", +"74fc49394859d048de8882baa5fe42f2189dfcea97e3f13141e90fbc05757f36", +"387942cfd03c71cab6eec51712cab7bd8a452b9e9ee7fcef78cc433f37077a8b", +"2698348edb6de2e5a14eda6ac5c051b8fd064d3752b821adef9aab961898d98b", +"86e09de1f4dd2ff1d164cef1ec22720421bb4d8d1a6554ecf163cd5eef84cab1", +"a5808462a492b788aef259096ff34b63b69def56023c5936672f1dbdc73ba3f3", +"4bc6f49f40314e2f88ede985192da44b350be2b64d2e43c24e2e3cda706624a6", +"21d92895ac2007bd647bdbf3f08ed66e35568ff25c8380130536fbd5cd2cb79a", +"374e30a2442f67cbfd4959a96c4d450e2c5f9bd2365301b5a6178ff7ac3827b9", +"92b0dfd5ac0b6e2e8dae164570747de735b1c26b71b5afb813e26ba22622fda8", +"c02ff35e2440c9f92aa8d983d4de5dea804498a2116364d0b402262559466cf5", +"17427404bfa931cc830db15bfdf6496afaaef006683394f811ea29a78857c5f5", +"a4a2b60ec629521a023e7a8c1648373380dad586c854c2dd6c8a66b2f8d3838c", +"31590d929b8f7bd494cf3068dea0b7d0a8c8df13d39094ec965f6cb14cb0fb31", +"192e00b2397d2e856339a6b37c522eaad4119e533ba9b1b50539b5cb763deb21", +"5432adbd56ecec66d599b346ca20778be6e89ace49e3a94f40daa87779bd2230", +"4084bcc49ec9335e8b062ce102dd164c74a5820f0611d285a64fa0ee3067cd0f", +"4831babf5696b40d935cc552991af295b6de7a51e4d85419695e28dec0739bd5", +"0b036dfcd54233607db0240ed765315133c90befe834949c470a7e132897c7e8", +"6a4dc9e10cf26b35569340a2cdca7fc1401b020dfa47e32fa2ec6cb384302b11", +"928c3fdd312b2b4a51da8d78d4c6e47b5b166a2bd97d6b148b12c182a5b8aee1", +"b8157616e0b5b009a95ec2a85a740ca336305de3fbef75382c09fc9030cd9795", +"c518a9a62e12ba2a04aef6f077f7c5a3211c7edf6d5a76d786428a4787ab5df5", +"011abd80c6e20e22a97a34a36e913fc4f2e80978d07ee2ce587aff0ea9203349", +"32ff541b5cad18e87fa6b62d792e6f55dd6193f1c95e9cb22c7a49bbe7eca5b5", +"5ed717a11d4656c86249585e246fb9a6e3d762b7d8dd56a4a729a6265323a3c3", +"0729bc59480d6e9bd3b3fc4fce30234932f82c5859b6cd871a866d4ae2e127df", +"2c14ea7eff9f3824b0d6903b70163da82019b3b7d73b21c869cb7b2e1e66de59", +"815d4153654775ff14bfb50fdc22845f5f50b037d2e536f8d79fed1718892981", +"0dfd89c14925f44746f457b8486ce1c01ad9112f6fbe959e17df36ff563f863b", +"d7083af4f38bc44bfd06f3d354c483a1a0056218c8cf5492e7bd511f3604c9c3", +"5a1f2617b4c599f5c6754331b30448e8c40e6a4678a7db59c265a0e6e6b33604", +"0d32c3235e1fadb089955e003501b5df491ef54f1a4ca7320ba9f119b48b370e", +"25ee6f16e57aaf5d7c617846f2f9586675a9a7fd468c8d3fa74e696f60a6d066", +"b5c00d9c573736609f09ff5ad1b68ccdb90f53eae4a43e26f003cceebba47b3b", +"40177bb93d21ebde38ec26fb248fcc8bef7d7bdaf763c9ee610a04594fe90173", +"4cbc0cb8dd6754b15e545a0a26a770b31464bc9f070db643adf743f7de00164b", +"4d3af0600554a229bd5e12822640d2a266ec6e73148289d671c5550c04f1c56e", +"895c8f042dca11dcf99b12efbd80b5d0ffef494702ff8d5555e1feaacac91626", +"ec36f3792241887786a9bfbe8980870436d88ad76a64bf40b67a08ec36076153", +"ebdf7587f0ef85c72731b5a38bd4dbccdc2eef1166c707caefb9e6859490753f", +"b935bfbd37db1723740fe6f076e5bbf53ebbc1052888396723dfe8e2543dea40", +"3246885e3589c52413fcb2844e78ecfaad3ec8a491a3e87934cb9122c0e22877", +"1112bbde51a5828ea530c58411a6954b6a2b8bb1c5cf2dfba39c8e93626f69ff", +"9f85769c112dad02f50eac20b2c55c2a0d654e6e56d38ca12df576f2992b1329", +"83ed96125a09c6a44132a796bde8b7cfd1b7a2570115afba3b8ee9bb1b5311c7", +"2e08967288c1f09f6fa3ba88bbcea7251b3a2e4ed5819b54685d04014cf76b8e", +"1b29e19a7bc81304c6ff0f8bee5ed79a6158e80cbd2ccd68e53d2df5db4849f0", +"d30090121e6c301a90f225114654138d79218ba1f98cdd93cb447aad1429cb84", +"8cad0ce4183bec55417bf7ffd6e88b288fb47dfce7a47594244481b6fa38474c", +"697ba73c8b5a8ed9f2a723ad53eb070500d85f41014cb97106634f980ee0b12d", +"bf051b3ea14377e615cfdc1a293df10f9a9b3d131c4b434bdeba2275976a806d", +"81b211124200c2b3ec489e6289ba20b3cc928cddf80451b8dfd902a03cbe5058", +"1e26608b6cc7749fae2791b52252232945aff710136f606d4c1851a7006aea1b", +"ac0b2b65014f3cb7184733f4a4c89d4e8753cd44ed511939e2834049a511e795", +"7dd0a92492df72707965c4ac76c4fb8f64f3b030bb9cedca3c72f242b67eb3bf", +"0f6da3c7964ab0d9ba023662bf8acd04012f7e543d3adc1caed9c8693c45f324", +"aac7a68805f394eed203b8448cab5ac384782bae222820ff8e83c3212e429121", +"a323e9cce1a9c4a3951dfcf9e525bdb78a77a34d73e2353beb79009c8f8a73d3", +"e81f8eedd954de2f654c3f52711eb14ef1cd595c3d242eea6c3bb4b8e5518137", +"0724eba43a43adce51563a908b2b99b7712283696fe4685fbeac1f9ab75dcb06", +"1d523aefb412d6c1923c4520cf514dd4187bccf78e2b10bcc9432456be782c51", +"34f0d4ac9f4630b64c9c9a04cd62b1b2a3d6c14e293c096c905336aba5107e2c", +"a9d0df2cdcf1e2cba59e46767732ef987a4698a500b9034a87e31682dd993e73", +"b37d59f476fec2ccaa6faf64478848f232e23e1beaba6151e972d882feb8b996", +"21aed7f6bdad94bf02279e528f53c49a401a1528f79291cb7f8da7d544d1e97d", +"0649494921eb68598debff5e4e012bab49a6df6849e8ed41a1a458a77c1df8f5", +"6f39f32d889e918be8be7a598e7ead09562e29ffabe476fc4a745d922977de11", +"8bac05ffebb73c22a2f65a8f7e7aaec07ddc16ae3cd0a7b74e4a34e04ea25832", +"3a584822f78a74c1a5267e20299adfa8de86caf85e5050e469868c4b11bf1301", +"b74e33932733bdf0738930c2b37fcb582baf5ea99b999ee4edac8127e980eea3", +"1f475e98718bad1ef571f381c9104adb1c7b0bf35934dc66a8cefc7a6228ea6e", +"2bc241bae962f35ad45d7edbd69b18b7da0608223ac8223af03e83c51816c494", +"c46bf1f9866e899e1728e8cd6a19adb5ddc76de33696ccf288cc2a68988c19b2", +"bc8abf0c2ad8027211595630f6e65eeea97145ebae192fdd5a47c9d5693cc7c0", +"2b4da2c9c6518efb5d8a5f1f51590f53454d103c0f7e85219007563cc6e247f8", +"ee1bf839d511d2b0de3b07a837d1175316a7f94ef0528d3f8dc94cca211620db", +"d5d43779b9d3a18d24cbb59b250bfc954a5ba8f45a251914c03c899bd2695158", +"106043e339be2d1875f53e1510712c7773c6d85bacb806034bf4ee0d6d7b6b76", +"7f66c3bee096d46b8ea320ddc416143aed6667dcb6c13dbf491444a052834f40", +"4891006444918b895fd13ec97d4b7d671141e0675b9d64fc751b5b1be7d4c3cd", +"aa102c4df0cf41da6fdda24cdf6186a236f8821c2fcff496dc7ddc63b81b5897", +"d2dda4c08eb46128861dbd7888afdc57f54e91525017dfca043206965686fb81", +"4ba80148b21f6314b3b322648ace69f967fecb3dbf66845c9d7cf7097c73a7bc", +"3266ba11cef272e1b67481c00099a6e0983743ee0f8497440585ddd167406899", +"f19b49bb97dd273f3ee09ccdc1acc4af8b512356dc9a2a6290a84b4b6296471f", +"271fbe461d06449bafa50f4a23be3cc1563b0fbe4894ec211ce7c637a68d0b98", +"ce3d9d5cf8f47b944e30e0f039b42da1f42ad72bd678542b28f446101e27579c", +"9d3324c6247091a7a0e9bd19fb1bb0e4084caa75cdf2ade6471068ef3dcbca3f", +"9cf11a5b54e4f05de724adb58ae5eb693a310fdfb47e1b906ace009b30391aa0", +"738841d2e2e86dd471d7f8dc50757825643dc546bc1180f8a7c37daf96a01763", +"1f835e16d0428d234a24cdb83f59c553a09328a4a07abaabd4e78eeea0f6bcaa", +"e69f50be8108372585f78c066336d52dc072b7cfac5be6193baaeefaf5916356", +"8972ab46c5c794bfb7db4e00f68584fba246e838b431a4a84f10e7c136ab3d16", +"c212dc36443128a8a0953331a8eb655d5d8fc9213781550db63ab7443cbaf1e2", +"c6ee231a1bafe97969c02b0025d777ac496c0afef31b6d3e5c42bff099b32e86", +"2734ec0d4290f62de2948d8c081befb55ec5d4cfa2b5e34a2016ffc521e64d35", +"eb3e29a4eb5af8387f80fb1bfd7799af6418c1243d8f637035cc6278a35d2747", +"0bdbed6184cd0ab994bdb191929f075d2bfff1c2bb2541a7f6aa2939c8f4adf6", +"271171197888fe90a37ef2fc0f579faf60e1d1075710780be8dbebdc2977c4ee", +"f1c71232d5b2b67f1f02c74d7027c3b63a84b02cae8bc738fdf627da1e4aa277", +"627a256c14b4d4841e67e79a2cc48d9f16934e77c304da17c52c230e7421153f", +"40042faf7f045b8a20165281feea6e116626cd9bf6150f8775b7c801050668e8", +"ffd77a505db6e6ebdccda183dd8590d5ae8bad19445c5377d2b3765a518938de", +"f2cced87d8f89e172229c1bab7650b87124a8f7493eb9d76c7ea49d056a07b2d", +"01321499b5bac27b175405193197f9c463182e8084176704dff84680390242d3", +"ab5775873b697ca3178b24d672faec5e84b1283a6da7e61391837a75605dd27d", +"fe7a7c15b26203eebfb146daefbd85c60f7afc83689a9f1bbc3002cc574aee48", +"5376f51ca5512fbae641a8fca0590d4f8fc4d0968c0956e3fab8243a849d2c98", +"5a8453e7be387d77dee25d95092e1bcbeb2ca752446b72089e51e2a263779439", +"6802224d6bc1c75b4229406e599a07ae4eb6d51edb699b184982b5eb4d381bb9", +"6f9554affddc3ae995ac0821a500494c00ff3dd07a076483cbede6b39b71f5da", +"7961456a8b5b93771bcfda66d1f7e694288ad65e3592ede99f3c5a0ac3c31795", +"872f81136150bcee7853b80b5af56742e4f1b3b5f149510ca9e96c0f5e707b58", +"2cf42abe1ece9b07360cd8839f9e0a8146608b3df9f63e0791af5d49d855a6e4", +"65d04046c2e30fb9748d3fe663210ae2291c63d7424ae311fefe2dd94f182c2a", +"3a5a537ef02cf705f04b1487dd5ad6f7686d5c27a7ad8c90678cb413c27c5808", +"ace7ee2696f4233eb9b7be1fc1161367ba18e7732a6126c761f826a021020bcb", +"0cc9e2bcbf970ab3811ffba16199ea2d65c4699964f1017419f21fd277652cc8", +"1502d382f1eab0e392f46e4621f933dced55db023528f605e5681a6159bb9bce", +"21f1250d9b600da8bcee76202636ea7c639c50c01d974b77531e55426c0fce8a", +"f9842cdc443cf7b45089bb89ec94ff118144361693a55f79e1becf66a76dbe17", +"5a439b172c629a06a9845e08ae4b189c70e1a247558696794dd84fbd2570108b", +"639ffa11da36ffcc3e9d896ef3d663eff48ba4c6d82d8fae2c67c6a4f9b001de", +"ba0ab4da907eb987c1f05a7ac49e2874849e4659aec28658a3fa8cae79ef225d", +"ee0bbce2db37397dfb71b032eb74ba3d3d11e08c87250bdf422014d7643ed4dc", +"dba982ad16ab90b15580fd26cc2de70440a237b7fe2596bcf6a8715e16c04e67", +"1e112179d6783031f1811071808bdf9406a8ad2cc99745465a79293d67ee994b", +"f1b800d77abd1d48ff1a091494dab0cefdbcf8f839b81b15787db45e78af5d4a", +"eeb00996a4acc6d9e23d49272da68a6a0089e98e0503e4b383616ec40df8c152", +"667af161e4ae2cab5c4b20fdfdf4f960443ef61c53564b6e44a310e557f5b52e", +"a28a2316ae84fcd8e2292bd53c76b1906d242acc791ab64be4c97d8335720679", +"f2acf0f4fa4c8b6c22b94033a45f3ce8160ec9aa1eaecfd86cf9a414525659ba", +"dd878673e6873015866cb99eb86c3dd0672a8cc946e5b3d1ea0ddb6e523e7720", +"82004e7a7b50d39c92b1cee4344b662c075e22816fb20399580212a66d122d9c", +"42f07c778a4aa7d5a4b1cd136ea68fef5fa2897dae0a8a9336de303ea869dbc7", +"f517f52f1703e44e4aabb501023bcb662ed64c7bf6ef144ae388e6090750d713", +"a52a484dcdbf7ae2c80118dfc72f36116d51b2e2cffd2f4fb327caaf80acf4e7", +"59023709e537b5ea015c196a98f7381fe2687fba887267552390794dcb18dcd7", +"7d11ba00dd419b6c810613c35150aff7c8842023f76f5aea496ee897b24df273", +"9b09ee41fbc37d041a35a7155f2bb063bf221be1770e55c036fb1d0db7990cd1", +"f91cce555b5fb90e14c7b3597073c23f972721bc93047c5bb0dabe57cb9aeff0", +"76b1733bb1f928f53c7dddd834a32c27d761d739086b5e00cdda57fbebf43b07", +"367737e345bb728e43b4a25070af42247ed782f987574bdeb6abea42f1517fcc", +"3dd26b1f5145df21fbdc23b4a5a1f7b897cd9871d209ee317801dd5713943096", +"4e1b57752fed92d5ea8c35fa95aa30a2e946a7e605474ef16c1462da2dc88d74", +"c26a9e9f55ebac4dba8323dbbcc398319c4e02ded2bcdbad5f7ab227a218f172", +"c9015c6ad7f7e59f8fcfb19f87347c9c9285a0356b668f969f3cfd31fd7dfd20", +"4d7efd5e88080862a687f3ebfc6d59ed481261f27bffe0d14cb3e50c94bfbe92", +"6a42f443f90a7ef3620b1f5c07529b40229b4a4e0bd8dee86e661182ab056e0d", +"480689a2cc9c577fc8c23fec07445d68df3e82b5d41ae0e2445270a232114d5a", +"f781ec4f3210fe0abe5e3036c6b6fbb68c5af853f7805abcf60e614ab8470a07", +"6473b797f5b28987987577a4b46acb49f5d154c6ee402ed9114f7db3bf939377", +"e141d8355689f92037da304b20386368bd04f707bbabc072ff9c61c19b186dff", +"fddb94f850d234d4870150812d16bfc2ae715ff67498c1b33acaa894b6e5a924", +"80e54effc477bcf6e47aef64558fa92dd475199f82f49a984e7bb448533a0697", +"5807d58c5b2ef18fea133b1a226dacd8d9d994998c531f441262aa6e3020d85d", +"482fd9128a281f14c6213b8065189a7f9a3b68da0716936b23b45976a4364c0f", +"a198135074353cd2fe81f55f62f2da556d6943f9faea0b616640bec36fc559fa", +"53dce74fdd2e3aff6c5738ed520c01530e3c239693eccd248675002765aaa721", +"f05102037b2e7be3de67e04e6e87909b7397761261ea819ecb0a9968bee5c92c", +"364ae47222dc0f386a8ac92450c00e661ba2d9f5f3716c7f2537d48df6cf016a", +"02935d56b9ecf8c7198c593b515411a1def3707e3c864443b9c3d56f42f087ca", +"2c4056c6d56ca5a1327030470af9cf11049b45fce1ceebfe395a042c48a70ef0", +"82c6cebacafe85c4099958f2f9a7bf6389d954c68037a3f84f9da23f22b0404f", +"7c13b3ac0d8d033eda4f82a9af0cf10da854d197e3d86e653b22c78367ee4c8e", +"6e36349972a9832f6b21ef94c6c3324bb39ccaba91c5c18a9e2d86fa1a72052b", +"d0c4cb2f0085b7ee58c12cd775826e1dc03f0ef36ddc61fbc775444e299188c5", +"5b5bbfd1ab139b2db868c8ab6fab3eff224cf4b60fb9b95100656f1eefae665d", +"db93fe55623be236ac7eb26868162a2cbd582891834553fb6ad22afd50a2b8ad", +"9b61b3a9aa37e79a344ba5a8dba9f48d3093b46282a73c3e1f550703c3eaadf1", +"f5c31d6cd985bfa0751b50cb32a9ade2f42e62188cbf57a9fca22096c2c475ad", +"7dca003d51c42ded7feb497a7bd163adacd9f8274302dfaa4f889c9507c667fc", +"5ea528b6479a8d92140661d2907dc101fd1a8bbca74d49d9df6c65cc68cfb9b0", +"549428a7c4f3f473e3d6bea478e499056a17b25d8006b37e08d77d9036a54caf", +"d5c9816ec7b5e4d73f00b56ed54f34a0961c13e2d2f44de472b195b41e7eb8a3", +"b6123d40feedc844647759588ef09b766064ac4d1dff5e64742791c4d4421d79", +"cb57c94eba2d5c02591a7dd4fb1785e725c82b37010bfac359f2f1e033889670", +"ab5d04c0efa5085cf881fda097f7c84ea08eec0306f0831e2efa12a0480fbb4c", +"cfa41109c8fc3db13c866c6a471c8aeebc02754d9630be274a2b5ebcf0e5bbf6", +"7f5584729c3cc91cd810bfef032900c2126d98897b03cf89d115916bb56751ba", +"89a311aa7fd60268c5c9b4144d88a4e8d210db42283ab2cb09271ee34bd7796e", +"f972613d7f1e92e7a0134efff684e3d91fe8dd36cc196b60cd449db52142c216", +"d2c2a7e374b6a7003a934f28b8300eb5b5c7b72fb342b115cfe415ebe798d55d", +"a24deb27115094d610054aa97a3913742363fca01df49f71bdb5239ce8b914ab", +"499d95a7df62fe6ac8e48ac40b76bb4c40866260d223ba9cc3a7bb2749560db5", +"d7bb952d73cce047178953c242bf0445362188901f2c855ea47cb9736589c848", +"5c57e90e80b5793f52c21bd8bff6524031d424c389fe1cbcdfd2b609b522e64e", +"977fbdbf4c84386f36a04025938fb937d5ad51a49f0e18b37f8db3aa44e2748c", +"d03a4c0196ed88ac144d2284b21c5478cbaf5c1a96319f18bf3ae76e0f5893b0", +"39e625579fbeb26c6a6feb83382f784fee7c69292f9d97136e82d70698544bd1", +"184b4d3f39af257a1d6542689094fa7159e71f4668eb8150e880b0b71ae3907b", +"df64016f1686c2d2ca0db01cc59e9d2d70c449633199f0c9a9ded4603a22a2f8", +"7c8cb4ffa06e6d6d26b7b557467fb646c3ab8c7c8c9aa6cb7a7fb69b72fea619", +"5ef25a5251e875714c49b0dde0aeb1ea9cd0de8b7c8a66927b9c8c2c59898d00", +"c15e4441def3f7f922461a12ee277afeda15979e344df421b89cec87536ed596", +"7fbd9853cdbde7cb5e11eafc0597b1cab3a895a39a05fc9c5b5981a0122bc144", +"344c2beb08d11433144fccc2772fc6f6199a312f13f59308a4b6711647aa7395", +"d766a88a811e504f11a240466641b16f79e22913e36c693b45551203303b8689", +"cfa6ecda43589d02e8b3637db9e2cd60ad5b00681e5f7ddf92743dea6576b02b", +"4b39c3652fda6453558dd999ad496c4ada603bd3bc77c9742fcda72a31ceacda", +"4863f946330aa9d3dac343600940b81656e6c59a879beb715a5db248f442c7b6", +"9382be2c14a31a042a6489369a118270b903d00301664d646fa2b3e5aa3969b8", +"8b27a452d1c0d29f04406f3ea1be8e3c3ad23138e84cf07d68e1e30d474fa284", +"bdc97186f7034b71bb38d03390afa087472a5d68295d66d5d53ff2fa1b6a322f", +"67b5315dd933ddc61ed5995088bfd7e2cb856ef815ca5fb034b37ea45875ae68", +"44ef3b2bffab078fe430407ae1810e6eb4f71e09a9d4973606ac79ea7ef096d4", +"d704e7779c22682a7ec6772cd0dde2ee311381b633f7eecc5ee66f06f1b53852", +"666f67427e0255853044f0afcca07310c91853d7df33832c6a786a8493f72165", +"d5a17e719a5519f459d7969efeea4fcc85d372f22f353a64844601889739aea1", +"5bc8bebe7854329a8c85d7aa6d8936010267ea8ceee4217bb047ea6bca7d5fb6", +"c3eca74e820cf051171a008e87fd3dd35926d1dbf83d92cacbecb6a5c46eba70", +"bfe0b8b5efcec8652e78040172b838b8b3a2904731591edba7e90d915ec3eb4e", +"bcfbe08160c723651915066e1838345254222b2f4fb7bd9461e9b7ecef66f4ff", +"92cb1a0676c7228afdd1203a467b785976f485c98d8d37474fb95894d6d08d62", +"91c44bbc37eb54b1717e2b2e4362746ceed5a25910fe3f4014556d1683de4855", +"58f4ab89c3f24bd539d3f3ba5931e0792f0316a0569c6e5bb7a4150b07aea87b", +"ba61babda725bb53aa37475a778988c3e84029081852f0f2be114a98f46385f6", +"a72e8cbadec2b9dfc5ba8fa553620dd13e597c1094dbae865b122453cfd27c94", +"e7d60a722a19dfde71d86727ab8208ad8f3cb0c9596e981950d04cc94d309242", +"e24360a549a9859b0d8fbd98f25dff341962d74891f5e68130989c5903343d88", +"17f1914e5c8e1243dd6ce054aee9cc603902291eabfc188108da0737beb66891", +"f20a77c40a58949dd8e270d9e74b94c81357d5913fa7cb9948a654f7aa2b47b3", +"684729bc71675324826e7cdeacd85c75b98c447dd2dcd2903a8d6c3c9d793ef8", +"c84e878b9011308f15642618bb87e682206ff1af3c7e1cabf48c40963e29d6fc", +"71a6ae73cf7fbb1690bdb3b9488fa508e4c92ce705cd292daa2968801773a4fc", +"84d34071f00c189ded6b7b7bdeb6c4d48c5085733efc51cf3a49013ac083f3bd", +"134ca7beec62027b08450788784b2e15f0c322d7d62f3ed831a7ef9e743fb23b", +"2f9d9071cd9ecba463f5c636faf106afa0a09ff87ac24497290a36d718cb70b5", +"e4cb79d5604c8e65d3e7b79969c269203e3b75ba3fc2d0d8216fd0acfa76b3c4", +"a666b714eceecaedc0c089441afbc014b996b4288f2bea69146aa42fdd01e5f5", +"3d90d0c89cc5dea2e84c35082618ba10266e222475d5ee2197824a7d59523857", +"8d1924f73f5c9260da8858d78db4693325944c330e4cda82791e3c8d632d4855", +"e9ad7580e9c06c3376aa72aa47c101dbea978d333fb1c85ac923d54e779edab5", +"ace9740a9f9ded55e044c49947b6594247a72034e1773faa820fdfaac84d1444", +"8156a3a8fbaa654b1e8b87fd5d059eb6a7aac9c758640527df295252bbe9a5fd", +"b3b4b5b931253a87de003d18b39ebd49982afda4021c4c7de7f4082bab04d188", +"d48d63d902ffff32b0bccec3b87bfa52af12afef7ee57e8c28856e5a97c2d35b", +"37bdf7aedfb7a4dd239a252cc7044bd83d0935a3874d514112c16fb47ddeab00", +"4fe9915b0802c2cbb9dd8bd87bb81712340c337cda8a02b9f0a86bceaeb941e9", +"18e689caf0d26182e5d901db06016671c4b261dc5dc328f85aec98e510c892fc", +"67dc7132bf41f578623328dc925781cb7d8de1ba9ae66f4bdfa5d53bf4492e5b", +"13ca31126b43525b3ae84aca0ee5cc888a799478e1aadc757937245e2988ae54", +"e4a1b1f85ba8d3bdb77397e30055fc02f18a4affa2f8baeb7c377ccc0cc2d244", +"23f407caadbd3b37baa95729db1ac07b11cb800de1c4e6471be7e69d7d821ec7", +"03cbf0dce5e14978781de36c97e7aa6f96e99f463bb8460bc2b2b04edaa576e9", +"d444b0e123078cc470214a7459f7cec22828e55ac0ef4ffae430780ba095d2cd", +"066288ab8a07a8ed4e2e99fe81dc2935b72302a2f74d29290d2b7a4d90b7bed3", +"14a2ac2eba70d3528144629054a800eca86b70b2788a73072016b5b4f7d74ac8", +"0b76c0fd4b90994ed02ba9fabe22a434aa99baa8723b1472d260a611fe82c93b", +"c4237fac6bb6881247866621b8441a08f3b41212cb25cd8f9d24f4b635065b5d", +"b127ee87a3bf6d2b4a655873a33488ad0948feef7ad1e5f17673e1335d25fe31", +"a23ee46829eb96259f556459694fc70f2f53b56a79ab32197b13d7b551235554", +"1a3d8bf7369b05c9e2662b8f38e58d7d80a1a9083e52b292f9d4a876a8891815", +"bfdc99612c4b5c897c478d4c3f9959feba23f908f2a132d32e53ab2130efcbab", +"59d14e2d50e975a9301406fdf922a3ab18902b311de1a3b36a3ad1d93228039f", +"d4956c0e29270ce7bfd33fa2f4cfce1ed0a29be6daaa9ef0b25815d84960b83f", +"e9b0f421e0d030e575a0833ab352d2307190ff2520c52982384486408579f37a", +"d55aca5b4557ee675e81889d1126530759a0d30820b1bf62a1eaa3fbc30ed56e", +"a887af491831652de111eca63da0423046089cc5381b347d70ed9bcc21fcee6d", +"958813406364057b4615566f1ed758a99a21e58d38e97331b680c06c209ee694", +"d27fa307db58219a9308804abdbffca489b5247c4d4ed827d8c9a23461b5c96f", +"1c1b1d00192649e3fbeb0d3788791916572f58ccb3cbe5bf3e71601b02ee4745", +"5542a6f2f8f87b5957fbb050fdcb1e68ce827cab703181d6ae56e79554151f8f", +"30f11d0627b5ba00615f00d7c5f8ae8ede781b4de4a2023b9557fa5c4044558e", +"909884b7d0b8e514ba9f1bc9925cbc0b4f817d5c5570fed07cbe3409bc8fe82c", +"ad06f3fc83e69ba0ed5249b1d28be9256854bf600c61ee853ce534b4526c9215", +"9145f71698b5ddfe75d65a91abaaf52bc64ddb46f74b84e20bdaa84c95fc3999", +"5ea25149284da4afd4e658cea371f0766bcef153f49354e4267fc58bdbdf26aa", +"b03287337f936b25f2a5b3040abcb0606fdda74a6eaa4063cea1c53a821fb965", +"120ebd8c6e16cb08c9752f18ac011b7ca71f571d2fb1624712653a7f633a5cc6", +"3874e84d2114a3631b27e2a89eaa2706de9d462cf5ca42b2aea29958649a481e", +"38d6e8570f385d2bf0cea6d3e3ea963a00b74cd1305111f7351f6f95cc830075", +"b93a3dee4b4d3bd2650f1a10dae6434975c846d77f056eeba7be2e28ed51d2ce", +"ffd919f6375d14bc6cc07af141dc2ca2f4d75a421146c2a3c7746a831efdf1ea", +"e53be1c11e0f2d225ba0edc6308ba1628599d0649cb66522de745c6ca5ac7537", +"053f9278ffafc8e50fd03cfda0982378d9eadf1980ae875541e23978f883b90d", +"a30e227f2ec8c79561d8f5e1311225dc8476459106dcb32b1769324b9de23afc", +"b7c2a4671798d7f87266b042043a59d4d3208eb769db7aa63f3a495f75be1d3e", +"f818642c0ce3294fa440a0d581a4527b8e2e0e70e3f47cd721c0c3e100743d0b", +"245c007e0de8e24d7cbc8c6ce3ddd0b5ac8cb97bbccb45b3409ce101e329b5e1", +"a992d481f7a931e961dd4051399a7982ee93b80313ba52adda62e5cc4e26a324", +"24e61a82c563e3517a442af6137fe0f7ae6b88b1d6b09f53af9520bd2159cf1d", +"dcd0afc8a1b077a4c8fadfac9513aec2f73535440c8d1f673bc66689d7cfd7e7", +"3a2779be4f068cebf26f91719ceeae2e1c858928b5519f5eb806379c9364cb2a", +"c300137ec88c0bcc42558bb00cd377048688942d7f71e0c064a3d9b1751ff61b", +"abe8327090aa007a1233b80d5e43edaa675d8a753bf69bc3ebd721b6fcfa7462", +"ba15e841189cdb3bce58b4a1c5a2248005fbaaf9b57ce3d0851d3edfe26286d5", +"99f12bc99f420c60d28263fc8d546f75acd2d7c8cd485e42658dab9f06930fb8", +"a61652c21910080e1ee57d6eac3de4176df4b3ecf3ae81153940b98f4a2332e7", +"a6a3db0e7c85368f1aad28359855230b6dd6678c751f60e19592ebb487abe17d", +"f27c6b23682a6912ddb3c0bd0547b920774d0ccd12e87d654f575b233e8555e3", +"b08a6d13a5eecc52f35eff8c6cb90749db2ead4c8fe3221a8ffe7e5c01e92206", +"623f3167a4a13437b1edfa4f5e6c220913a4635ebf0be1cb5c52e149dc69faa0", +"a2223061eb10d65d7a76621f0427ff6c08b238a25b86b5ad93431d2022906615", +"20288fd650524b6ce3c45522c412664ceb382bc78b0d72c65e964d7e0a168606", +"75eb18c4d8bf7f6076a490672c83bb3d588ddff257e76d44aa58c5d3507aea65", +"dd6063ba6e79383f900b7300ecf53e4535173e76b0ed1e46eb80aaed8633156c", +"91c023eff478100e94768b16a8c6b0c95a3d8abaa51e47117196c26064ad5a71", +"9ae6faa95145b7fe72dd5d7c55bcae439d9b0f14f8c9da6eb7fc29aafab8641e", +"2208d4d9cb8d2c4f0d922dfff75a5610e495fa69346ace3e2460c629272d67e7", +"21a35f23ef8bb1f20bd309738445653b35794f195464d641b823416e0d220ee6", +"cbcfeb1b4fd45a8145692b81c17f9f8344c292efc023d71f1212d4134c38bacd", +"dbb5c004fe3f8eb7f720e61cc82caa08a4821cb8049393b893bcc1208223f29c", +"318bff1092b9f05b5a0921ae93db816dfffa8993f78d057217b5611f61be9be0", +"56f0aa36b51c2971c5e5337de1c9ccbed6570f6a57f81c7ce1d4c1c54bc60ae4", +"f93a88ca79470a5bfb6d6ee27a33cd2579a9392b23c483ef9bb4a61a8db53401", +"3edeabcd5fc164fac3067643bca935d5c81984194a3a48f923a83ed8f63b2c22", +"45226dc82b41a1754b1ce1f18e264f447d6fe8d7757d12d601938906965f58d7", +"41671c742a150a339237ee364a18347cc82273d67013b82648ff7c0c029ada9f", +"70f3e24551dc1157ecb91511d9f5ff1e7b6b3e46ce15595d17af9477976f8a24", +"7cb537397a8f371c82b503a459cb2efd310d62f1a36d3578e840e0bb59790db8", +"415929456b25fcfc8ae9e6f8f8f5a3d91a5003252cbb5a67d5aaa2d2668b52b1", +"0c74aa73bd8c3dde03268a2447d1df9dba8ff05c48e14e70f62c785f8c0e0447", +"5ecf18f584bdf50ca374bfe49104126da278b87286df2df5d90c73b64133502f", +"4b8098a0d8f94adf60ebf974800b44964e7232a690cfbde4614c4e05cc7a9d71", +"84b42b5184434cece8408c5e0e8708938487adcee6b8622c0ba478dbc361edd2", +"8502bc65a8e9b11996a912da4eb42378c57b6a89a0c940422a9e61e62e5932e1", +"3ddd5113dd25ac5156fd99967e2cad71fd379b2b2ba32f43191431d50bae1023", +"40cbacf6bf1de3d1a9fb786beda3fdb462efae97b54f42cd02c4d998f01e1f22", +"4e1d7a2148a78f116442c3c654b875c7835fc8ec221d12685f91e27b44062667", +"8b97c2be125d2d4d404c3383fd8072b25d46a482aaaccf55730a56742fb157e5", +"119662c086ea71d1b6c89a910a71bd89349018e910395a3c645011374cc96a87", +"3e132b65fd592f06d11a4882fa68a7d0d7d85965fb43dbc0a852489bed0c7608", +"e6591e00399eed11156c4548a2764b323f67b98639a3f9d11ee7f9089f578005", +"941491b05aa5f85b69b1833891f970f1cf0f203fbb5b80541c92e6c2f6473f88", +"60adeb042c157c4d80c39f7301813207153cb31154630b05a9602ad8bc2e0741", +"1875fb5a0954b458bdaaf2d2051848f0d20ddcfd1f274551d009829a4ca2a599", +"f11d3ca9f33bc0c029aa718f1736e0ee854dd41b3cf0e23a03400955989e3f43", +"9b7694276ae3f1f73274cb5b3aa730cacf4ca3f050402ba49f35b34ff8726557", +"6b24b193560a087522e6e02b53fa1c32cf56b0c14e78c69bb9d529bf5e9c5a4c", +"25ce6b8d39d02471c43a804569a72015e81f3400170c1b8f5418e6938cd60b74", +"8aeb368d2a29616c4fb689fd9f236d9b8587a5b091b18dfe79a40ea947dd1a31", +"b51eeb48316bc7143e4deb2b2696b1fb2ca92516a008a7305668cfe491f86b8d", +"1fb4f0ca43898c25340ca1d504c92cb4187fac8d3154f9adfb437d332337b71e", +"499fd482997639f924eab76744d1b1c6c3901e6ff944057081b70c7eafda21ad", +"6f6ef4b81707e397dd1a032c3c3537b3bdaaf83f3d9e1360e3e4d8b273103b13", +"00bbd67a57b7ed2e6f46a0bb0b31c221c0867943778db6922ee9cb4bccb1a931", +"c94bc2d1b33fc6bdea57b3008f25f3cac75a82bbe644367b13391ec6eef37e93", +"ed0da9ea9849d55b25ca6b7074d217aaf442f4bf2a7613b1a1c9b4c3ac07587d", +"39239656b9e496c7f8199908f229c5c6a2b7822d5c0d3bfd36fb2d6355e2cb1f", +"91235731f97f5dcf65d7b8cfc7b82b6ad2ee05c4229ea9879ffd693f7356f5d6", +"3c096a9f245bcb0ff1ed90278d2752d11a822bf1813f896155519b0b237eb20e", +"3d2ce117aa4a1e0d271dbe98b8fe1a93d66a22f454611da6b0f903a29d07d86c", +"6089239513b862adabff013dcb5516fc4d591df9bd74cffec4dc82b5a0244414", +"82759072950f9f88c64a7e33bd096899bc70d3d9ffc667b90b15d1698c2b06eb", +"47e49fb45d2940fd9e7a905cb76e42b51ce1e69d7b990ea3807e56d1c4bc8cd5", +"a2ae606640dc06206b09f12e38419b64f640b4e538078971bb0758bf03ec65aa", +"0eb252c01c7a74e104b664fe14a99b6d6fc4ef5b9eb22a4deb6708030bb993ae", +"2331814ae1de551b87950fb6540fdc00258971360f341fa848c6cf951144aba9", +"83ed46d4a973f050cf9a1fd62b25f58b5bf17d88e2eccfb6223753a44e6a60b2", +"0e5aacff67c94420d3fdb5b04d324fadf73d6e4fcc8e3498fa752dc02aca7e01", +"368f9821b2a55a4560865f0711624ab207158eed8b72777fd3081a5669704b10", +"3d45cfcc8f8ea0126e1e84a4d84d3fb4ba4481c3fdaad535c797e059bf02d9ed", +"b26b80ba9aef7d5550c2aed207cab3fa463a3fad2fbde60570a17f8e054b10af", +"39dfedea1f573ed3d848a30afb525c41abddb167c4c2fc5e7723e0c67eb83b13", +"d54763e40d7c730668e369beef8d67299eb4df9d4ff5bb3b98d7ca52539bcbc2", +"5db4998af317f13c80bc0689c9d8708277767e9267b86b776abb26e5b3e7fd5e", +"c1da40532a16412890aa0d3d1be8e293a5abc073881ef11ca3d73afb78ba675d", +"fd569e6f14e5b61abc8b463a65d586cbdecb89d4069691f07e629c579ad97d78", +"824fa1787cb630d21b94b07aeff37f04a550280c6b210a2141b1a75fed78a19f", +"90e470d2c5110f20fb9c2f4dc762de341f13821f2029544926712a421598f4a8", +"e93efc232ae23f32afaa98704d4669d3efdc4332463de377c951f542b6232e35", +"62baa93a61368d738ff19510d6bdefb483d22db1abea00b8a144a86a538c9043", +"1da883051a17ab3a8cb2fa64859fbb717aa481a1c821d33e2717a746f1fbdea3", +"27fde813f981a417db07f007393d75b6d877b39cf07cc3ccb08949531b5ac443", +"005a926f9b47c844bfa40aa36af1f5c5ed969de1dcaf0817e27b8e8114853bfa", +"435f717ba0f812cd4032e22feeea49b4ab2a49aab092254923a6152ad303f462", +"ce90c1a446bdc7360ca1b51677eb66cb8de05e825152f6b79863fe10295dd785", +"e891e36dbd2300630b861c14d633fe3a9855814662ccef52dbee6cab61146346", +"bc6dac0eab42c59638a7b0df16e92d5b35535ef6cb925a1cb594d739996b12ba", +"0ed889d89a873a3b3872b85aeb438331f9f34dbd833a8e0476342df14cc5bf50", +"d493c02ede0463a74aaf120dd1cf140c2e7f5b119ec3556ebc5eb6994bead7aa", +"8a8c77517ae397e1228d9e88d692ce7846f723717379087d811b5da4d4142fdf", +"fe5b5f5aa3a2651aa69439b3c1c50b168c815c8ccf62eb689ffc325292b52036", +"b419e5abfea56ae0b6c0b5ef75633514de82073a41edfc50fe2f9fd1baa4594a", +"471c98dc80ba06d9e688692ac09fef073be040534f8fe842918b4607b285d9ab", +"55c895644c2474fad12301a9a9385df92779850efec50d43363109ad501eccb2", +"af47c709f231ac70d3729d3d798fc1ab180720568e0d5469280fc6df9456e481", +"57a7c1d46ca2bd284d8dbe0f415d0f961a6b02bf134cab8f480346551a028ccd", +"34b0b230c0de68bc37b137f39fb33fbb6ce7fe301281e72c9fc2cbd06b100680", +"71cceb56ecde070538cb20f4a49590690c7f48d9ada1335aa91572ab34708338", +"cdaab03910dac6288e916113ff03bcd1c23b0be97f57ebee7ce9b09dff582296", +"fddcf1e50a67ac5416609d603e29ba40d250bf67ae9c25ebeb587b82fbbf0373", +"a6c482493b714e1380de0933b638d8c5fe911d1408e2443e31ac6b2f76e99c10", +"53c67086b5d5aca3d64a8cbcc72a47d6d68b5a8ab7c03091eb272766c01ecee2", +"97f76ec97efcc942b172142455f3b20f10742c6d8a6011037657a1d5d5eb4ea4", +"0adc4a57d322771a90f93e7b907b6a3510c8065b162bdba54c6d0a9b7e9d1126", +"f46c0b0dfc887b3fa919cabccad0d37e06e9820f9b9d420e430165a80da44006", +"26dbcffef067af792cb6f3bbdca66fa8ac765f44c02fbe06efcf37909a681842", +"cf3d6b3ddc50fc2db5a91659950979c7bde4cbf0640600e6edab37e4548f33b4", +"1bead20256df5fbe932c987cc8b2fbc90c74ebc945f8951a1793172b8dea40dc", +"4c90ba5341aab5f292b4098b373bd85284fc971ef5f542a5c23dfe623f5dc3fe", +"4d0d75e7381aec27db8239c2771d05e9f689c14b9e13db6ad6273060ea9caedb", +"96c4c195d0556bd414a88f4cf99fb43f005d8c5fbe4c59f836709b9d709f50b1", +"1ece66443e87cc9694a4af0b4fa55f99d0fd444eb84d0bb4e6904d355dc163f3", +"6bdd2c7f1ec7f16fd39aa259e3290c27b7528d6f267d82a928b8dcc5d5d2932a", +"d7b1c521c8dad3de547b8ab06dfb5cc87546f983503421e7e7cbb55d9df264b3", +"3cbc09c1a9d11af20dc25ab018a77d33e878029ee76f0bc2d358f23630274d60", +"f5c11fab01dfba033bf5f6d7ac7a8305a5d6e156f541a6556d61b02411fc591a", +"9b8fc94e029b8594ca10de2bbf2488403dc0e45e68186bbc7c64376308743efa", +"65a4eff69fbf97128816534c40cbd3e4bc4eb1559d9cb314dad354a55661c03f", +"2c6c58338eddec9db1d914e214446ea718002e30bb5e7334c444b970277ad264", +"35ff769d552b6bf8cb43b87c5b9cf928ef853d6c9e4622b5fc46412f509b65a6", +"99708d0ba80cc26e8e025e3b1e0e4794e52fa7b94604fcedb9b1c77da16a4958", +"956ee08e61f6e266e8de04145cca8a5a1f802cbfb5a93a2f175e501948ff1d1e", +"0007f518fef4069ed7afe6f093fc73da3447133d5d6abd59c1978a2b597b6aa6", +"165498741a253759a44cc7458c9465fc9fee103c90d40d54554b9c9e699d93e9", +"30604dde9d5aca8b3afe3085e006be1c315d70726dd5764e7f5a6d633e825a80", +"ee967c6d8d73bcc91e49c269d94f96eaa8eea5dceb2a1457c0bd1987682fc5ca", +"ff1c53cc990d72ca0945b9b9a90b3441b5dffe8945e66b15d726d8cc5eadb90d", +"c67f93f277fe77cd78ea15fa184fa70f352a82508ad2fa66d972ddf49c08d5c2", +"09b341f61a0f40aef8442abe48eece12bb3844178eeee55aee73f003d3e9c37d", +"157c905fbbbbc2653a371f4b1d8e0672963e45d0baa01cb81ad9b41115eef281", +"d9c35922a4c7f0f6e0c7ae397232009375505686d13fabf211f72730977ebcd8", +"7da5fd597c97f37ed5e8553ac7f140ca877eb6a4fbcb351c3d070996088bc4da", +"fd73a72d5d89bdb2048f0f1ad0de85ec4df9b47b422b075caf611886e6ad7d5a", +"7b145b7994e8aaeb5558086d9991f61bdaa61e5e1a1f8ce9f2801491b5df289c", +"b0b6560074404e15ae2bba4f10f8730be816c9ba717ea5a6df06bf6d4016a665", +"5246ea08c3eca4fb23e14d1d2a040fc55ea7834223eb7d235b2d47eb950f5574", +"6ad2432e7390f323434708677009a5a9f160f01d1483c1d84d310256cd623be7", +"dc5a626c9414c1d79c0071ee763f621e3a9fac9351bb34ac3cec3241fd5c4cc0", +"3371c6a0843d8222b78a48f0a6403636d1ee9cde5b45e224d5141ef3b492f9c0", +"15229d101ee197bf179a9e11235901992d486cd99aafa632086733afb1e8008d", +"99caac952114658b7346bb27358582bccaa3ab9f24543ee981742a0439ee29f0", +"cb0e2ffdb432fa68b7c0046235053f3d3a339dcf1c58fbb16cd182aeca0f4734", +"b555084484f2a68e036af36c9c10aa4f449a00192e554151abdc3192e188f3d0", +"91e8f3330a08b726ae815fa05119e4cd53f329c018306b8568571399314a7ee9", +"b730f37f448dea4d9e18ec3e8bb0886a3d0257bd2c8a42a67a5a96815f586c61", +"19c40dff6bfee010ca3d3f41461b540109ec1d5d9fbb70584810eb2e4226666c", +"65502666bae303206d40cb1b8bf2500b6698a8d4a66289b791468f4f49703b7c", +"698675daa46546737422856fe2eb81fcb4740eb19d1f4cf7837c27603c9f9ce7", +"840a97091ab8d4f782c0daf9f3c7f67eaf7eae595a42d5fe0f7778a4cb3797b6", +"a9b7552649c39dfb83b9acab7d7cc8529e8b0438b7919e9a9360de28a64c5f02"] \ No newline at end of file diff --git a/node_modules/.bin/node-gyp-build b/node_modules/.bin/node-gyp-build new file mode 120000 index 0000000..671c6eb --- /dev/null +++ b/node_modules/.bin/node-gyp-build @@ -0,0 +1 @@ +../node-gyp-build/bin.js \ No newline at end of file diff --git a/node_modules/.bin/node-gyp-build-optional b/node_modules/.bin/node-gyp-build-optional new file mode 120000 index 0000000..46d347e --- /dev/null +++ b/node_modules/.bin/node-gyp-build-optional @@ -0,0 +1 @@ +../node-gyp-build/optional.js \ No newline at end of file diff --git a/node_modules/.bin/node-gyp-build-test b/node_modules/.bin/node-gyp-build-test new file mode 120000 index 0000000..d11de1b --- /dev/null +++ b/node_modules/.bin/node-gyp-build-test @@ -0,0 +1 @@ +../node-gyp-build/build-test.js \ No newline at end of file diff --git a/node_modules/.package-lock.json b/node_modules/.package-lock.json new file mode 100644 index 0000000..0ce71ca --- /dev/null +++ b/node_modules/.package-lock.json @@ -0,0 +1,174 @@ +{ + "name": "code-challenge-2024-amitx13", + "lockfileVersion": 3, + "requires": true, + "packages": { + "node_modules/bech32-buffer": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/bech32-buffer/-/bech32-buffer-0.2.1.tgz", + "integrity": "sha512-fCG1TyZuCN48Sdw97p/IR39fvqpFlWDVpG7qnuU1Uc3+Xtc/0uqAp8U7bMW/bGuVF5CcNVIXwxQsWwUr6un6FQ==", + "engines": { + "node": ">=8" + } + }, + "node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/brorand": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz", + "integrity": "sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==" + }, + "node_modules/elliptic": { + "version": "6.5.5", + "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.5.tgz", + "integrity": "sha512-7EjbcmUm17NQFu4Pmgmq2olYMj8nwMnpcddByChSUjArp8F5DQWcIcpriwO4ZToLNAJig0yiyjswfyGNje/ixw==", + "dependencies": { + "bn.js": "^4.11.9", + "brorand": "^1.1.0", + "hash.js": "^1.0.0", + "hmac-drbg": "^1.0.1", + "inherits": "^2.0.4", + "minimalistic-assert": "^1.0.1", + "minimalistic-crypto-utils": "^1.0.1" + } + }, + "node_modules/hash-base": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/hash-base/-/hash-base-3.1.0.tgz", + "integrity": "sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA==", + "dependencies": { + "inherits": "^2.0.4", + "readable-stream": "^3.6.0", + "safe-buffer": "^5.2.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/hash.js": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz", + "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==", + "dependencies": { + "inherits": "^2.0.3", + "minimalistic-assert": "^1.0.1" + } + }, + "node_modules/hmac-drbg": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz", + "integrity": "sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==", + "dependencies": { + "hash.js": "^1.0.3", + "minimalistic-assert": "^1.0.0", + "minimalistic-crypto-utils": "^1.0.1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "node_modules/js-sha256": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/js-sha256/-/js-sha256-0.11.0.tgz", + "integrity": "sha512-6xNlKayMZvds9h1Y1VWc0fQHQ82BxTXizWPEtEeGvmOUYpBRy4gbWroHLpzowe6xiQhHpelCQiE7HEdznyBL9Q==" + }, + "node_modules/minimalistic-assert": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz", + "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==" + }, + "node_modules/minimalistic-crypto-utils": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz", + "integrity": "sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==" + }, + "node_modules/node-addon-api": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-5.1.0.tgz", + "integrity": "sha512-eh0GgfEkpnoWDq+VY8OyvYhFEzBk6jIYbRKdIlyTiAXIVJ8PyBaKb0rp7oDtoddbdoHWhq8wwr+XZ81F1rpNdA==" + }, + "node_modules/node-gyp-build": { + "version": "4.8.0", + "resolved": "https://registry.npmjs.org/node-gyp-build/-/node-gyp-build-4.8.0.tgz", + "integrity": "sha512-u6fs2AEUljNho3EYTJNBfImO5QTo/J/1Etd+NVdCj7qWKUSN/bSLkZwhDv7I+w/MSC6qJ4cknepkAYykDdK8og==", + "bin": { + "node-gyp-build": "bin.js", + "node-gyp-build-optional": "optional.js", + "node-gyp-build-test": "build-test.js" + } + }, + "node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/ripemd160": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/ripemd160/-/ripemd160-2.0.2.tgz", + "integrity": "sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==", + "dependencies": { + "hash-base": "^3.0.0", + "inherits": "^2.0.1" + } + }, + "node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/secp256k1": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/secp256k1/-/secp256k1-5.0.0.tgz", + "integrity": "sha512-TKWX8xvoGHrxVdqbYeZM9w+izTF4b9z3NhSaDkdn81btvuh+ivbIMGT/zQvDtTFWhRlThpoz6LEYTr7n8A5GcA==", + "hasInstallScript": true, + "dependencies": { + "elliptic": "^6.5.4", + "node-addon-api": "^5.0.0", + "node-gyp-build": "^4.2.0" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/string_decoder": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", + "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", + "dependencies": { + "safe-buffer": "~5.2.0" + } + }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==" + } + } +} diff --git a/node_modules/bech32-buffer/CHANGELOG.md b/node_modules/bech32-buffer/CHANGELOG.md new file mode 100644 index 0000000..3dd6370 --- /dev/null +++ b/node_modules/bech32-buffer/CHANGELOG.md @@ -0,0 +1,62 @@ +# Changelog + +All notable changes to this project will be documented in this file. +The project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.html). + +## [Unreleased] + +## 0.2.1 - 2022-09-27 + +### Fixed + +- Fix encoding with uppercase prefix. Previously, such prefixes led to incorrect + checksum. With the fix, the prefix case is retained for the encoded string. + +## 0.2.0 - 2021-11-21 + +No substantial changes compared to the 0.2.0-rc.0 release. + +## 0.2.0-rc.0 - 2021-11-07 + +### Added + +- Support [Bech32m encoding] for arbitrary data and Bitcoin addresses. + The new functionality is available on [the demo web page](https://slowli.github.io/bech32-buffer/). + +## 0.1.3 - 2021-05-10 + +### Added + +- Document TypeScript typings. + +- Update readme to better document main package APIs. + +### Internal improvements + +- Update dev dependencies. + +- Update Bootstrap / JS dependencies for the demo page. + +## 0.1.2 - 2018-12-23 + +### Added + +- Add TypeScript typings for the package. + +### Internal improvements + +- Update dev dependencies. + +## 0.1.1 - 2018-12-15 + +### Added + +- Test browser support and add a browser version of the package in the `dist` directory. + +- Support Bitcoin addresses via `BitcoinAddress` class. + +## 0.1.0 - 2017-07-23 + +The initial release of `bech32-buffer`. + +[Bech32m encoding]: https://github.com/bitcoin/bips/blob/master/bip-0350.mediawiki diff --git a/node_modules/bech32-buffer/LICENSE b/node_modules/bech32-buffer/LICENSE new file mode 100644 index 0000000..8f71f43 --- /dev/null +++ b/node_modules/bech32-buffer/LICENSE @@ -0,0 +1,202 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright {yyyy} {name of copyright owner} + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + diff --git a/node_modules/bech32-buffer/README.md b/node_modules/bech32-buffer/README.md new file mode 100644 index 0000000..499c189 --- /dev/null +++ b/node_modules/bech32-buffer/README.md @@ -0,0 +1,185 @@ +# Bech32(m) Encoding for Arbitrary Buffers + +[![Build status][workflow-image]][workflow-url] +[![Code coverage][coveralls-image]][coveralls-url] +[![Demo][demo-image]][demo-url] + +[workflow-image]: https://github.com/slowli/bech32-buffer/workflows/Node.js%20CI/badge.svg?branch=master +[workflow-url]: https://github.com/slowli/bech32-buffer/actions +[coveralls-image]: https://img.shields.io/coveralls/slowli/bech32-buffer.svg +[coveralls-url]: https://coveralls.io/github/slowli/bech32-buffer +[demo-image]: https://img.shields.io/badge/demo-live-blue.svg +[demo-url]: https://slowli.github.io/bech32-buffer/ + +**Bech32** is a Bitcoin address format specified in [BIP 173][bip-173] and [BIP 350][bip-350]. +Among its advantages are: better adaptability to QR codes and in voice conversations, +and improved error detection. This library generalizes Bech32 and its modified version +(Bech32m) to encode any reasonably short byte buffers. + +## Usage + +### Encoding data + +```typescript +declare function encode( + prefix: string, + data: Uint8Array, + encoding: 'bech32' | 'bech32m' = 'bech32' +): string; +``` + +Encodes binary `data` with the specified human-readable `prefix` into a Bech32(m) string. +The case is preserved: if the prefix is uppercase, then the output will be uppercase +as well; otherwise, the output will be lowercase (including the case when the prefix does +not contain any letters). + +#### Arguments + +- **prefix:** string + Human-readable prefix to hint what kind of data Bech32 encodes. Must contain + ASCII chars in the range 33-126 +- **data:** Uint8Array + Binary data to encode +- **encoding:** `bech32` or `bech32m` + Specifies whether to use the original Bech32 encoding from [BIP 173][bip-173] + or the modified encoding from [BIP 350][bip-350]; they differ + in how a checksum is computed. If omitted, the original encoding is used. + +#### Return value + +String containing: + +1. `prefix` +2. `'1'` separator char +3. `data` encoded with the variant of base32 encoding used by Bech32, and +4. 6-char checksum calculated based on `prefix` and `data` + +#### Example + +```javascript +const bech32 = require('bech32-buffer'); +const data = new Uint8Array(20); +bech32.encode('test', data); +// 'test1qqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqql6aptf' +``` + +### Decoding data + +```typescript +declare function decode(message: string): { + prefix: string, + encoding: 'bech32' | 'bech32m', + data: Uint8Array +}; +``` + +Extracts human-readable prefix and binary data from the Bech32-encoded string. + +#### Arguments + +- **data:** string + String to decode + +#### Return value + +An object with the following fields: + +- **prefix:** string + Human-readable prefix +- **encoding:** `bech32` or `bech32m` + Encoding variant inferred from the checksum. +- **data:** Uint8Array + Binary data encoded into the input string + +Decoding may fail for a variety of reasons (e.g., invalid checksum, or invalid +chars in the input). In this case, `decode()` throws an exception +with a descriptive message. + +#### Example + +```javascript +const bech32 = require('bech32-buffer'); +const data = 'lost1qsyq7yqh9gk0szs5'; +bech32.decode(data); +// { +// prefix: 'lost', +// encoding: 'bech32', +// data: Uint8Array([ 4, 8, 15, 16, 23, 42 ]) +// } +``` + +### Bitcoin addresses + +```typescript +declare class BitcoinAddress { + prefix: 'bc' | 'tb'; + scriptVersion: number; + data: Uint8Array; + + static decode(message: string): BitcoinAddress; + constructor(prefix: 'bc' | 'tb', scriptVersion: number, data: Uint8Array); + encode(): string; + type(): void | 'p2wsh' | 'p2wpkh'; +} +``` + +Provides basic functionality to work with Bech32 encoding of Bitcoin addresses. +Addresses can be `decode`d from strings and `encode`d into strings. +It is also possible to check the `type` of an address. P2WSH and P2WPKH address +types are defined per [BIP 141]. Encoding constraints are defined per [BIP 173][bip-173] +and [BIP 350][bip-350]. + +#### Example + +```javascript +const { BitcoinAddress } = require('bech32-buffer'); +const address = BitcoinAddress.decode('BC1QW508D6QEJXTDG4Y5R3ZARVARY0C5XW7KV8F3T4'); +// address.prefix === 'bc' +// address.scriptVersion === 0 +// address.data.length === 20 +// address.type() === 'p2wpkh' +``` + +## Use in Browsers + +Use `dist/bech32-buffer.min.js` from the package distribution +or your favorite browserifier. In the first case, +the library will be available as a `bech32` global variable: + +```html + + + +``` + +Check out [the web demo](https://slowli.github.io/bech32-buffer/) to see how +**bech32-buffer** works in browser. It is also available in the `examples` +directory of the package. + +## Acknowledgements + +[BIP 173][bip-173] is authored by Pieter Wuille and Greg Maxwell and is licensed +under the 2-clause BSD license. +[BIP 350][bip-350] is authored by Pieter Wuille and is licensed +under the 2-clause BSD license. + +There are at least 2 existing implementations of Bech32 for JavaScript: + +- [The reference implementation][ref] by Pieter Wuille +- [Another implementation][bech32] available as the [`bech32` package][bech32-pkg] + +Both implementations are Bitcoin-specific, and the reference implementation +is also not in the Npm / yarn package manager. + +## License + +**bech32-buffer** is available under [Apache-2.0 license](LICENSE). + +[bip-173]: https://github.com/bitcoin/bips/blob/master/bip-0173.mediawiki +[bip-350]: https://github.com/bitcoin/bips/blob/master/bip-0350.mediawiki +[ref]: https://github.com/sipa/bech32/tree/master/ref/javascript +[bech32]: https://github.com/bitcoinjs/bech32 +[bech32-pkg]: https://www.npmjs.com/package/bech32 +[BIP 141]: https://github.com/bitcoin/bips/blob/master/bip-0141.mediawiki diff --git a/node_modules/bech32-buffer/dist/bech32-buffer.min.js b/node_modules/bech32-buffer/dist/bech32-buffer.min.js new file mode 100644 index 0000000..0664a0c --- /dev/null +++ b/node_modules/bech32-buffer/dist/bech32-buffer.min.js @@ -0,0 +1,11 @@ +(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.bech32 = f()}})(function(){var define,module,exports;return (function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i=e;)i-=e,d[j]=h>>i&g,j+=1}),f)0=c)throw new Error("Excessive padding: ".concat(i," (max ").concat(c-1," allowed)"));if(0!=h%(1<b)throw new RangeError("Invalid bits per element; 1 to 8 expected");return convert(toBitArrayUnchecked(a),8,c,b,!0),c}function fromBits(a,b,c){if(8b)throw new RangeError("Invalid bits per element; 1 to 8 expected");return convert(a,b,toBitArrayUnchecked(c),8,!1),c} + +},{}],2:[function(require,module,exports){ +"use strict";Object.defineProperty(exports,"__esModule",{value:!0}),exports.CHECKSUM_LENGTH=void 0,exports.createChecksum=createChecksum,exports.decode=decode,exports.decodeWithPrefix=decodeWithPrefix,exports.detectCase=detectCase,exports.encode=encode,exports.expandPrefix=expandPrefix,exports.verifyChecksum=verifyChecksum;var _bitConverter=require("./bit-converter"),CHARSET="qpzry9x8gf2tvdw0s3jn54khce6mua7l",BECH32M_CHECKSUM=734539939,MIN_CHAR_CODE=33,MAX_CHAR_CODE=126,CHECKSUM_LENGTH=6;exports.CHECKSUM_LENGTH=CHECKSUM_LENGTH;var CHAR_LOOKUP=function(){for(var a=new Map,b=0;b>25>>d)?b:b^c},(33554431&a)<<5^b)},1)}function expandPrefix(a,b){for(var c,d=0;d>5,b[d+a.length+1]=31&c;b[a.length]=0}function verifyChecksum(a){switch(polymod(a)){case 1:return"bech32";case BECH32M_CHECKSUM:return"bech32m";default:}}function createChecksum(a,b){var c;switch(b){case"bech32":c=1;break;case"bech32m":c=BECH32M_CHECKSUM;break;default:throw Error("Invalid encoding value: ".concat(b,"; expected bech32 or bech32m"));}for(var d,e=polymod(a)^c,f=0;f>d}function encode(a){return a.reduce(function(a,b){return a+CHARSET[b]},"")}function decode(a,b){for(var c,d=b||(0,_bitConverter.createBitArray)(a.length),e=0;eMAX_CHAR_CODE)throw new TypeError("Invalid char in ".concat(c,": ").concat(b,"; ")+"should be in ASCII range ".concat(MIN_CHAR_CODE,"-").concat(MAX_CHAR_CODE));e=e||65<=b&&90>=b,d=d||97<=b&&122>=b}if(d&&e)throw new TypeError("Mixed-case ".concat(c));else return e?"upper":d?"lower":null} + +},{"./bit-converter":1}],3:[function(require,module,exports){ +"use strict";Object.defineProperty(exports,"__esModule",{value:!0}),exports.BitcoinAddress=void 0,exports.decode=decode,exports.decodeTo5BitArray=decodeTo5BitArray,exports.encode=encode,exports.encode5BitArray=encode5BitArray,exports.from5BitArray=from5BitArray,exports.to5BitArray=to5BitArray;var _bitConverter=require("./bit-converter"),_encoding=require("./encoding");function _classCallCheck(a,b){if(!(a instanceof b))throw new TypeError("Cannot call a class as a function")}function _defineProperties(a,b){for(var c,d=0;dMAX_ENC_LENGTH)throw new Error("Message to be produced is too long (max ".concat(MAX_ENC_LENGTH," supported)"));var f=null!==(c=(0,_encoding.detectCase)(a,"prefix"))&&void 0!==c?c:"lower",g=(0,_bitConverter.createBitArray)(e);(0,_encoding.expandPrefix)(a.toLowerCase(),g.subarray(0,2*a.length+1));var h=g.subarray(2*a.length+1,g.length-_encoding.CHECKSUM_LENGTH);h.set(b),(0,_encoding.createChecksum)(g,d);var i=(0,_encoding.encode)(g.subarray(2*a.length+1));return"upper"===f&&(i=i.toUpperCase()),"".concat(a,"1").concat(i)}function encode(a,b){var c=2MAX_ENC_LENGTH)throw new TypeError("Message too long; max ".concat(MAX_ENC_LENGTH," expected"));(0,_encoding.detectCase)(a);var b=a.toLowerCase(),c=b.lastIndexOf("1");if(0>c)throw new Error("No separator char (\"1\") found");if(c>a.length-_encoding.CHECKSUM_LENGTH-1)throw new Error("Data part of the message too short (at least ".concat(_encoding.CHECKSUM_LENGTH," chars expected)"));var d=b.substring(0,c),e=(0,_encoding.decodeWithPrefix)(d,b.substring(c+1)),f=(0,_encoding.verifyChecksum)(e);if(f===void 0)throw new Error("Invalid checksum");return{prefix:d,encoding:f,data:e.subarray(2*d.length+1,e.length-_encoding.CHECKSUM_LENGTH)}}function decode(a){var b=decodeTo5BitArray(a),c=b.prefix,d=b.encoding,e=b.data;return{prefix:c,encoding:d,data:from5BitArray(e)}}var BitcoinAddress=function(){var a=Math.ceil;function b(a,c,d){if(_classCallCheck(this,b),"bc"!==a&&"tb"!==a)throw new Error("Invalid human-readable prefix, \"bc\" or \"tb\" expected");if(0>c||16d.length||40 + + + + + + + + Bech32 Demo + + + + +
+
+

bech32-buffer

+

Bech32 is a Bitcoin address format proposed by Pieter Wuille and Greg Maxwell in BIP 173 and later amended by BIP 350. + Besides Bitcoin addresses, Bech32 can encode any short binary data.

+
+
+ +
+
+
+
+
+ +
+
+
+
+
+
+ +
+ +
+
A short tag describing what a Bech32-encoded string is used for.
+
+
+ +
+
+ + + + +
+
+
+
+
+
+ +
+
+ + + + +
+
+
+
+ +
+ +
+
Note that version 0 scripts should have length 20 or 32 bytes.
+
+
+
+
+ + +
Binary data (in hex) to be encoded into Bech32.
+
+ + + Random data +
+
+
+
+ +
+
+

Bech32

+
+
+
+
+ + +
Encoded Bech32 string.
+
+ + +
+
+
+
+
+ + +
+ + + + + + + + + + diff --git a/node_modules/bech32-buffer/examples/qrcode.min.js b/node_modules/bech32-buffer/examples/qrcode.min.js new file mode 100644 index 0000000..6451a97 --- /dev/null +++ b/node_modules/bech32-buffer/examples/qrcode.min.js @@ -0,0 +1,1991 @@ +(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.qrcode = f()}})(function(){var define,module,exports;return (function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i= 7) { + setupTypeNumber(test); + } + + if (_dataCache == null) { + _dataCache = createData(_typeNumber, _errorCorrectionLevel, _dataList); + } + + mapData(_dataCache, maskPattern); + }; + + var setupPositionProbePattern = function (row, col) { + for (var r = -1; r <= 7; r += 1) { + if (row + r <= -1 || _moduleCount <= row + r) continue; + + for (var c = -1; c <= 7; c += 1) { + if (col + c <= -1 || _moduleCount <= col + c) continue; + + if (0 <= r && r <= 6 && (c == 0 || c == 6) || 0 <= c && c <= 6 && (r == 0 || r == 6) || 2 <= r && r <= 4 && 2 <= c && c <= 4) { + _modules[row + r][col + c] = true; + } else { + _modules[row + r][col + c] = false; + } + } + } + }; + + var getBestMaskPattern = function () { + var minLostPoint = 0; + var pattern = 0; + + for (var i = 0; i < 8; i += 1) { + makeImpl(true, i); + var lostPoint = QRUtil.getLostPoint(_this); + + if (i == 0 || minLostPoint > lostPoint) { + minLostPoint = lostPoint; + pattern = i; + } + } + + return pattern; + }; + + var setupTimingPattern = function () { + for (var r = 8; r < _moduleCount - 8; r += 1) { + if (_modules[r][6] != null) { + continue; + } + + _modules[r][6] = r % 2 == 0; + } + + for (var c = 8; c < _moduleCount - 8; c += 1) { + if (_modules[6][c] != null) { + continue; + } + + _modules[6][c] = c % 2 == 0; + } + }; + + var setupPositionAdjustPattern = function () { + var pos = QRUtil.getPatternPosition(_typeNumber); + + for (var i = 0; i < pos.length; i += 1) { + for (var j = 0; j < pos.length; j += 1) { + var row = pos[i]; + var col = pos[j]; + + if (_modules[row][col] != null) { + continue; + } + + for (var r = -2; r <= 2; r += 1) { + for (var c = -2; c <= 2; c += 1) { + if (r == -2 || r == 2 || c == -2 || c == 2 || r == 0 && c == 0) { + _modules[row + r][col + c] = true; + } else { + _modules[row + r][col + c] = false; + } + } + } + } + } + }; + + var setupTypeNumber = function (test) { + var bits = QRUtil.getBCHTypeNumber(_typeNumber); + + for (var i = 0; i < 18; i += 1) { + var mod = !test && (bits >> i & 1) == 1; + _modules[Math.floor(i / 3)][i % 3 + _moduleCount - 8 - 3] = mod; + } + + for (var i = 0; i < 18; i += 1) { + var mod = !test && (bits >> i & 1) == 1; + _modules[i % 3 + _moduleCount - 8 - 3][Math.floor(i / 3)] = mod; + } + }; + + var setupTypeInfo = function (test, maskPattern) { + var data = _errorCorrectionLevel << 3 | maskPattern; + var bits = QRUtil.getBCHTypeInfo(data); // vertical + + for (var i = 0; i < 15; i += 1) { + var mod = !test && (bits >> i & 1) == 1; + + if (i < 6) { + _modules[i][8] = mod; + } else if (i < 8) { + _modules[i + 1][8] = mod; + } else { + _modules[_moduleCount - 15 + i][8] = mod; + } + } // horizontal + + + for (var i = 0; i < 15; i += 1) { + var mod = !test && (bits >> i & 1) == 1; + + if (i < 8) { + _modules[8][_moduleCount - i - 1] = mod; + } else if (i < 9) { + _modules[8][15 - i - 1 + 1] = mod; + } else { + _modules[8][15 - i - 1] = mod; + } + } // fixed module + + + _modules[_moduleCount - 8][8] = !test; + }; + + var mapData = function (data, maskPattern) { + var inc = -1; + var row = _moduleCount - 1; + var bitIndex = 7; + var byteIndex = 0; + var maskFunc = QRUtil.getMaskFunction(maskPattern); + + for (var col = _moduleCount - 1; col > 0; col -= 2) { + if (col == 6) col -= 1; + + while (true) { + for (var c = 0; c < 2; c += 1) { + if (_modules[row][col - c] == null) { + var dark = false; + + if (byteIndex < data.length) { + dark = (data[byteIndex] >>> bitIndex & 1) == 1; + } + + var mask = maskFunc(row, col - c); + + if (mask) { + dark = !dark; + } + + _modules[row][col - c] = dark; + bitIndex -= 1; + + if (bitIndex == -1) { + byteIndex += 1; + bitIndex = 7; + } + } + } + + row += inc; + + if (row < 0 || _moduleCount <= row) { + row -= inc; + inc = -inc; + break; + } + } + } + }; + + var createBytes = function (buffer, rsBlocks) { + var offset = 0; + var maxDcCount = 0; + var maxEcCount = 0; + var dcdata = new Array(rsBlocks.length); + var ecdata = new Array(rsBlocks.length); + + for (var r = 0; r < rsBlocks.length; r += 1) { + var dcCount = rsBlocks[r].dataCount; + var ecCount = rsBlocks[r].totalCount - dcCount; + maxDcCount = Math.max(maxDcCount, dcCount); + maxEcCount = Math.max(maxEcCount, ecCount); + dcdata[r] = new Array(dcCount); + + for (var i = 0; i < dcdata[r].length; i += 1) { + dcdata[r][i] = 0xff & buffer.getBuffer()[i + offset]; + } + + offset += dcCount; + var rsPoly = QRUtil.getErrorCorrectPolynomial(ecCount); + var rawPoly = qrPolynomial(dcdata[r], rsPoly.getLength() - 1); + var modPoly = rawPoly.mod(rsPoly); + ecdata[r] = new Array(rsPoly.getLength() - 1); + + for (var i = 0; i < ecdata[r].length; i += 1) { + var modIndex = i + modPoly.getLength() - ecdata[r].length; + ecdata[r][i] = modIndex >= 0 ? modPoly.getAt(modIndex) : 0; + } + } + + var totalCodeCount = 0; + + for (var i = 0; i < rsBlocks.length; i += 1) { + totalCodeCount += rsBlocks[i].totalCount; + } + + var data = new Array(totalCodeCount); + var index = 0; + + for (var i = 0; i < maxDcCount; i += 1) { + for (var r = 0; r < rsBlocks.length; r += 1) { + if (i < dcdata[r].length) { + data[index] = dcdata[r][i]; + index += 1; + } + } + } + + for (var i = 0; i < maxEcCount; i += 1) { + for (var r = 0; r < rsBlocks.length; r += 1) { + if (i < ecdata[r].length) { + data[index] = ecdata[r][i]; + index += 1; + } + } + } + + return data; + }; + + var createData = function (typeNumber, errorCorrectionLevel, dataList) { + var rsBlocks = QRRSBlock.getRSBlocks(typeNumber, errorCorrectionLevel); + var buffer = qrBitBuffer(); + + for (var i = 0; i < dataList.length; i += 1) { + var data = dataList[i]; + buffer.put(data.getMode(), 4); + buffer.put(data.getLength(), QRUtil.getLengthInBits(data.getMode(), typeNumber)); + data.write(buffer); + } // calc num max data. + + + var totalDataCount = 0; + + for (var i = 0; i < rsBlocks.length; i += 1) { + totalDataCount += rsBlocks[i].dataCount; + } + + if (buffer.getLengthInBits() > totalDataCount * 8) { + throw 'code length overflow. (' + buffer.getLengthInBits() + '>' + totalDataCount * 8 + ')'; + } // end code + + + if (buffer.getLengthInBits() + 4 <= totalDataCount * 8) { + buffer.put(0, 4); + } // padding + + + while (buffer.getLengthInBits() % 8 != 0) { + buffer.putBit(false); + } // padding + + + while (true) { + if (buffer.getLengthInBits() >= totalDataCount * 8) { + break; + } + + buffer.put(PAD0, 8); + + if (buffer.getLengthInBits() >= totalDataCount * 8) { + break; + } + + buffer.put(PAD1, 8); + } + + return createBytes(buffer, rsBlocks); + }; + + _this.addData = function (data, mode) { + mode = mode || 'Byte'; + var newData = null; + + switch (mode) { + case 'Numeric': + newData = qrNumber(data); + break; + + case 'Alphanumeric': + newData = qrAlphaNum(data); + break; + + case 'Byte': + newData = qr8BitByte(data); + break; + + case 'Kanji': + newData = qrKanji(data); + break; + + default: + throw 'mode:' + mode; + } + + _dataList.push(newData); + + _dataCache = null; + }; + + _this.isDark = function (row, col) { + if (row < 0 || _moduleCount <= row || col < 0 || _moduleCount <= col) { + throw row + ',' + col; + } + + return _modules[row][col]; + }; + + _this.getModuleCount = function () { + return _moduleCount; + }; + + _this.make = function () { + if (_typeNumber < 1) { + var typeNumber = 1; + + for (; typeNumber < 40; typeNumber++) { + var rsBlocks = QRRSBlock.getRSBlocks(typeNumber, _errorCorrectionLevel); + var buffer = qrBitBuffer(); + + for (var i = 0; i < _dataList.length; i++) { + var data = _dataList[i]; + buffer.put(data.getMode(), 4); + buffer.put(data.getLength(), QRUtil.getLengthInBits(data.getMode(), typeNumber)); + data.write(buffer); + } + + var totalDataCount = 0; + + for (var i = 0; i < rsBlocks.length; i++) { + totalDataCount += rsBlocks[i].dataCount; + } + + if (buffer.getLengthInBits() <= totalDataCount * 8) { + break; + } + } + + _typeNumber = typeNumber; + } + + makeImpl(false, getBestMaskPattern()); + }; + + _this.createTableTag = function (cellSize, margin) { + cellSize = cellSize || 2; + margin = typeof margin == 'undefined' ? cellSize * 4 : margin; + var qrHtml = ''; + qrHtml += ''; + qrHtml += ''; + + for (var r = 0; r < _this.getModuleCount(); r += 1) { + qrHtml += ''; + + for (var c = 0; c < _this.getModuleCount(); c += 1) { + qrHtml += ''; + } + + qrHtml += ''; + qrHtml += '
'; + } + + qrHtml += '
'; + return qrHtml; + }; + + _this.createSvgTag = function (cellSize, margin, alt, title) { + var opts = {}; + + if (typeof arguments[0] == 'object') { + // Called by options. + opts = arguments[0]; // overwrite cellSize and margin. + + cellSize = opts.cellSize; + margin = opts.margin; + alt = opts.alt; + title = opts.title; + } + + cellSize = cellSize || 2; + margin = typeof margin == 'undefined' ? cellSize * 4 : margin; // Compose alt property surrogate + + alt = typeof alt === 'string' ? { + text: alt + } : alt || {}; + alt.text = alt.text || null; + alt.id = alt.text ? alt.id || 'qrcode-description' : null; // Compose title property surrogate + + title = typeof title === 'string' ? { + text: title + } : title || {}; + title.text = title.text || null; + title.id = title.text ? title.id || 'qrcode-title' : null; + var size = _this.getModuleCount() * cellSize + margin * 2; + var c, + mc, + r, + mr, + qrSvg = '', + rect; + rect = 'l' + cellSize + ',0 0,' + cellSize + ' -' + cellSize + ',0 0,-' + cellSize + 'z '; + qrSvg += '' + escapeXml(title.text) + '' : ''; + qrSvg += alt.text ? '' + escapeXml(alt.text) + '' : ''; + qrSvg += ''; + qrSvg += ''; + qrSvg += ''; + return qrSvg; + }; + + _this.createDataURL = function (cellSize, margin) { + cellSize = cellSize || 2; + margin = typeof margin == 'undefined' ? cellSize * 4 : margin; + var size = _this.getModuleCount() * cellSize + margin * 2; + var min = margin; + var max = size - margin; + return createDataURL(size, size, function (x, y) { + if (min <= x && x < max && min <= y && y < max) { + var c = Math.floor((x - min) / cellSize); + var r = Math.floor((y - min) / cellSize); + return _this.isDark(r, c) ? 0 : 1; + } else { + return 1; + } + }); + }; + + _this.createImgTag = function (cellSize, margin, alt) { + cellSize = cellSize || 2; + margin = typeof margin == 'undefined' ? cellSize * 4 : margin; + var size = _this.getModuleCount() * cellSize + margin * 2; + var img = ''; + img += '': + escaped += '>'; + break; + + case '&': + escaped += '&'; + break; + + case '"': + escaped += '"'; + break; + + default: + escaped += c; + break; + } + } + + return escaped; + }; + + var _createHalfASCII = function (margin) { + var cellSize = 1; + margin = typeof margin == 'undefined' ? cellSize * 2 : margin; + var size = _this.getModuleCount() * cellSize + margin * 2; + var min = margin; + var max = size - margin; + var y, x, r1, r2, p; + var blocks = { + '██': '█', + '█ ': '▀', + ' █': '▄', + ' ': ' ' + }; + var blocksLastLineNoMargin = { + '██': '▀', + '█ ': '▀', + ' █': ' ', + ' ': ' ' + }; + var ascii = ''; + + for (y = 0; y < size; y += 2) { + r1 = Math.floor((y - min) / cellSize); + r2 = Math.floor((y + 1 - min) / cellSize); + + for (x = 0; x < size; x += 1) { + p = '█'; + + if (min <= x && x < max && min <= y && y < max && _this.isDark(r1, Math.floor((x - min) / cellSize))) { + p = ' '; + } + + if (min <= x && x < max && min <= y + 1 && y + 1 < max && _this.isDark(r2, Math.floor((x - min) / cellSize))) { + p += ' '; + } else { + p += '█'; + } // Output 2 characters per pixel, to create full square. 1 character per pixels gives only half width of square. + + + ascii += margin < 1 && y + 1 >= max ? blocksLastLineNoMargin[p] : blocks[p]; + } + + ascii += '\n'; + } + + if (size % 2 && margin > 0) { + return ascii.substring(0, ascii.length - size - 1) + Array(size + 1).join('▀'); + } + + return ascii.substring(0, ascii.length - 1); + }; + + _this.createASCII = function (cellSize, margin) { + cellSize = cellSize || 1; + + if (cellSize < 2) { + return _createHalfASCII(margin); + } + + cellSize -= 1; + margin = typeof margin == 'undefined' ? cellSize * 2 : margin; + var size = _this.getModuleCount() * cellSize + margin * 2; + var min = margin; + var max = size - margin; + var y, x, r, p; + var white = Array(cellSize + 1).join('██'); + var black = Array(cellSize + 1).join(' '); + var ascii = ''; + var line = ''; + + for (y = 0; y < size; y += 1) { + r = Math.floor((y - min) / cellSize); + line = ''; + + for (x = 0; x < size; x += 1) { + p = 1; + + if (min <= x && x < max && min <= y && y < max && _this.isDark(r, Math.floor((x - min) / cellSize))) { + p = 0; + } // Output 2 characters per pixel, to create full square. 1 character per pixels gives only half width of square. + + + line += p ? white : black; + } + + for (r = 0; r < cellSize; r += 1) { + ascii += line + '\n'; + } + } + + return ascii.substring(0, ascii.length - 1); + }; + + _this.renderTo2dContext = function (context, cellSize) { + cellSize = cellSize || 2; + + var length = _this.getModuleCount(); + + for (var row = 0; row < length; row++) { + for (var col = 0; col < length; col++) { + context.fillStyle = _this.isDark(row, col) ? 'black' : 'white'; + context.fillRect(row * cellSize, col * cellSize, cellSize, cellSize); + } + } + }; + + return _this; + }; //--------------------------------------------------------------------- + // qrcode.stringToBytes + //--------------------------------------------------------------------- + + + qrcode.stringToBytesFuncs = { + 'default': function (s) { + var bytes = []; + + for (var i = 0; i < s.length; i += 1) { + var c = s.charCodeAt(i); + bytes.push(c & 0xff); + } + + return bytes; + } + }; + qrcode.stringToBytes = qrcode.stringToBytesFuncs['default']; //--------------------------------------------------------------------- + // qrcode.createStringToBytes + //--------------------------------------------------------------------- + + /** + * @param unicodeData base64 string of byte array. + * [16bit Unicode],[16bit Bytes], ... + * @param numChars + */ + + qrcode.createStringToBytes = function (unicodeData, numChars) { + // create conversion map. + var unicodeMap = function () { + var bin = base64DecodeInputStream(unicodeData); + + var read = function () { + var b = bin.read(); + if (b == -1) throw 'eof'; + return b; + }; + + var count = 0; + var unicodeMap = {}; + + while (true) { + var b0 = bin.read(); + if (b0 == -1) break; + var b1 = read(); + var b2 = read(); + var b3 = read(); + var k = String.fromCharCode(b0 << 8 | b1); + var v = b2 << 8 | b3; + unicodeMap[k] = v; + count += 1; + } + + if (count != numChars) { + throw count + ' != ' + numChars; + } + + return unicodeMap; + }(); + + var unknownChar = '?'.charCodeAt(0); + return function (s) { + var bytes = []; + + for (var i = 0; i < s.length; i += 1) { + var c = s.charCodeAt(i); + + if (c < 128) { + bytes.push(c); + } else { + var b = unicodeMap[s.charAt(i)]; + + if (typeof b == 'number') { + if ((b & 0xff) == b) { + // 1byte + bytes.push(b); + } else { + // 2bytes + bytes.push(b >>> 8); + bytes.push(b & 0xff); + } + } else { + bytes.push(unknownChar); + } + } + } + + return bytes; + }; + }; //--------------------------------------------------------------------- + // QRMode + //--------------------------------------------------------------------- + + + var QRMode = { + MODE_NUMBER: 1 << 0, + MODE_ALPHA_NUM: 1 << 1, + MODE_8BIT_BYTE: 1 << 2, + MODE_KANJI: 1 << 3 + }; //--------------------------------------------------------------------- + // QRErrorCorrectionLevel + //--------------------------------------------------------------------- + + var QRErrorCorrectionLevel = { + L: 1, + M: 0, + Q: 3, + H: 2 + }; //--------------------------------------------------------------------- + // QRMaskPattern + //--------------------------------------------------------------------- + + var QRMaskPattern = { + PATTERN000: 0, + PATTERN001: 1, + PATTERN010: 2, + PATTERN011: 3, + PATTERN100: 4, + PATTERN101: 5, + PATTERN110: 6, + PATTERN111: 7 + }; //--------------------------------------------------------------------- + // QRUtil + //--------------------------------------------------------------------- + + var QRUtil = function () { + var PATTERN_POSITION_TABLE = [[], [6, 18], [6, 22], [6, 26], [6, 30], [6, 34], [6, 22, 38], [6, 24, 42], [6, 26, 46], [6, 28, 50], [6, 30, 54], [6, 32, 58], [6, 34, 62], [6, 26, 46, 66], [6, 26, 48, 70], [6, 26, 50, 74], [6, 30, 54, 78], [6, 30, 56, 82], [6, 30, 58, 86], [6, 34, 62, 90], [6, 28, 50, 72, 94], [6, 26, 50, 74, 98], [6, 30, 54, 78, 102], [6, 28, 54, 80, 106], [6, 32, 58, 84, 110], [6, 30, 58, 86, 114], [6, 34, 62, 90, 118], [6, 26, 50, 74, 98, 122], [6, 30, 54, 78, 102, 126], [6, 26, 52, 78, 104, 130], [6, 30, 56, 82, 108, 134], [6, 34, 60, 86, 112, 138], [6, 30, 58, 86, 114, 142], [6, 34, 62, 90, 118, 146], [6, 30, 54, 78, 102, 126, 150], [6, 24, 50, 76, 102, 128, 154], [6, 28, 54, 80, 106, 132, 158], [6, 32, 58, 84, 110, 136, 162], [6, 26, 54, 82, 110, 138, 166], [6, 30, 58, 86, 114, 142, 170]]; + var G15 = 1 << 10 | 1 << 8 | 1 << 5 | 1 << 4 | 1 << 2 | 1 << 1 | 1 << 0; + var G18 = 1 << 12 | 1 << 11 | 1 << 10 | 1 << 9 | 1 << 8 | 1 << 5 | 1 << 2 | 1 << 0; + var G15_MASK = 1 << 14 | 1 << 12 | 1 << 10 | 1 << 4 | 1 << 1; + var _this = {}; + + var getBCHDigit = function (data) { + var digit = 0; + + while (data != 0) { + digit += 1; + data >>>= 1; + } + + return digit; + }; + + _this.getBCHTypeInfo = function (data) { + var d = data << 10; + + while (getBCHDigit(d) - getBCHDigit(G15) >= 0) { + d ^= G15 << getBCHDigit(d) - getBCHDigit(G15); + } + + return (data << 10 | d) ^ G15_MASK; + }; + + _this.getBCHTypeNumber = function (data) { + var d = data << 12; + + while (getBCHDigit(d) - getBCHDigit(G18) >= 0) { + d ^= G18 << getBCHDigit(d) - getBCHDigit(G18); + } + + return data << 12 | d; + }; + + _this.getPatternPosition = function (typeNumber) { + return PATTERN_POSITION_TABLE[typeNumber - 1]; + }; + + _this.getMaskFunction = function (maskPattern) { + switch (maskPattern) { + case QRMaskPattern.PATTERN000: + return function (i, j) { + return (i + j) % 2 == 0; + }; + + case QRMaskPattern.PATTERN001: + return function (i, j) { + return i % 2 == 0; + }; + + case QRMaskPattern.PATTERN010: + return function (i, j) { + return j % 3 == 0; + }; + + case QRMaskPattern.PATTERN011: + return function (i, j) { + return (i + j) % 3 == 0; + }; + + case QRMaskPattern.PATTERN100: + return function (i, j) { + return (Math.floor(i / 2) + Math.floor(j / 3)) % 2 == 0; + }; + + case QRMaskPattern.PATTERN101: + return function (i, j) { + return i * j % 2 + i * j % 3 == 0; + }; + + case QRMaskPattern.PATTERN110: + return function (i, j) { + return (i * j % 2 + i * j % 3) % 2 == 0; + }; + + case QRMaskPattern.PATTERN111: + return function (i, j) { + return (i * j % 3 + (i + j) % 2) % 2 == 0; + }; + + default: + throw 'bad maskPattern:' + maskPattern; + } + }; + + _this.getErrorCorrectPolynomial = function (errorCorrectLength) { + var a = qrPolynomial([1], 0); + + for (var i = 0; i < errorCorrectLength; i += 1) { + a = a.multiply(qrPolynomial([1, QRMath.gexp(i)], 0)); + } + + return a; + }; + + _this.getLengthInBits = function (mode, type) { + if (1 <= type && type < 10) { + // 1 - 9 + switch (mode) { + case QRMode.MODE_NUMBER: + return 10; + + case QRMode.MODE_ALPHA_NUM: + return 9; + + case QRMode.MODE_8BIT_BYTE: + return 8; + + case QRMode.MODE_KANJI: + return 8; + + default: + throw 'mode:' + mode; + } + } else if (type < 27) { + // 10 - 26 + switch (mode) { + case QRMode.MODE_NUMBER: + return 12; + + case QRMode.MODE_ALPHA_NUM: + return 11; + + case QRMode.MODE_8BIT_BYTE: + return 16; + + case QRMode.MODE_KANJI: + return 10; + + default: + throw 'mode:' + mode; + } + } else if (type < 41) { + // 27 - 40 + switch (mode) { + case QRMode.MODE_NUMBER: + return 14; + + case QRMode.MODE_ALPHA_NUM: + return 13; + + case QRMode.MODE_8BIT_BYTE: + return 16; + + case QRMode.MODE_KANJI: + return 12; + + default: + throw 'mode:' + mode; + } + } else { + throw 'type:' + type; + } + }; + + _this.getLostPoint = function (qrcode) { + var moduleCount = qrcode.getModuleCount(); + var lostPoint = 0; // LEVEL1 + + for (var row = 0; row < moduleCount; row += 1) { + for (var col = 0; col < moduleCount; col += 1) { + var sameCount = 0; + var dark = qrcode.isDark(row, col); + + for (var r = -1; r <= 1; r += 1) { + if (row + r < 0 || moduleCount <= row + r) { + continue; + } + + for (var c = -1; c <= 1; c += 1) { + if (col + c < 0 || moduleCount <= col + c) { + continue; + } + + if (r == 0 && c == 0) { + continue; + } + + if (dark == qrcode.isDark(row + r, col + c)) { + sameCount += 1; + } + } + } + + if (sameCount > 5) { + lostPoint += 3 + sameCount - 5; + } + } + } + + ; // LEVEL2 + + for (var row = 0; row < moduleCount - 1; row += 1) { + for (var col = 0; col < moduleCount - 1; col += 1) { + var count = 0; + if (qrcode.isDark(row, col)) count += 1; + if (qrcode.isDark(row + 1, col)) count += 1; + if (qrcode.isDark(row, col + 1)) count += 1; + if (qrcode.isDark(row + 1, col + 1)) count += 1; + + if (count == 0 || count == 4) { + lostPoint += 3; + } + } + } // LEVEL3 + + + for (var row = 0; row < moduleCount; row += 1) { + for (var col = 0; col < moduleCount - 6; col += 1) { + if (qrcode.isDark(row, col) && !qrcode.isDark(row, col + 1) && qrcode.isDark(row, col + 2) && qrcode.isDark(row, col + 3) && qrcode.isDark(row, col + 4) && !qrcode.isDark(row, col + 5) && qrcode.isDark(row, col + 6)) { + lostPoint += 40; + } + } + } + + for (var col = 0; col < moduleCount; col += 1) { + for (var row = 0; row < moduleCount - 6; row += 1) { + if (qrcode.isDark(row, col) && !qrcode.isDark(row + 1, col) && qrcode.isDark(row + 2, col) && qrcode.isDark(row + 3, col) && qrcode.isDark(row + 4, col) && !qrcode.isDark(row + 5, col) && qrcode.isDark(row + 6, col)) { + lostPoint += 40; + } + } + } // LEVEL4 + + + var darkCount = 0; + + for (var col = 0; col < moduleCount; col += 1) { + for (var row = 0; row < moduleCount; row += 1) { + if (qrcode.isDark(row, col)) { + darkCount += 1; + } + } + } + + var ratio = Math.abs(100 * darkCount / moduleCount / moduleCount - 50) / 5; + lostPoint += ratio * 10; + return lostPoint; + }; + + return _this; + }(); //--------------------------------------------------------------------- + // QRMath + //--------------------------------------------------------------------- + + + var QRMath = function () { + var EXP_TABLE = new Array(256); + var LOG_TABLE = new Array(256); // initialize tables + + for (var i = 0; i < 8; i += 1) { + EXP_TABLE[i] = 1 << i; + } + + for (var i = 8; i < 256; i += 1) { + EXP_TABLE[i] = EXP_TABLE[i - 4] ^ EXP_TABLE[i - 5] ^ EXP_TABLE[i - 6] ^ EXP_TABLE[i - 8]; + } + + for (var i = 0; i < 255; i += 1) { + LOG_TABLE[EXP_TABLE[i]] = i; + } + + var _this = {}; + + _this.glog = function (n) { + if (n < 1) { + throw 'glog(' + n + ')'; + } + + return LOG_TABLE[n]; + }; + + _this.gexp = function (n) { + while (n < 0) { + n += 255; + } + + while (n >= 256) { + n -= 255; + } + + return EXP_TABLE[n]; + }; + + return _this; + }(); //--------------------------------------------------------------------- + // qrPolynomial + //--------------------------------------------------------------------- + + + function qrPolynomial(num, shift) { + if (typeof num.length == 'undefined') { + throw num.length + '/' + shift; + } + + var _num = function () { + var offset = 0; + + while (offset < num.length && num[offset] == 0) { + offset += 1; + } + + var _num = new Array(num.length - offset + shift); + + for (var i = 0; i < num.length - offset; i += 1) { + _num[i] = num[i + offset]; + } + + return _num; + }(); + + var _this = {}; + + _this.getAt = function (index) { + return _num[index]; + }; + + _this.getLength = function () { + return _num.length; + }; + + _this.multiply = function (e) { + var num = new Array(_this.getLength() + e.getLength() - 1); + + for (var i = 0; i < _this.getLength(); i += 1) { + for (var j = 0; j < e.getLength(); j += 1) { + num[i + j] ^= QRMath.gexp(QRMath.glog(_this.getAt(i)) + QRMath.glog(e.getAt(j))); + } + } + + return qrPolynomial(num, 0); + }; + + _this.mod = function (e) { + if (_this.getLength() - e.getLength() < 0) { + return _this; + } + + var ratio = QRMath.glog(_this.getAt(0)) - QRMath.glog(e.getAt(0)); + var num = new Array(_this.getLength()); + + for (var i = 0; i < _this.getLength(); i += 1) { + num[i] = _this.getAt(i); + } + + for (var i = 0; i < e.getLength(); i += 1) { + num[i] ^= QRMath.gexp(QRMath.glog(e.getAt(i)) + ratio); + } // recursive call + + + return qrPolynomial(num, 0).mod(e); + }; + + return _this; + } + + ; //--------------------------------------------------------------------- + // QRRSBlock + //--------------------------------------------------------------------- + + var QRRSBlock = function () { + var RS_BLOCK_TABLE = [// L + // M + // Q + // H + // 1 + [1, 26, 19], [1, 26, 16], [1, 26, 13], [1, 26, 9], // 2 + [1, 44, 34], [1, 44, 28], [1, 44, 22], [1, 44, 16], // 3 + [1, 70, 55], [1, 70, 44], [2, 35, 17], [2, 35, 13], // 4 + [1, 100, 80], [2, 50, 32], [2, 50, 24], [4, 25, 9], // 5 + [1, 134, 108], [2, 67, 43], [2, 33, 15, 2, 34, 16], [2, 33, 11, 2, 34, 12], // 6 + [2, 86, 68], [4, 43, 27], [4, 43, 19], [4, 43, 15], // 7 + [2, 98, 78], [4, 49, 31], [2, 32, 14, 4, 33, 15], [4, 39, 13, 1, 40, 14], // 8 + [2, 121, 97], [2, 60, 38, 2, 61, 39], [4, 40, 18, 2, 41, 19], [4, 40, 14, 2, 41, 15], // 9 + [2, 146, 116], [3, 58, 36, 2, 59, 37], [4, 36, 16, 4, 37, 17], [4, 36, 12, 4, 37, 13], // 10 + [2, 86, 68, 2, 87, 69], [4, 69, 43, 1, 70, 44], [6, 43, 19, 2, 44, 20], [6, 43, 15, 2, 44, 16], // 11 + [4, 101, 81], [1, 80, 50, 4, 81, 51], [4, 50, 22, 4, 51, 23], [3, 36, 12, 8, 37, 13], // 12 + [2, 116, 92, 2, 117, 93], [6, 58, 36, 2, 59, 37], [4, 46, 20, 6, 47, 21], [7, 42, 14, 4, 43, 15], // 13 + [4, 133, 107], [8, 59, 37, 1, 60, 38], [8, 44, 20, 4, 45, 21], [12, 33, 11, 4, 34, 12], // 14 + [3, 145, 115, 1, 146, 116], [4, 64, 40, 5, 65, 41], [11, 36, 16, 5, 37, 17], [11, 36, 12, 5, 37, 13], // 15 + [5, 109, 87, 1, 110, 88], [5, 65, 41, 5, 66, 42], [5, 54, 24, 7, 55, 25], [11, 36, 12, 7, 37, 13], // 16 + [5, 122, 98, 1, 123, 99], [7, 73, 45, 3, 74, 46], [15, 43, 19, 2, 44, 20], [3, 45, 15, 13, 46, 16], // 17 + [1, 135, 107, 5, 136, 108], [10, 74, 46, 1, 75, 47], [1, 50, 22, 15, 51, 23], [2, 42, 14, 17, 43, 15], // 18 + [5, 150, 120, 1, 151, 121], [9, 69, 43, 4, 70, 44], [17, 50, 22, 1, 51, 23], [2, 42, 14, 19, 43, 15], // 19 + [3, 141, 113, 4, 142, 114], [3, 70, 44, 11, 71, 45], [17, 47, 21, 4, 48, 22], [9, 39, 13, 16, 40, 14], // 20 + [3, 135, 107, 5, 136, 108], [3, 67, 41, 13, 68, 42], [15, 54, 24, 5, 55, 25], [15, 43, 15, 10, 44, 16], // 21 + [4, 144, 116, 4, 145, 117], [17, 68, 42], [17, 50, 22, 6, 51, 23], [19, 46, 16, 6, 47, 17], // 22 + [2, 139, 111, 7, 140, 112], [17, 74, 46], [7, 54, 24, 16, 55, 25], [34, 37, 13], // 23 + [4, 151, 121, 5, 152, 122], [4, 75, 47, 14, 76, 48], [11, 54, 24, 14, 55, 25], [16, 45, 15, 14, 46, 16], // 24 + [6, 147, 117, 4, 148, 118], [6, 73, 45, 14, 74, 46], [11, 54, 24, 16, 55, 25], [30, 46, 16, 2, 47, 17], // 25 + [8, 132, 106, 4, 133, 107], [8, 75, 47, 13, 76, 48], [7, 54, 24, 22, 55, 25], [22, 45, 15, 13, 46, 16], // 26 + [10, 142, 114, 2, 143, 115], [19, 74, 46, 4, 75, 47], [28, 50, 22, 6, 51, 23], [33, 46, 16, 4, 47, 17], // 27 + [8, 152, 122, 4, 153, 123], [22, 73, 45, 3, 74, 46], [8, 53, 23, 26, 54, 24], [12, 45, 15, 28, 46, 16], // 28 + [3, 147, 117, 10, 148, 118], [3, 73, 45, 23, 74, 46], [4, 54, 24, 31, 55, 25], [11, 45, 15, 31, 46, 16], // 29 + [7, 146, 116, 7, 147, 117], [21, 73, 45, 7, 74, 46], [1, 53, 23, 37, 54, 24], [19, 45, 15, 26, 46, 16], // 30 + [5, 145, 115, 10, 146, 116], [19, 75, 47, 10, 76, 48], [15, 54, 24, 25, 55, 25], [23, 45, 15, 25, 46, 16], // 31 + [13, 145, 115, 3, 146, 116], [2, 74, 46, 29, 75, 47], [42, 54, 24, 1, 55, 25], [23, 45, 15, 28, 46, 16], // 32 + [17, 145, 115], [10, 74, 46, 23, 75, 47], [10, 54, 24, 35, 55, 25], [19, 45, 15, 35, 46, 16], // 33 + [17, 145, 115, 1, 146, 116], [14, 74, 46, 21, 75, 47], [29, 54, 24, 19, 55, 25], [11, 45, 15, 46, 46, 16], // 34 + [13, 145, 115, 6, 146, 116], [14, 74, 46, 23, 75, 47], [44, 54, 24, 7, 55, 25], [59, 46, 16, 1, 47, 17], // 35 + [12, 151, 121, 7, 152, 122], [12, 75, 47, 26, 76, 48], [39, 54, 24, 14, 55, 25], [22, 45, 15, 41, 46, 16], // 36 + [6, 151, 121, 14, 152, 122], [6, 75, 47, 34, 76, 48], [46, 54, 24, 10, 55, 25], [2, 45, 15, 64, 46, 16], // 37 + [17, 152, 122, 4, 153, 123], [29, 74, 46, 14, 75, 47], [49, 54, 24, 10, 55, 25], [24, 45, 15, 46, 46, 16], // 38 + [4, 152, 122, 18, 153, 123], [13, 74, 46, 32, 75, 47], [48, 54, 24, 14, 55, 25], [42, 45, 15, 32, 46, 16], // 39 + [20, 147, 117, 4, 148, 118], [40, 75, 47, 7, 76, 48], [43, 54, 24, 22, 55, 25], [10, 45, 15, 67, 46, 16], // 40 + [19, 148, 118, 6, 149, 119], [18, 75, 47, 31, 76, 48], [34, 54, 24, 34, 55, 25], [20, 45, 15, 61, 46, 16]]; + + var qrRSBlock = function (totalCount, dataCount) { + var _this = {}; + _this.totalCount = totalCount; + _this.dataCount = dataCount; + return _this; + }; + + var _this = {}; + + var getRsBlockTable = function (typeNumber, errorCorrectionLevel) { + switch (errorCorrectionLevel) { + case QRErrorCorrectionLevel.L: + return RS_BLOCK_TABLE[(typeNumber - 1) * 4 + 0]; + + case QRErrorCorrectionLevel.M: + return RS_BLOCK_TABLE[(typeNumber - 1) * 4 + 1]; + + case QRErrorCorrectionLevel.Q: + return RS_BLOCK_TABLE[(typeNumber - 1) * 4 + 2]; + + case QRErrorCorrectionLevel.H: + return RS_BLOCK_TABLE[(typeNumber - 1) * 4 + 3]; + + default: + return undefined; + } + }; + + _this.getRSBlocks = function (typeNumber, errorCorrectionLevel) { + var rsBlock = getRsBlockTable(typeNumber, errorCorrectionLevel); + + if (typeof rsBlock == 'undefined') { + throw 'bad rs block @ typeNumber:' + typeNumber + '/errorCorrectionLevel:' + errorCorrectionLevel; + } + + var length = rsBlock.length / 3; + var list = []; + + for (var i = 0; i < length; i += 1) { + var count = rsBlock[i * 3 + 0]; + var totalCount = rsBlock[i * 3 + 1]; + var dataCount = rsBlock[i * 3 + 2]; + + for (var j = 0; j < count; j += 1) { + list.push(qrRSBlock(totalCount, dataCount)); + } + } + + return list; + }; + + return _this; + }(); //--------------------------------------------------------------------- + // qrBitBuffer + //--------------------------------------------------------------------- + + + var qrBitBuffer = function () { + var _buffer = []; + var _length = 0; + var _this = {}; + + _this.getBuffer = function () { + return _buffer; + }; + + _this.getAt = function (index) { + var bufIndex = Math.floor(index / 8); + return (_buffer[bufIndex] >>> 7 - index % 8 & 1) == 1; + }; + + _this.put = function (num, length) { + for (var i = 0; i < length; i += 1) { + _this.putBit((num >>> length - i - 1 & 1) == 1); + } + }; + + _this.getLengthInBits = function () { + return _length; + }; + + _this.putBit = function (bit) { + var bufIndex = Math.floor(_length / 8); + + if (_buffer.length <= bufIndex) { + _buffer.push(0); + } + + if (bit) { + _buffer[bufIndex] |= 0x80 >>> _length % 8; + } + + _length += 1; + }; + + return _this; + }; //--------------------------------------------------------------------- + // qrNumber + //--------------------------------------------------------------------- + + + var qrNumber = function (data) { + var _mode = QRMode.MODE_NUMBER; + var _data = data; + var _this = {}; + + _this.getMode = function () { + return _mode; + }; + + _this.getLength = function (buffer) { + return _data.length; + }; + + _this.write = function (buffer) { + var data = _data; + var i = 0; + + while (i + 2 < data.length) { + buffer.put(strToNum(data.substring(i, i + 3)), 10); + i += 3; + } + + if (i < data.length) { + if (data.length - i == 1) { + buffer.put(strToNum(data.substring(i, i + 1)), 4); + } else if (data.length - i == 2) { + buffer.put(strToNum(data.substring(i, i + 2)), 7); + } + } + }; + + var strToNum = function (s) { + var num = 0; + + for (var i = 0; i < s.length; i += 1) { + num = num * 10 + chatToNum(s.charAt(i)); + } + + return num; + }; + + var chatToNum = function (c) { + if ('0' <= c && c <= '9') { + return c.charCodeAt(0) - '0'.charCodeAt(0); + } + + throw 'illegal char :' + c; + }; + + return _this; + }; //--------------------------------------------------------------------- + // qrAlphaNum + //--------------------------------------------------------------------- + + + var qrAlphaNum = function (data) { + var _mode = QRMode.MODE_ALPHA_NUM; + var _data = data; + var _this = {}; + + _this.getMode = function () { + return _mode; + }; + + _this.getLength = function (buffer) { + return _data.length; + }; + + _this.write = function (buffer) { + var s = _data; + var i = 0; + + while (i + 1 < s.length) { + buffer.put(getCode(s.charAt(i)) * 45 + getCode(s.charAt(i + 1)), 11); + i += 2; + } + + if (i < s.length) { + buffer.put(getCode(s.charAt(i)), 6); + } + }; + + var getCode = function (c) { + if ('0' <= c && c <= '9') { + return c.charCodeAt(0) - '0'.charCodeAt(0); + } else if ('A' <= c && c <= 'Z') { + return c.charCodeAt(0) - 'A'.charCodeAt(0) + 10; + } else { + switch (c) { + case ' ': + return 36; + + case '$': + return 37; + + case '%': + return 38; + + case '*': + return 39; + + case '+': + return 40; + + case '-': + return 41; + + case '.': + return 42; + + case '/': + return 43; + + case ':': + return 44; + + default: + throw 'illegal char :' + c; + } + } + }; + + return _this; + }; //--------------------------------------------------------------------- + // qr8BitByte + //--------------------------------------------------------------------- + + + var qr8BitByte = function (data) { + var _mode = QRMode.MODE_8BIT_BYTE; + var _data = data; + + var _bytes = qrcode.stringToBytes(data); + + var _this = {}; + + _this.getMode = function () { + return _mode; + }; + + _this.getLength = function (buffer) { + return _bytes.length; + }; + + _this.write = function (buffer) { + for (var i = 0; i < _bytes.length; i += 1) { + buffer.put(_bytes[i], 8); + } + }; + + return _this; + }; //--------------------------------------------------------------------- + // qrKanji + //--------------------------------------------------------------------- + + + var qrKanji = function (data) { + var _mode = QRMode.MODE_KANJI; + var _data = data; + var stringToBytes = qrcode.stringToBytesFuncs['SJIS']; + + if (!stringToBytes) { + throw 'sjis not supported.'; + } + + !function (c, code) { + // self test for sjis support. + var test = stringToBytes(c); + + if (test.length != 2 || (test[0] << 8 | test[1]) != code) { + throw 'sjis not supported.'; + } + }('\u53cb', 0x9746); + + var _bytes = stringToBytes(data); + + var _this = {}; + + _this.getMode = function () { + return _mode; + }; + + _this.getLength = function (buffer) { + return ~~(_bytes.length / 2); + }; + + _this.write = function (buffer) { + var data = _bytes; + var i = 0; + + while (i + 1 < data.length) { + var c = (0xff & data[i]) << 8 | 0xff & data[i + 1]; + + if (0x8140 <= c && c <= 0x9FFC) { + c -= 0x8140; + } else if (0xE040 <= c && c <= 0xEBBF) { + c -= 0xC140; + } else { + throw 'illegal char at ' + (i + 1) + '/' + c; + } + + c = (c >>> 8 & 0xff) * 0xC0 + (c & 0xff); + buffer.put(c, 13); + i += 2; + } + + if (i < data.length) { + throw 'illegal char at ' + (i + 1); + } + }; + + return _this; + }; //===================================================================== + // GIF Support etc. + // + //--------------------------------------------------------------------- + // byteArrayOutputStream + //--------------------------------------------------------------------- + + + var byteArrayOutputStream = function () { + var _bytes = []; + var _this = {}; + + _this.writeByte = function (b) { + _bytes.push(b & 0xff); + }; + + _this.writeShort = function (i) { + _this.writeByte(i); + + _this.writeByte(i >>> 8); + }; + + _this.writeBytes = function (b, off, len) { + off = off || 0; + len = len || b.length; + + for (var i = 0; i < len; i += 1) { + _this.writeByte(b[i + off]); + } + }; + + _this.writeString = function (s) { + for (var i = 0; i < s.length; i += 1) { + _this.writeByte(s.charCodeAt(i)); + } + }; + + _this.toByteArray = function () { + return _bytes; + }; + + _this.toString = function () { + var s = ''; + s += '['; + + for (var i = 0; i < _bytes.length; i += 1) { + if (i > 0) { + s += ','; + } + + s += _bytes[i]; + } + + s += ']'; + return s; + }; + + return _this; + }; //--------------------------------------------------------------------- + // base64EncodeOutputStream + //--------------------------------------------------------------------- + + + var base64EncodeOutputStream = function () { + var _buffer = 0; + var _buflen = 0; + var _length = 0; + var _base64 = ''; + var _this = {}; + + var writeEncoded = function (b) { + _base64 += String.fromCharCode(encode(b & 0x3f)); + }; + + var encode = function (n) { + if (n < 0) {// error. + } else if (n < 26) { + return 0x41 + n; + } else if (n < 52) { + return 0x61 + (n - 26); + } else if (n < 62) { + return 0x30 + (n - 52); + } else if (n == 62) { + return 0x2b; + } else if (n == 63) { + return 0x2f; + } + + throw 'n:' + n; + }; + + _this.writeByte = function (n) { + _buffer = _buffer << 8 | n & 0xff; + _buflen += 8; + _length += 1; + + while (_buflen >= 6) { + writeEncoded(_buffer >>> _buflen - 6); + _buflen -= 6; + } + }; + + _this.flush = function () { + if (_buflen > 0) { + writeEncoded(_buffer << 6 - _buflen); + _buffer = 0; + _buflen = 0; + } + + if (_length % 3 != 0) { + // padding + var padlen = 3 - _length % 3; + + for (var i = 0; i < padlen; i += 1) { + _base64 += '='; + } + } + }; + + _this.toString = function () { + return _base64; + }; + + return _this; + }; //--------------------------------------------------------------------- + // base64DecodeInputStream + //--------------------------------------------------------------------- + + + var base64DecodeInputStream = function (str) { + var _str = str; + var _pos = 0; + var _buffer = 0; + var _buflen = 0; + var _this = {}; + + _this.read = function () { + while (_buflen < 8) { + if (_pos >= _str.length) { + if (_buflen == 0) { + return -1; + } + + throw 'unexpected end of file./' + _buflen; + } + + var c = _str.charAt(_pos); + + _pos += 1; + + if (c == '=') { + _buflen = 0; + return -1; + } else if (c.match(/^\s$/)) { + // ignore if whitespace. + continue; + } + + _buffer = _buffer << 6 | decode(c.charCodeAt(0)); + _buflen += 6; + } + + var n = _buffer >>> _buflen - 8 & 0xff; + _buflen -= 8; + return n; + }; + + var decode = function (c) { + if (0x41 <= c && c <= 0x5a) { + return c - 0x41; + } else if (0x61 <= c && c <= 0x7a) { + return c - 0x61 + 26; + } else if (0x30 <= c && c <= 0x39) { + return c - 0x30 + 52; + } else if (c == 0x2b) { + return 62; + } else if (c == 0x2f) { + return 63; + } else { + throw 'c:' + c; + } + }; + + return _this; + }; //--------------------------------------------------------------------- + // gifImage (B/W) + //--------------------------------------------------------------------- + + + var gifImage = function (width, height) { + var _width = width; + var _height = height; + + var _data = new Array(width * height); + + var _this = {}; + + _this.setPixel = function (x, y, pixel) { + _data[y * _width + x] = pixel; + }; + + _this.write = function (out) { + //--------------------------------- + // GIF Signature + out.writeString('GIF87a'); //--------------------------------- + // Screen Descriptor + + out.writeShort(_width); + out.writeShort(_height); + out.writeByte(0x80); // 2bit + + out.writeByte(0); + out.writeByte(0); //--------------------------------- + // Global Color Map + // black + + out.writeByte(0x00); + out.writeByte(0x00); + out.writeByte(0x00); // white + + out.writeByte(0xff); + out.writeByte(0xff); + out.writeByte(0xff); //--------------------------------- + // Image Descriptor + + out.writeString(','); + out.writeShort(0); + out.writeShort(0); + out.writeShort(_width); + out.writeShort(_height); + out.writeByte(0); //--------------------------------- + // Local Color Map + //--------------------------------- + // Raster Data + + var lzwMinCodeSize = 2; + var raster = getLZWRaster(lzwMinCodeSize); + out.writeByte(lzwMinCodeSize); + var offset = 0; + + while (raster.length - offset > 255) { + out.writeByte(255); + out.writeBytes(raster, offset, 255); + offset += 255; + } + + out.writeByte(raster.length - offset); + out.writeBytes(raster, offset, raster.length - offset); + out.writeByte(0x00); //--------------------------------- + // GIF Terminator + + out.writeString(';'); + }; + + var bitOutputStream = function (out) { + var _out = out; + var _bitLength = 0; + var _bitBuffer = 0; + var _this = {}; + + _this.write = function (data, length) { + if (data >>> length != 0) { + throw 'length over'; + } + + while (_bitLength + length >= 8) { + _out.writeByte(0xff & (data << _bitLength | _bitBuffer)); + + length -= 8 - _bitLength; + data >>>= 8 - _bitLength; + _bitBuffer = 0; + _bitLength = 0; + } + + _bitBuffer = data << _bitLength | _bitBuffer; + _bitLength = _bitLength + length; + }; + + _this.flush = function () { + if (_bitLength > 0) { + _out.writeByte(_bitBuffer); + } + }; + + return _this; + }; + + var getLZWRaster = function (lzwMinCodeSize) { + var clearCode = 1 << lzwMinCodeSize; + var endCode = (1 << lzwMinCodeSize) + 1; + var bitLength = lzwMinCodeSize + 1; // Setup LZWTable + + var table = lzwTable(); + + for (var i = 0; i < clearCode; i += 1) { + table.add(String.fromCharCode(i)); + } + + table.add(String.fromCharCode(clearCode)); + table.add(String.fromCharCode(endCode)); + var byteOut = byteArrayOutputStream(); + var bitOut = bitOutputStream(byteOut); // clear code + + bitOut.write(clearCode, bitLength); + var dataIndex = 0; + var s = String.fromCharCode(_data[dataIndex]); + dataIndex += 1; + + while (dataIndex < _data.length) { + var c = String.fromCharCode(_data[dataIndex]); + dataIndex += 1; + + if (table.contains(s + c)) { + s = s + c; + } else { + bitOut.write(table.indexOf(s), bitLength); + + if (table.size() < 0xfff) { + if (table.size() == 1 << bitLength) { + bitLength += 1; + } + + table.add(s + c); + } + + s = c; + } + } + + bitOut.write(table.indexOf(s), bitLength); // end code + + bitOut.write(endCode, bitLength); + bitOut.flush(); + return byteOut.toByteArray(); + }; + + var lzwTable = function () { + var _map = {}; + var _size = 0; + var _this = {}; + + _this.add = function (key) { + if (_this.contains(key)) { + throw 'dup key:' + key; + } + + _map[key] = _size; + _size += 1; + }; + + _this.size = function () { + return _size; + }; + + _this.indexOf = function (key) { + return _map[key]; + }; + + _this.contains = function (key) { + return typeof _map[key] != 'undefined'; + }; + + return _this; + }; + + return _this; + }; + + var createDataURL = function (width, height, getPixel) { + var gif = gifImage(width, height); + + for (var y = 0; y < height; y += 1) { + for (var x = 0; x < width; x += 1) { + gif.setPixel(x, y, getPixel(x, y)); + } + } + + var b = byteArrayOutputStream(); + gif.write(b); + var base64 = base64EncodeOutputStream(); + var bytes = b.toByteArray(); + + for (var i = 0; i < bytes.length; i += 1) { + base64.writeByte(bytes[i]); + } + + base64.flush(); + return 'data:image/gif;base64,' + base64; + }; //--------------------------------------------------------------------- + // returns qrcode function. + + + return qrcode; +}(); // multibyte support + + +!function () { + qrcode.stringToBytesFuncs['UTF-8'] = function (s) { + // http://stackoverflow.com/questions/18729405/how-to-convert-utf8-string-to-byte-array + function toUTF8Array(str) { + var utf8 = []; + + for (var i = 0; i < str.length; i++) { + var charcode = str.charCodeAt(i); + if (charcode < 0x80) utf8.push(charcode);else if (charcode < 0x800) { + utf8.push(0xc0 | charcode >> 6, 0x80 | charcode & 0x3f); + } else if (charcode < 0xd800 || charcode >= 0xe000) { + utf8.push(0xe0 | charcode >> 12, 0x80 | charcode >> 6 & 0x3f, 0x80 | charcode & 0x3f); + } // surrogate pair + else { + i++; // UTF-16 encodes 0x10000-0x10FFFF by + // subtracting 0x10000 and splitting the + // 20 bits of 0x0-0xFFFFF into two halves + + charcode = 0x10000 + ((charcode & 0x3ff) << 10 | str.charCodeAt(i) & 0x3ff); + utf8.push(0xf0 | charcode >> 18, 0x80 | charcode >> 12 & 0x3f, 0x80 | charcode >> 6 & 0x3f, 0x80 | charcode & 0x3f); + } + } + + return utf8; + } + + return toUTF8Array(s); + }; +}(); + +(function (factory) { + if (typeof define === 'function' && define.amd) { + define([], factory); + } else if (typeof exports === 'object') { + module.exports = factory(); + } +})(function () { + return qrcode; +}); + +},{}]},{},[1])(1) +}); diff --git a/node_modules/bech32-buffer/index.d.ts b/node_modules/bech32-buffer/index.d.ts new file mode 100644 index 0000000..48356c9 --- /dev/null +++ b/node_modules/bech32-buffer/index.d.ts @@ -0,0 +1,180 @@ +/** + * Virtual type for an array in which each element represents 5 bits. + */ +export declare type FiveBitArray = Uint8Array; +// As TypeScript (unlike Flow) always uses structural typing for determining type compatibility, +// there seems to be no way to express this better. + +/** + * Encoding from the Bech32 family used during encoding / decoding. + */ +export declare type Encoding = 'bech32' | 'bech32m'; + +/** + * Converts a Uint8Array into a Uint8Array variant, in which each element + * encodes 5 bits of the original byte array. + * + * @param src + * Input to convert + * @param dst + * Optional output buffer. If specified, the sequence of 5-bit chunks will be written there; + * if not specified, the output buffer will be created from scratch. The length + * of `outBuffer` is not checked. + * @returns + * Output buffer consisting of 5-bit chunks + */ +export declare function to5BitArray(src: Uint8Array, dst?: FiveBitArray): FiveBitArray; + +/** + * Converts a sequence of 5-bit chunks into an ordinary Uint8Array. + * + * @param src + * Input to convert + * @param dst + * Optional output buffer. If specified, the converted bytes will be written there; + * if not specified, the output buffer will be created from scratch. The length + * of `outBuffer` is not checked. + * @returns + * Output buffer + */ +export declare function from5BitArray(src: FiveBitArray, dst?: Uint8Array): Uint8Array; + +/** + * Encodes binary data into Bech32 encoding. + * + * The case is preserved: if the prefix is uppercase, then the output will be uppercase + * as well; otherwise, the output will be lowercase (including the case when the prefix does + * not contain any letters). + * + * Ordinarily, you may want to use `encode` because it converts + * binary data to an array of 5-bit integers automatically. + * + * @param prefix + * Human-readable prefix to place at the beginning of the encoding + * @param data + * Array of 5-bit integers with data to encode + * @param encoding + * Encoding to use; influences the checksum computation. If not specified, + * Bech32 encoding will be used. + * @returns + * Bech32(m) encoding of data in the form `1` + * @throws If the prefix is mixed-case or contains chars that are not eligible for Bech32 encoding + */ +export declare function encode5BitArray( + prefix: string, + data: FiveBitArray, + encoding?: Encoding +): string; + +/** + * Result of a decoding operation. + */ +export interface DecodeResult { + /** Human-readable prefix. */ + prefix: string; + /** Variation of the Bech32 encoding inferred from the checksum. */ + encoding: Encoding; + /** Decoded data. */ + data: T; +} + +/** + * Decodes data from Bech32 encoding into an array of 5-bit integers. + * + * Ordinarily, you may want to use `decode` because it automatically + * converts the array of 5-bit integers into an ordinary `Uint8Array`. + * + * @param message + * Bech32-encoded message + * @returns + * Decoded object with `prefix` and `data` fields, which contain the human-readable + * prefix and the array of 5-bit integers respectively. + */ +export declare function decodeTo5BitArray(message: string): DecodeResult; + +/** + * Encodes binary data into Bech32 encoding. + * + * The case is preserved: if the prefix is uppercase, then the output will be uppercase + * as well; otherwise, the output will be lowercase (including the case when the prefix does + * not contain any letters). + * + * @param prefix + * Human-readable prefix to place at the beginning of the encoding + * @param data + * Binary data to encode + * @param encoding + * Encoding to use; influences the checksum computation. If not specified, + * Bech32 encoding will be used. + * @returns + * Bech32 encoding of data in the form `1` + * @throws If the prefix is mixed-case or contains chars that are not eligible for Bech32 encoding + */ +export declare function encode( + prefix: string, + data: Uint8Array, + encoding?: Encoding +): string; + +/** + * Decodes data from Bech32 encoding into an array of 5-bit integers. + * + * @param message + * Bech32-encoded message + * @returns + * Decoded object with `prefix` and `data` fields, which contain the human-readable + * prefix and the decoded binary data respectively. + */ +export declare function decode(message: string): DecodeResult; + +/** + * Bitcoin address. + */ +export declare class BitcoinAddress { + /** + * Human-readable prefix. Equals `'bc'` (for mainnet addresses) + * or `'tb'` (for testnet addresses). + */ + prefix: 'bc' | 'tb'; + /** + * Script version. An integer between 0 and 16 (inclusive). + */ + scriptVersion: number; + /** + * Script data. A byte string with length 2 to 40 (inclusive). + */ + data: Uint8Array; + + /** + * Decodes a Bitcoin address from a Bech32 string. + * As per BIP 350, the original encoding is expected for version 0 scripts, while + * other script versions expect the modified encoding. + * + * This method does not check whether the address is well-formed; + * use `type()` method on returned address to find that out. + */ + static decode(message: string): BitcoinAddress; + + /** + * Creates a new address based on provided data. + * + * Validation is performed on the fields as specified in their description (e.g., + * it is checked that `scriptVersion` is between 0 and 16). Additionally, + * for `scriptVersion == 0` it is checked that `data` is either 20 or 32 bytes long. + * + * @throws {Error} If provided fields do not pass validation. + */ + constructor(prefix: 'bc' | 'tb', scriptVersion: number, data: Uint8Array); + + /** + * Encodes this address in Bech32 or Bech32m format, depending on the script version. + * Version 0 scripts are encoded using original Bech32 encoding as per BIP 173, + * while versions 1-16 are encoded using the modified encoding as per BIP 350. + */ + encode(): string; + + /** + * Guesses the address type based on its internal structure. + */ + type(): void | 'p2wsh' | 'p2wpkh'; +} diff --git a/node_modules/bech32-buffer/lib/bit-converter.js b/node_modules/bech32-buffer/lib/bit-converter.js new file mode 100644 index 0000000..3c76160 --- /dev/null +++ b/node_modules/bech32-buffer/lib/bit-converter.js @@ -0,0 +1,115 @@ +"use strict"; + +Object.defineProperty(exports, "__esModule", { + value: true +}); +exports.createBitArray = createBitArray; +exports.fromBits = fromBits; +exports.toBits = toBits; + +/* eslint-disable no-unused-vars */ + +/** + * Virtual type for bit arrays, i.e., arrays in which each element contains + * an integer in range `[0, 1 << L)`, where `1 <= L <= 8`. + */ + +/* eslint-enable no-unused-vars */ + +/** + * Performs unchecked conversion from `Uint8Array` to `BitArray`. + * This function is translated as the indentity operation by Babel; it's needed purely + * for Flow type checks. + * + * @param {Uint8Array} src + * array to convert + * @returns {Uint8Array} + * `src` interpreted as a `BitArray` with the specified bitness + * + * @api private + */ +function toBitArrayUnchecked(src) { + return src; +} +/** + * Creates a new array with specified bitness. + * + * @param {number} len + * length of the created array + * @returns {Uint8Array} + * + * @api private + */ + + +function createBitArray(len) { + return toBitArrayUnchecked(new Uint8Array(len)); +} +/** + * Converts an array from one number of bits per element to another. + * + * @api private + */ + + +function convert(src, srcBits, dst, dstBits, pad) { + var mask = (1 << dstBits) - 1; + var acc = 0; + var bits = 0; + var pos = 0; + src.forEach(function (b) { + // Pull next bits from the input buffer into accumulator. + acc = (acc << srcBits) + b; + bits += srcBits; // Push into the output buffer while there are enough bits in the accumulator. + + while (bits >= dstBits) { + bits -= dstBits; + dst[pos] = acc >> bits & mask; + pos += 1; + } + }); + + if (pad) { + if (bits > 0) { + // `dstBits - rem.bits` is the number of trailing zero bits needed to be appended + // to accumulator bits to get the trailing bit group. + dst[pos] = acc << dstBits - bits & mask; + } + } else { + // Truncate the remaining padding, but make sure that it is zeroed and not + // overly long first. + if (bits >= srcBits) { + throw new Error("Excessive padding: ".concat(bits, " (max ").concat(srcBits - 1, " allowed)")); + } + + if (acc % (1 << bits) !== 0) { + throw new Error('Non-zero padding'); + } + } +} +/** + * Encodes a `Uint8Array` buffer as an array with a lesser number of bits per element. + * + * @api private + */ + + +function toBits(src, bits, dst) { + if (bits > 8 || bits < 1) { + throw new RangeError('Invalid bits per element; 1 to 8 expected'); + } // `BitArray<8>` is equivalent to `Uint8Array`; unfortunately, Flow + // has problems expressing this, so the explicit conversion is performed here. + + + convert(toBitArrayUnchecked(src), 8, dst, bits, true); + return dst; +} + +function fromBits(src, bits, dst) { + if (bits > 8 || bits < 1) { + throw new RangeError('Invalid bits per element; 1 to 8 expected'); + } + + convert(src, bits, toBitArrayUnchecked(dst), 8, false); + return dst; +} \ No newline at end of file diff --git a/node_modules/bech32-buffer/lib/encoding.js b/node_modules/bech32-buffer/lib/encoding.js new file mode 100644 index 0000000..2829f22 --- /dev/null +++ b/node_modules/bech32-buffer/lib/encoding.js @@ -0,0 +1,217 @@ +"use strict"; + +Object.defineProperty(exports, "__esModule", { + value: true +}); +exports.CHECKSUM_LENGTH = void 0; +exports.createChecksum = createChecksum; +exports.decode = decode; +exports.decodeWithPrefix = decodeWithPrefix; +exports.detectCase = detectCase; +exports.encode = encode; +exports.expandPrefix = expandPrefix; +exports.verifyChecksum = verifyChecksum; + +var _bitConverter = require("./bit-converter"); + +// Alphabet for Bech32 +var CHARSET = 'qpzry9x8gf2tvdw0s3jn54khce6mua7l'; // Checksum constant for Bech32m. + +var BECH32M_CHECKSUM = 0x2bc830a3; // Minimum char code that could be present in the encoded message + +var MIN_CHAR_CODE = 33; // Maximum char code that could be present in the encoded message + +var MAX_CHAR_CODE = 126; +var CHECKSUM_LENGTH = 6; // Reverse lookup for characters + +exports.CHECKSUM_LENGTH = CHECKSUM_LENGTH; + +var CHAR_LOOKUP = function () { + var lookup = new Map(); + + for (var i = 0; i < CHARSET.length; i += 1) { + lookup.set(CHARSET[i], i); + } + + return lookup; +}(); // Poly generators + + +var GEN = [0x3b6a57b2, 0x26508e6d, 0x1ea119fa, 0x3d4233dd, 0x2a1462b3]; + +function polymod(values) { + return values.reduce(function (checksum, value) { + var bits = checksum >> 25; + var newChecksum = (checksum & 0x1ffffff) << 5 ^ value; + return GEN.reduce(function (chk, gen, i) { + return (bits >> i & 1) === 0 ? chk : chk ^ gen; + }, newChecksum); + }, + /* initial checksum */ + 1); +} +/** + * Expands a prefix into the specified output buffer. + */ + + +function expandPrefix(prefix, outBuffer) { + for (var i = 0; i < prefix.length; i += 1) { + var code = prefix.charCodeAt(i); + outBuffer[i] = code >> 5; + outBuffer[i + prefix.length + 1] = code & 31; + } + + outBuffer[prefix.length] = 0; +} +/** + * Verifies the checksum for a particular buffer. + */ + + +function verifyChecksum(buffer) { + switch (polymod(buffer)) { + case 1: + return 'bech32'; + + case BECH32M_CHECKSUM: + return 'bech32m'; + + default: + return undefined; + } +} +/** + * Creates a checksum for a buffer and writes it to the last 6 5-bit groups + * of the buffer. + */ + + +function createChecksum(buffer, encoding) { + var checksumConstant; + + switch (encoding) { + case 'bech32': + checksumConstant = 1; + break; + + case 'bech32m': + checksumConstant = BECH32M_CHECKSUM; + break; + + default: + throw Error("Invalid encoding value: ".concat(encoding, "; expected bech32 or bech32m")); + } + + var mod = polymod(buffer) ^ checksumConstant; + + for (var i = 0; i < CHECKSUM_LENGTH; i += 1) { + var shift = 5 * (5 - i); + buffer[buffer.length - CHECKSUM_LENGTH + i] = mod >> shift & 31; + } +} +/** + * Encodes an array of 5-bit groups into a string. + * + * @param {Uint8Array} buffer + * @returns {string} + * + * @api private + */ + + +function encode(buffer) { + return buffer.reduce(function (acc, bits) { + return acc + CHARSET[bits]; + }, ''); +} +/** + * Decodes a string into an array of 5-bit groups. + * + * @param {string} message + * @param {Uint8Array} [dst] + * Optional array to write the output to. If not specified, the array is created. + * @returns {Uint8Array} + * Array with the result of decoding + * + * @throws {Error} + * if there are characters in `message` not present in the encoding alphabet + * + * @api private + */ + + +function decode(message, dst) { + var realDst = dst || (0, _bitConverter.createBitArray)(message.length); + + for (var i = 0; i < message.length; i += 1) { + var idx = CHAR_LOOKUP.get(message[i]); + + if (idx === undefined) { + throw new Error("Invalid char in message: ".concat(message[i])); + } + + realDst[i] = idx; + } + + return realDst; +} +/** + * Decodes a string and a human-readable prefix into an array of 5-bit groups. + * The prefix is expanded as specified by Bech32. + * + * @param {string} prefix + * @param {string} message + * @returns {Uint8Array} + * Array with the result of decoding + * + * @api private + */ + + +function decodeWithPrefix(prefix, message) { + var len = message.length + 2 * prefix.length + 1; + var dst = (0, _bitConverter.createBitArray)(len); + expandPrefix(prefix, dst.subarray(0, 2 * prefix.length + 1)); + decode(message, dst.subarray(2 * prefix.length + 1)); + return dst; +} +/** + * Detects the character case used in `message`. If the message doesn't + * contain either lower-case or upper-case chars, returns `null`. + * + * @param {string} message + * @param {string} messageDescription + * Human-readable description of the message to put into an error message, should any occur + * @returns {'lower'|'upper'|null} + * @throws if the message contains both lowercase and uppercase chars, + * or contains chars not valid for Bech32(m) encoding + */ + + +function detectCase(message) { + var messageDescription = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'message'; + var hasLowerCase = false; + var hasUpperCase = false; + + for (var i = 0; i < message.length; i += 1) { + var ord = message.charCodeAt(i); // 3. Allowed chars in the encoding + + if (ord < MIN_CHAR_CODE || ord > MAX_CHAR_CODE) { + throw new TypeError("Invalid char in ".concat(messageDescription, ": ").concat(ord, "; ") + "should be in ASCII range ".concat(MIN_CHAR_CODE, "-").concat(MAX_CHAR_CODE)); + } + + hasUpperCase = hasUpperCase || ord >= 65 && ord <= 90; + hasLowerCase = hasLowerCase || ord >= 97 && ord <= 122; + } + + if (hasLowerCase && hasUpperCase) { + throw new TypeError("Mixed-case ".concat(messageDescription)); + } else if (hasUpperCase) { + return 'upper'; + } else if (hasLowerCase) { + return 'lower'; + } else { + return null; + } +} \ No newline at end of file diff --git a/node_modules/bech32-buffer/lib/index.js b/node_modules/bech32-buffer/lib/index.js new file mode 100644 index 0000000..1f5e7ed --- /dev/null +++ b/node_modules/bech32-buffer/lib/index.js @@ -0,0 +1,349 @@ +"use strict"; + +Object.defineProperty(exports, "__esModule", { + value: true +}); +exports.BitcoinAddress = void 0; +exports.decode = decode; +exports.decodeTo5BitArray = decodeTo5BitArray; +exports.encode = encode; +exports.encode5BitArray = encode5BitArray; +exports.from5BitArray = from5BitArray; +exports.to5BitArray = to5BitArray; + +var _bitConverter = require("./bit-converter"); + +var _encoding = require("./encoding"); + +function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } + +function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } + +function _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); Object.defineProperty(Constructor, "prototype", { writable: false }); return Constructor; } + +// Maximum encoded message length +var MAX_ENC_LENGTH = 90; + +/** + * Converts a Uint8Array into a Uint8Array variant, in which each element + * encodes 5 bits of the original byte array. + * + * @param {Uint8Array} src + * Input to convert + * @param {?Uint8Array} dst + * Optional output buffer. If specified, the sequence of 5-bit chunks will be written there; + * if not specified, the output buffer will be created from scratch. The length + * of `outBuffer` is not checked. + * @returns {Uint8Array} + * Output buffer consisting of 5-bit chunks + * + * @api public + */ +function to5BitArray(src, dst) { + var len = Math.ceil(src.length * 8 / 5); + var realDst = dst || (0, _bitConverter.createBitArray)(len); + return (0, _bitConverter.toBits)(src, 5, realDst); +} + +function from5BitArray(src, dst) { + var len = Math.floor(src.length * 5 / 8); + var realDst = dst || new Uint8Array(len); + return (0, _bitConverter.fromBits)(src, 5, realDst); +} +/** + * Encodes binary data into Bech32 encoding. + * + * The case is preserved: if the prefix is uppercase, then the output will be uppercase + * as well; otherwise, the output will be lowercase (including the case when the prefix does + * not contain any letters). + * + * Ordinarily, you may want to use [`encode`](#encode) because it converts + * binary data to an array of 5-bit integers automatically. + * + * @param {string} prefix + * Human-readable prefix to place at the beginning of the encoding + * @param {Uint8Array} data + * Array of 5-bit integers with data to encode + * @param {Encoding} encoding + * Encoding to use; influences the checksum computation. If not specified, + * Bech32 encoding will be used. + * @returns {string} + * Bech32 encoding of data in the form `1` + * @throws If the prefix is mixed-case or contains chars that are not eligible for Bech32 encoding + * + * @api public + */ + + +function encode5BitArray(prefix, data) { + var _detectCase; + + var encoding = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 'bech32'; + // 1. Allocate buffer for all operations + var len = 2 * prefix.length + 1 // expanded prefix + + data.length // five-bit data encoding + + _encoding.CHECKSUM_LENGTH; // checksum + + if (len - prefix.length > MAX_ENC_LENGTH) { + throw new Error("Message to be produced is too long (max ".concat(MAX_ENC_LENGTH, " supported)")); + } + + var prefixCase = (_detectCase = (0, _encoding.detectCase)(prefix, 'prefix')) !== null && _detectCase !== void 0 ? _detectCase : 'lower'; + var buffer = (0, _bitConverter.createBitArray)(len); // 2. Expand the human-readable prefix into the beginning of the buffer + + (0, _encoding.expandPrefix)(prefix.toLowerCase(), buffer.subarray(0, 2 * prefix.length + 1)); // 3. Copy `data` into the output + + var dataBuffer = buffer.subarray(2 * prefix.length + 1, buffer.length - _encoding.CHECKSUM_LENGTH); + dataBuffer.set(data); // 4. Create the checksum + + (0, _encoding.createChecksum)(buffer, encoding); // 5. Convert into string + + var encoded = (0, _encoding.encode)(buffer.subarray(2 * prefix.length + 1)); + + if (prefixCase === 'upper') { + encoded = encoded.toUpperCase(); + } + + return "".concat(prefix, "1").concat(encoded); +} +/** + * Encodes binary data into Bech32 encoding. + * + * The case is preserved: if the prefix is uppercase, then the output will be uppercase + * as well; otherwise, the output will be lowercase (including the case when the prefix does + * not contain any letters). + * + * @param {string} prefix + * Human-readable prefix to place at the beginning of the encoding + * @param {Uint8Array} data + * Binary data to encode + * @param {Encoding} encoding + * Encoding to use; influences the checksum computation. If not specified, + * Bech32 encoding will be used. + * @returns {string} + * Bech32 encoding of data in the form `1` + * @throws If the prefix is mixed-case or contains chars that are not eligible for Bech32 encoding + * + * @api public + */ + + +function encode(prefix, data) { + var encoding = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 'bech32'; + return encode5BitArray(prefix, to5BitArray(data), encoding); +} +/** + * Decodes data from Bech32 encoding into an array of 5-bit integers. + * + * Ordinarily, you may want to use [`decode`](#decode) because it automatically + * converts the array of 5-bit integers into an ordinary `Uint8Array`. + * + * @param {string} message + * Bech32-encoded message + * @returns {DecodeResult} + * Decoded object with `prefix` and `data` fields, which contain the human-readable + * prefix and the array of 5-bit integers respectively. + * + * @api public + */ + + +function decodeTo5BitArray(message) { + // Check preconditions + // 1. Message length + if (message.length > MAX_ENC_LENGTH) { + throw new TypeError("Message too long; max ".concat(MAX_ENC_LENGTH, " expected")); + } // 2. Mixed case + + + (0, _encoding.detectCase)(message); // we don't care about the result, only about checks. + + var lowerCaseMsg = message.toLowerCase(); // 4. Existence of the separator char + + var sepIdx = lowerCaseMsg.lastIndexOf('1'); + + if (sepIdx < 0) { + throw new Error('No separator char ("1") found'); + } // 5. Placing of the separator char in the message + + + if (sepIdx > message.length - _encoding.CHECKSUM_LENGTH - 1) { + throw new Error("Data part of the message too short (at least ".concat(_encoding.CHECKSUM_LENGTH, " chars expected)")); + } + + var prefix = lowerCaseMsg.substring(0, sepIdx); // Checked within `decodeWithPrefix`: + // 6. Invalid chars in the data part of the message + + var bitArray = (0, _encoding.decodeWithPrefix)(prefix, lowerCaseMsg.substring(sepIdx + 1)); // 7. Checksum + + var encoding = (0, _encoding.verifyChecksum)(bitArray); + + if (encoding === undefined) { + throw new Error('Invalid checksum'); + } + + return { + prefix: prefix, + encoding: encoding, + // Strip off the prefix from the front and the checksum from the end + data: bitArray.subarray(2 * prefix.length + 1, bitArray.length - _encoding.CHECKSUM_LENGTH) + }; +} +/** + * Decodes data from Bech32 encoding into an array of 5-bit integers. + * + * @param {string} message + * Bech32-encoded message + * @returns {DecodeResult} + * Decoded object with `prefix` and `data` fields, which contain the human-readable + * prefix and the decoded binary data respectively. + * + * @api public + */ + + +function decode(message) { + var _decodeTo5BitArray = decodeTo5BitArray(message), + prefix = _decodeTo5BitArray.prefix, + encoding = _decodeTo5BitArray.encoding, + bitArray = _decodeTo5BitArray.data; + + return { + prefix: prefix, + encoding: encoding, + data: from5BitArray(bitArray) + }; +} +/** + * Bitcoin address. + */ + + +var BitcoinAddress = /*#__PURE__*/function () { + function BitcoinAddress(prefix, scriptVersion, data) { + _classCallCheck(this, BitcoinAddress); + + if (prefix !== 'bc' && prefix !== 'tb') { + throw new Error('Invalid human-readable prefix, "bc" or "tb" expected'); + } + + if (scriptVersion < 0 || scriptVersion > 16) { + throw new RangeError('Invalid scriptVersion, value in range [0, 16] expected'); + } + + if (data.length < 2 || data.length > 40) { + throw new RangeError('Invalid script length: expected 2 to 40 bytes'); + } + + if (scriptVersion === 0 && data.length !== 20 && data.length !== 32) { + throw new Error('Invalid v0 script length: expected 20 or 32 bytes'); + } + + this.prefix = prefix; + this.scriptVersion = scriptVersion; + this.data = data; + } + /** + * Guesses the address type based on its internal structure. + * + * @returns {void | 'p2wpkh' | 'p2wsh'} + */ + + + _createClass(BitcoinAddress, [{ + key: "type", + value: function type() { + if (this.scriptVersion !== 0) { + return undefined; + } + + switch (this.data.length) { + case 20: + return 'p2wpkh'; + + case 32: + return 'p2wsh'; + // should be unreachable, but it's JS, so you never know + + default: + return undefined; + } + } + /** + * Encodes this address in Bech32 or Bech32m format, depending on the script version. + * Version 0 scripts are encoded using original Bech32 encoding as per BIP 173, + * while versions 1-16 are encoded using the modified encoding as per BIP 350. + * + * @returns {string} + * Bech32(m)-encoded address + */ + + }, { + key: "encode", + value: function encode() { + // Bitcoin addresses use Bech32 in a peculiar way - script version is + // not a part of the serialized binary data, but is rather prepended as 5-bit value + // before the rest of the script. This necessitates some plumbing here. + var len = Math.ceil(this.data.length * 8 / 5); + var converted = (0, _bitConverter.createBitArray)(len + 1); + converted[0] = this.scriptVersion; + to5BitArray(this.data, converted.subarray(1)); + var encoding = this.scriptVersion === 0 ? 'bech32' : 'bech32m'; + return encode5BitArray(this.prefix, converted, encoding); + } + }], [{ + key: "decode", + value: + /** + * Human-readable prefix. Equal to `'bc'` (for mainnet addresses) + * or `'tb'` (for testnet addresses). + */ + + /** + * Script version. An integer between 0 and 16 (inclusive). + */ + + /** + * Script data. A byte string with length 2 to 40 (inclusive). + */ + + /** + * Decodes a Bitcoin address from a Bech32(m) string. + * As per BIP 350, the original encoding is expected for version 0 scripts, while + * other script versions expect the modified encoding. + * + * This method does not check whether the address is well-formed; + * use `type()` method on returned address to find that out. + * + * @param {string} message + * @returns {BitcoinAddress} + */ + function decode(message) { + var _decodeTo5BitArray2 = decodeTo5BitArray(message), + prefix = _decodeTo5BitArray2.prefix, + data = _decodeTo5BitArray2.data, + encoding = _decodeTo5BitArray2.encoding; // Extra check to satisfy Flow. + + + if (prefix !== 'bc' && prefix !== 'tb') { + throw new Error('Invalid human-readable prefix, "bc" or "tb" expected'); + } + + var scriptVersion = data[0]; + + if (scriptVersion === 0 && encoding !== 'bech32') { + throw Error("Unexpected encoding ".concat(encoding, " used for version 0 script")); + } + + if (scriptVersion > 0 && encoding !== 'bech32m') { + throw Error("Unexpected encoding ".concat(encoding, " used for version ").concat(scriptVersion, " script")); + } + + return new this(prefix, scriptVersion, from5BitArray(data.subarray(1))); + } + }]); + + return BitcoinAddress; +}(); + +exports.BitcoinAddress = BitcoinAddress; \ No newline at end of file diff --git a/node_modules/bech32-buffer/package.json b/node_modules/bech32-buffer/package.json new file mode 100644 index 0000000..dc464c8 --- /dev/null +++ b/node_modules/bech32-buffer/package.json @@ -0,0 +1,87 @@ +{ + "name": "bech32-buffer", + "version": "0.2.1", + "description": "Bech32(m) encoding for byte buffers", + "repository": { + "type": "git", + "url": "https://github.com/slowli/bech32-buffer.git" + }, + "bugs": { + "url": "https://github.com/slowli/bech32-buffer/issues" + }, + "engines": { + "node": ">=8" + }, + "main": "lib/index.js", + "files": [ + "lib", + "dist", + "examples", + "index.d.ts", + "CHANGELOG.md" + ], + "types": "index.d.ts", + "scripts": { + "test": "mocha -r @babel/register", + "test-browser": "karma start", + "cover": "cross-env NODE_ENV=test nyc mocha", + "report-cover": "nyc report", + "browser": "mkdirp dist && cross-env BABEL_ENV=browser browserify src/index.js -t [ babelify ] --standalone bech32 -o dist/bech32-buffer.min.js", + "minify-qr": "cross-env BABEL_ENV=browser browserify node_modules/qrcode-generator/qrcode.js -t [ babelify ] --standalone qrcode -o examples/qrcode.min.js", + "compile": "babel src/ -d lib/", + "prepare": "npm run compile && npm run browser", + "lint": "eslint --ignore-path .gitignore . && flow check", + "test-ts": "npm run compile && tsc -p test && node test/types.js" + }, + "keywords": [ + "bech32", + "bech32m", + "base32", + "bitcoin" + ], + "author": "Alex Ostrovski ", + "license": "Apache-2.0", + "devDependencies": { + "@babel/cli": "^7.18.6", + "@babel/core": "^7.19.1", + "@babel/eslint-parser": "^7.19.1", + "@babel/preset-env": "^7.19.1", + "@babel/preset-flow": "^7.18.6", + "@babel/register": "^7.18.6", + "@types/chai": "^4.2.22", + "babel-plugin-istanbul": "^6.0.0", + "babel-preset-minify": "^0.5.1", + "babelify": "^10.0.0", + "browserify": "^17.0.0", + "chai": "^4.3.4", + "chai-bytes": "^0.1.2", + "cross-env": "^7.0.3", + "dirty-chai": "^2.0.1", + "eslint": "^8.24.0", + "eslint-config-airbnb-base": "^15.0.0", + "eslint-plugin-import": "^2.24.2", + "flow-bin": "^0.187.1", + "karma": "^6.4.0", + "karma-browserify": "^8.0.0", + "karma-chrome-launcher": "^3.1.0", + "karma-firefox-launcher": "^2.1.1", + "karma-mocha": "^2.0.1", + "karma-mocha-reporter": "^2.2.3", + "mkdirp": "^1.0.4", + "mocha": "^10.0.0", + "nyc": "^15.1.0", + "qrcode-generator": "^1.4.4", + "typescript": "^4.8.2" + }, + "nyc": { + "require": [ + "@babel/register" + ], + "exclude": [ + "lib", + "test" + ], + "sourceMap": false, + "instrument": false + } +} diff --git a/node_modules/bn.js/LICENSE b/node_modules/bn.js/LICENSE new file mode 100644 index 0000000..c328f04 --- /dev/null +++ b/node_modules/bn.js/LICENSE @@ -0,0 +1,19 @@ +Copyright Fedor Indutny, 2015. + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/node_modules/bn.js/README.md b/node_modules/bn.js/README.md new file mode 100644 index 0000000..aecc3ba --- /dev/null +++ b/node_modules/bn.js/README.md @@ -0,0 +1,200 @@ +# bn.js + +> BigNum in pure javascript + +[![Build Status](https://secure.travis-ci.org/indutny/bn.js.png)](http://travis-ci.org/indutny/bn.js) + +## Install +`npm install --save bn.js` + +## Usage + +```js +const BN = require('bn.js'); + +var a = new BN('dead', 16); +var b = new BN('101010', 2); + +var res = a.add(b); +console.log(res.toString(10)); // 57047 +``` + +**Note**: decimals are not supported in this library. + +## Notation + +### Prefixes + +There are several prefixes to instructions that affect the way the work. Here +is the list of them in the order of appearance in the function name: + +* `i` - perform operation in-place, storing the result in the host object (on + which the method was invoked). Might be used to avoid number allocation costs +* `u` - unsigned, ignore the sign of operands when performing operation, or + always return positive value. Second case applies to reduction operations + like `mod()`. In such cases if the result will be negative - modulo will be + added to the result to make it positive + +### Postfixes + +The only available postfix at the moment is: + +* `n` - which means that the argument of the function must be a plain JavaScript + Number. Decimals are not supported. + +### Examples + +* `a.iadd(b)` - perform addition on `a` and `b`, storing the result in `a` +* `a.umod(b)` - reduce `a` modulo `b`, returning positive value +* `a.iushln(13)` - shift bits of `a` left by 13 + +## Instructions + +Prefixes/postfixes are put in parens at the of the line. `endian` - could be +either `le` (little-endian) or `be` (big-endian). + +### Utilities + +* `a.clone()` - clone number +* `a.toString(base, length)` - convert to base-string and pad with zeroes +* `a.toNumber()` - convert to Javascript Number (limited to 53 bits) +* `a.toJSON()` - convert to JSON compatible hex string (alias of `toString(16)`) +* `a.toArray(endian, length)` - convert to byte `Array`, and optionally zero + pad to length, throwing if already exceeding +* `a.toArrayLike(type, endian, length)` - convert to an instance of `type`, + which must behave like an `Array` +* `a.toBuffer(endian, length)` - convert to Node.js Buffer (if available). For + compatibility with browserify and similar tools, use this instead: + `a.toArrayLike(Buffer, endian, length)` +* `a.bitLength()` - get number of bits occupied +* `a.zeroBits()` - return number of less-significant consequent zero bits + (example: `1010000` has 4 zero bits) +* `a.byteLength()` - return number of bytes occupied +* `a.isNeg()` - true if the number is negative +* `a.isEven()` - no comments +* `a.isOdd()` - no comments +* `a.isZero()` - no comments +* `a.cmp(b)` - compare numbers and return `-1` (a `<` b), `0` (a `==` b), or `1` (a `>` b) + depending on the comparison result (`ucmp`, `cmpn`) +* `a.lt(b)` - `a` less than `b` (`n`) +* `a.lte(b)` - `a` less than or equals `b` (`n`) +* `a.gt(b)` - `a` greater than `b` (`n`) +* `a.gte(b)` - `a` greater than or equals `b` (`n`) +* `a.eq(b)` - `a` equals `b` (`n`) +* `a.toTwos(width)` - convert to two's complement representation, where `width` is bit width +* `a.fromTwos(width)` - convert from two's complement representation, where `width` is the bit width +* `BN.isBN(object)` - returns true if the supplied `object` is a BN.js instance + +### Arithmetics + +* `a.neg()` - negate sign (`i`) +* `a.abs()` - absolute value (`i`) +* `a.add(b)` - addition (`i`, `n`, `in`) +* `a.sub(b)` - subtraction (`i`, `n`, `in`) +* `a.mul(b)` - multiply (`i`, `n`, `in`) +* `a.sqr()` - square (`i`) +* `a.pow(b)` - raise `a` to the power of `b` +* `a.div(b)` - divide (`divn`, `idivn`) +* `a.mod(b)` - reduct (`u`, `n`) (but no `umodn`) +* `a.divRound(b)` - rounded division + +### Bit operations + +* `a.or(b)` - or (`i`, `u`, `iu`) +* `a.and(b)` - and (`i`, `u`, `iu`, `andln`) (NOTE: `andln` is going to be replaced + with `andn` in future) +* `a.xor(b)` - xor (`i`, `u`, `iu`) +* `a.setn(b)` - set specified bit to `1` +* `a.shln(b)` - shift left (`i`, `u`, `iu`) +* `a.shrn(b)` - shift right (`i`, `u`, `iu`) +* `a.testn(b)` - test if specified bit is set +* `a.maskn(b)` - clear bits with indexes higher or equal to `b` (`i`) +* `a.bincn(b)` - add `1 << b` to the number +* `a.notn(w)` - not (for the width specified by `w`) (`i`) + +### Reduction + +* `a.gcd(b)` - GCD +* `a.egcd(b)` - Extended GCD results (`{ a: ..., b: ..., gcd: ... }`) +* `a.invm(b)` - inverse `a` modulo `b` + +## Fast reduction + +When doing lots of reductions using the same modulo, it might be beneficial to +use some tricks: like [Montgomery multiplication][0], or using special algorithm +for [Mersenne Prime][1]. + +### Reduction context + +To enable this tricks one should create a reduction context: + +```js +var red = BN.red(num); +``` +where `num` is just a BN instance. + +Or: + +```js +var red = BN.red(primeName); +``` + +Where `primeName` is either of these [Mersenne Primes][1]: + +* `'k256'` +* `'p224'` +* `'p192'` +* `'p25519'` + +Or: + +```js +var red = BN.mont(num); +``` + +To reduce numbers with [Montgomery trick][0]. `.mont()` is generally faster than +`.red(num)`, but slower than `BN.red(primeName)`. + +### Converting numbers + +Before performing anything in reduction context - numbers should be converted +to it. Usually, this means that one should: + +* Convert inputs to reducted ones +* Operate on them in reduction context +* Convert outputs back from the reduction context + +Here is how one may convert numbers to `red`: + +```js +var redA = a.toRed(red); +``` +Where `red` is a reduction context created using instructions above + +Here is how to convert them back: + +```js +var a = redA.fromRed(); +``` + +### Red instructions + +Most of the instructions from the very start of this readme have their +counterparts in red context: + +* `a.redAdd(b)`, `a.redIAdd(b)` +* `a.redSub(b)`, `a.redISub(b)` +* `a.redShl(num)` +* `a.redMul(b)`, `a.redIMul(b)` +* `a.redSqr()`, `a.redISqr()` +* `a.redSqrt()` - square root modulo reduction context's prime +* `a.redInvm()` - modular inverse of the number +* `a.redNeg()` +* `a.redPow(b)` - modular exponentiation + +## LICENSE + +This software is licensed under the MIT License. + +[0]: https://en.wikipedia.org/wiki/Montgomery_modular_multiplication +[1]: https://en.wikipedia.org/wiki/Mersenne_prime diff --git a/node_modules/bn.js/lib/bn.js b/node_modules/bn.js/lib/bn.js new file mode 100644 index 0000000..3a4371e --- /dev/null +++ b/node_modules/bn.js/lib/bn.js @@ -0,0 +1,3446 @@ +(function (module, exports) { + 'use strict'; + + // Utils + function assert (val, msg) { + if (!val) throw new Error(msg || 'Assertion failed'); + } + + // Could use `inherits` module, but don't want to move from single file + // architecture yet. + function inherits (ctor, superCtor) { + ctor.super_ = superCtor; + var TempCtor = function () {}; + TempCtor.prototype = superCtor.prototype; + ctor.prototype = new TempCtor(); + ctor.prototype.constructor = ctor; + } + + // BN + + function BN (number, base, endian) { + if (BN.isBN(number)) { + return number; + } + + this.negative = 0; + this.words = null; + this.length = 0; + + // Reduction context + this.red = null; + + if (number !== null) { + if (base === 'le' || base === 'be') { + endian = base; + base = 10; + } + + this._init(number || 0, base || 10, endian || 'be'); + } + } + if (typeof module === 'object') { + module.exports = BN; + } else { + exports.BN = BN; + } + + BN.BN = BN; + BN.wordSize = 26; + + var Buffer; + try { + if (typeof window !== 'undefined' && typeof window.Buffer !== 'undefined') { + Buffer = window.Buffer; + } else { + Buffer = require('buffer').Buffer; + } + } catch (e) { + } + + BN.isBN = function isBN (num) { + if (num instanceof BN) { + return true; + } + + return num !== null && typeof num === 'object' && + num.constructor.wordSize === BN.wordSize && Array.isArray(num.words); + }; + + BN.max = function max (left, right) { + if (left.cmp(right) > 0) return left; + return right; + }; + + BN.min = function min (left, right) { + if (left.cmp(right) < 0) return left; + return right; + }; + + BN.prototype._init = function init (number, base, endian) { + if (typeof number === 'number') { + return this._initNumber(number, base, endian); + } + + if (typeof number === 'object') { + return this._initArray(number, base, endian); + } + + if (base === 'hex') { + base = 16; + } + assert(base === (base | 0) && base >= 2 && base <= 36); + + number = number.toString().replace(/\s+/g, ''); + var start = 0; + if (number[0] === '-') { + start++; + this.negative = 1; + } + + if (start < number.length) { + if (base === 16) { + this._parseHex(number, start, endian); + } else { + this._parseBase(number, base, start); + if (endian === 'le') { + this._initArray(this.toArray(), base, endian); + } + } + } + }; + + BN.prototype._initNumber = function _initNumber (number, base, endian) { + if (number < 0) { + this.negative = 1; + number = -number; + } + if (number < 0x4000000) { + this.words = [ number & 0x3ffffff ]; + this.length = 1; + } else if (number < 0x10000000000000) { + this.words = [ + number & 0x3ffffff, + (number / 0x4000000) & 0x3ffffff + ]; + this.length = 2; + } else { + assert(number < 0x20000000000000); // 2 ^ 53 (unsafe) + this.words = [ + number & 0x3ffffff, + (number / 0x4000000) & 0x3ffffff, + 1 + ]; + this.length = 3; + } + + if (endian !== 'le') return; + + // Reverse the bytes + this._initArray(this.toArray(), base, endian); + }; + + BN.prototype._initArray = function _initArray (number, base, endian) { + // Perhaps a Uint8Array + assert(typeof number.length === 'number'); + if (number.length <= 0) { + this.words = [ 0 ]; + this.length = 1; + return this; + } + + this.length = Math.ceil(number.length / 3); + this.words = new Array(this.length); + for (var i = 0; i < this.length; i++) { + this.words[i] = 0; + } + + var j, w; + var off = 0; + if (endian === 'be') { + for (i = number.length - 1, j = 0; i >= 0; i -= 3) { + w = number[i] | (number[i - 1] << 8) | (number[i - 2] << 16); + this.words[j] |= (w << off) & 0x3ffffff; + this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff; + off += 24; + if (off >= 26) { + off -= 26; + j++; + } + } + } else if (endian === 'le') { + for (i = 0, j = 0; i < number.length; i += 3) { + w = number[i] | (number[i + 1] << 8) | (number[i + 2] << 16); + this.words[j] |= (w << off) & 0x3ffffff; + this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff; + off += 24; + if (off >= 26) { + off -= 26; + j++; + } + } + } + return this.strip(); + }; + + function parseHex4Bits (string, index) { + var c = string.charCodeAt(index); + // 'A' - 'F' + if (c >= 65 && c <= 70) { + return c - 55; + // 'a' - 'f' + } else if (c >= 97 && c <= 102) { + return c - 87; + // '0' - '9' + } else { + return (c - 48) & 0xf; + } + } + + function parseHexByte (string, lowerBound, index) { + var r = parseHex4Bits(string, index); + if (index - 1 >= lowerBound) { + r |= parseHex4Bits(string, index - 1) << 4; + } + return r; + } + + BN.prototype._parseHex = function _parseHex (number, start, endian) { + // Create possibly bigger array to ensure that it fits the number + this.length = Math.ceil((number.length - start) / 6); + this.words = new Array(this.length); + for (var i = 0; i < this.length; i++) { + this.words[i] = 0; + } + + // 24-bits chunks + var off = 0; + var j = 0; + + var w; + if (endian === 'be') { + for (i = number.length - 1; i >= start; i -= 2) { + w = parseHexByte(number, start, i) << off; + this.words[j] |= w & 0x3ffffff; + if (off >= 18) { + off -= 18; + j += 1; + this.words[j] |= w >>> 26; + } else { + off += 8; + } + } + } else { + var parseLength = number.length - start; + for (i = parseLength % 2 === 0 ? start + 1 : start; i < number.length; i += 2) { + w = parseHexByte(number, start, i) << off; + this.words[j] |= w & 0x3ffffff; + if (off >= 18) { + off -= 18; + j += 1; + this.words[j] |= w >>> 26; + } else { + off += 8; + } + } + } + + this.strip(); + }; + + function parseBase (str, start, end, mul) { + var r = 0; + var len = Math.min(str.length, end); + for (var i = start; i < len; i++) { + var c = str.charCodeAt(i) - 48; + + r *= mul; + + // 'a' + if (c >= 49) { + r += c - 49 + 0xa; + + // 'A' + } else if (c >= 17) { + r += c - 17 + 0xa; + + // '0' - '9' + } else { + r += c; + } + } + return r; + } + + BN.prototype._parseBase = function _parseBase (number, base, start) { + // Initialize as zero + this.words = [ 0 ]; + this.length = 1; + + // Find length of limb in base + for (var limbLen = 0, limbPow = 1; limbPow <= 0x3ffffff; limbPow *= base) { + limbLen++; + } + limbLen--; + limbPow = (limbPow / base) | 0; + + var total = number.length - start; + var mod = total % limbLen; + var end = Math.min(total, total - mod) + start; + + var word = 0; + for (var i = start; i < end; i += limbLen) { + word = parseBase(number, i, i + limbLen, base); + + this.imuln(limbPow); + if (this.words[0] + word < 0x4000000) { + this.words[0] += word; + } else { + this._iaddn(word); + } + } + + if (mod !== 0) { + var pow = 1; + word = parseBase(number, i, number.length, base); + + for (i = 0; i < mod; i++) { + pow *= base; + } + + this.imuln(pow); + if (this.words[0] + word < 0x4000000) { + this.words[0] += word; + } else { + this._iaddn(word); + } + } + + this.strip(); + }; + + BN.prototype.copy = function copy (dest) { + dest.words = new Array(this.length); + for (var i = 0; i < this.length; i++) { + dest.words[i] = this.words[i]; + } + dest.length = this.length; + dest.negative = this.negative; + dest.red = this.red; + }; + + BN.prototype.clone = function clone () { + var r = new BN(null); + this.copy(r); + return r; + }; + + BN.prototype._expand = function _expand (size) { + while (this.length < size) { + this.words[this.length++] = 0; + } + return this; + }; + + // Remove leading `0` from `this` + BN.prototype.strip = function strip () { + while (this.length > 1 && this.words[this.length - 1] === 0) { + this.length--; + } + return this._normSign(); + }; + + BN.prototype._normSign = function _normSign () { + // -0 = 0 + if (this.length === 1 && this.words[0] === 0) { + this.negative = 0; + } + return this; + }; + + BN.prototype.inspect = function inspect () { + return (this.red ? ''; + }; + + /* + + var zeros = []; + var groupSizes = []; + var groupBases = []; + + var s = ''; + var i = -1; + while (++i < BN.wordSize) { + zeros[i] = s; + s += '0'; + } + groupSizes[0] = 0; + groupSizes[1] = 0; + groupBases[0] = 0; + groupBases[1] = 0; + var base = 2 - 1; + while (++base < 36 + 1) { + var groupSize = 0; + var groupBase = 1; + while (groupBase < (1 << BN.wordSize) / base) { + groupBase *= base; + groupSize += 1; + } + groupSizes[base] = groupSize; + groupBases[base] = groupBase; + } + + */ + + var zeros = [ + '', + '0', + '00', + '000', + '0000', + '00000', + '000000', + '0000000', + '00000000', + '000000000', + '0000000000', + '00000000000', + '000000000000', + '0000000000000', + '00000000000000', + '000000000000000', + '0000000000000000', + '00000000000000000', + '000000000000000000', + '0000000000000000000', + '00000000000000000000', + '000000000000000000000', + '0000000000000000000000', + '00000000000000000000000', + '000000000000000000000000', + '0000000000000000000000000' + ]; + + var groupSizes = [ + 0, 0, + 25, 16, 12, 11, 10, 9, 8, + 8, 7, 7, 7, 7, 6, 6, + 6, 6, 6, 6, 6, 5, 5, + 5, 5, 5, 5, 5, 5, 5, + 5, 5, 5, 5, 5, 5, 5 + ]; + + var groupBases = [ + 0, 0, + 33554432, 43046721, 16777216, 48828125, 60466176, 40353607, 16777216, + 43046721, 10000000, 19487171, 35831808, 62748517, 7529536, 11390625, + 16777216, 24137569, 34012224, 47045881, 64000000, 4084101, 5153632, + 6436343, 7962624, 9765625, 11881376, 14348907, 17210368, 20511149, + 24300000, 28629151, 33554432, 39135393, 45435424, 52521875, 60466176 + ]; + + BN.prototype.toString = function toString (base, padding) { + base = base || 10; + padding = padding | 0 || 1; + + var out; + if (base === 16 || base === 'hex') { + out = ''; + var off = 0; + var carry = 0; + for (var i = 0; i < this.length; i++) { + var w = this.words[i]; + var word = (((w << off) | carry) & 0xffffff).toString(16); + carry = (w >>> (24 - off)) & 0xffffff; + if (carry !== 0 || i !== this.length - 1) { + out = zeros[6 - word.length] + word + out; + } else { + out = word + out; + } + off += 2; + if (off >= 26) { + off -= 26; + i--; + } + } + if (carry !== 0) { + out = carry.toString(16) + out; + } + while (out.length % padding !== 0) { + out = '0' + out; + } + if (this.negative !== 0) { + out = '-' + out; + } + return out; + } + + if (base === (base | 0) && base >= 2 && base <= 36) { + // var groupSize = Math.floor(BN.wordSize * Math.LN2 / Math.log(base)); + var groupSize = groupSizes[base]; + // var groupBase = Math.pow(base, groupSize); + var groupBase = groupBases[base]; + out = ''; + var c = this.clone(); + c.negative = 0; + while (!c.isZero()) { + var r = c.modn(groupBase).toString(base); + c = c.idivn(groupBase); + + if (!c.isZero()) { + out = zeros[groupSize - r.length] + r + out; + } else { + out = r + out; + } + } + if (this.isZero()) { + out = '0' + out; + } + while (out.length % padding !== 0) { + out = '0' + out; + } + if (this.negative !== 0) { + out = '-' + out; + } + return out; + } + + assert(false, 'Base should be between 2 and 36'); + }; + + BN.prototype.toNumber = function toNumber () { + var ret = this.words[0]; + if (this.length === 2) { + ret += this.words[1] * 0x4000000; + } else if (this.length === 3 && this.words[2] === 0x01) { + // NOTE: at this stage it is known that the top bit is set + ret += 0x10000000000000 + (this.words[1] * 0x4000000); + } else if (this.length > 2) { + assert(false, 'Number can only safely store up to 53 bits'); + } + return (this.negative !== 0) ? -ret : ret; + }; + + BN.prototype.toJSON = function toJSON () { + return this.toString(16); + }; + + BN.prototype.toBuffer = function toBuffer (endian, length) { + assert(typeof Buffer !== 'undefined'); + return this.toArrayLike(Buffer, endian, length); + }; + + BN.prototype.toArray = function toArray (endian, length) { + return this.toArrayLike(Array, endian, length); + }; + + BN.prototype.toArrayLike = function toArrayLike (ArrayType, endian, length) { + var byteLength = this.byteLength(); + var reqLength = length || Math.max(1, byteLength); + assert(byteLength <= reqLength, 'byte array longer than desired length'); + assert(reqLength > 0, 'Requested array length <= 0'); + + this.strip(); + var littleEndian = endian === 'le'; + var res = new ArrayType(reqLength); + + var b, i; + var q = this.clone(); + if (!littleEndian) { + // Assume big-endian + for (i = 0; i < reqLength - byteLength; i++) { + res[i] = 0; + } + + for (i = 0; !q.isZero(); i++) { + b = q.andln(0xff); + q.iushrn(8); + + res[reqLength - i - 1] = b; + } + } else { + for (i = 0; !q.isZero(); i++) { + b = q.andln(0xff); + q.iushrn(8); + + res[i] = b; + } + + for (; i < reqLength; i++) { + res[i] = 0; + } + } + + return res; + }; + + if (Math.clz32) { + BN.prototype._countBits = function _countBits (w) { + return 32 - Math.clz32(w); + }; + } else { + BN.prototype._countBits = function _countBits (w) { + var t = w; + var r = 0; + if (t >= 0x1000) { + r += 13; + t >>>= 13; + } + if (t >= 0x40) { + r += 7; + t >>>= 7; + } + if (t >= 0x8) { + r += 4; + t >>>= 4; + } + if (t >= 0x02) { + r += 2; + t >>>= 2; + } + return r + t; + }; + } + + BN.prototype._zeroBits = function _zeroBits (w) { + // Short-cut + if (w === 0) return 26; + + var t = w; + var r = 0; + if ((t & 0x1fff) === 0) { + r += 13; + t >>>= 13; + } + if ((t & 0x7f) === 0) { + r += 7; + t >>>= 7; + } + if ((t & 0xf) === 0) { + r += 4; + t >>>= 4; + } + if ((t & 0x3) === 0) { + r += 2; + t >>>= 2; + } + if ((t & 0x1) === 0) { + r++; + } + return r; + }; + + // Return number of used bits in a BN + BN.prototype.bitLength = function bitLength () { + var w = this.words[this.length - 1]; + var hi = this._countBits(w); + return (this.length - 1) * 26 + hi; + }; + + function toBitArray (num) { + var w = new Array(num.bitLength()); + + for (var bit = 0; bit < w.length; bit++) { + var off = (bit / 26) | 0; + var wbit = bit % 26; + + w[bit] = (num.words[off] & (1 << wbit)) >>> wbit; + } + + return w; + } + + // Number of trailing zero bits + BN.prototype.zeroBits = function zeroBits () { + if (this.isZero()) return 0; + + var r = 0; + for (var i = 0; i < this.length; i++) { + var b = this._zeroBits(this.words[i]); + r += b; + if (b !== 26) break; + } + return r; + }; + + BN.prototype.byteLength = function byteLength () { + return Math.ceil(this.bitLength() / 8); + }; + + BN.prototype.toTwos = function toTwos (width) { + if (this.negative !== 0) { + return this.abs().inotn(width).iaddn(1); + } + return this.clone(); + }; + + BN.prototype.fromTwos = function fromTwos (width) { + if (this.testn(width - 1)) { + return this.notn(width).iaddn(1).ineg(); + } + return this.clone(); + }; + + BN.prototype.isNeg = function isNeg () { + return this.negative !== 0; + }; + + // Return negative clone of `this` + BN.prototype.neg = function neg () { + return this.clone().ineg(); + }; + + BN.prototype.ineg = function ineg () { + if (!this.isZero()) { + this.negative ^= 1; + } + + return this; + }; + + // Or `num` with `this` in-place + BN.prototype.iuor = function iuor (num) { + while (this.length < num.length) { + this.words[this.length++] = 0; + } + + for (var i = 0; i < num.length; i++) { + this.words[i] = this.words[i] | num.words[i]; + } + + return this.strip(); + }; + + BN.prototype.ior = function ior (num) { + assert((this.negative | num.negative) === 0); + return this.iuor(num); + }; + + // Or `num` with `this` + BN.prototype.or = function or (num) { + if (this.length > num.length) return this.clone().ior(num); + return num.clone().ior(this); + }; + + BN.prototype.uor = function uor (num) { + if (this.length > num.length) return this.clone().iuor(num); + return num.clone().iuor(this); + }; + + // And `num` with `this` in-place + BN.prototype.iuand = function iuand (num) { + // b = min-length(num, this) + var b; + if (this.length > num.length) { + b = num; + } else { + b = this; + } + + for (var i = 0; i < b.length; i++) { + this.words[i] = this.words[i] & num.words[i]; + } + + this.length = b.length; + + return this.strip(); + }; + + BN.prototype.iand = function iand (num) { + assert((this.negative | num.negative) === 0); + return this.iuand(num); + }; + + // And `num` with `this` + BN.prototype.and = function and (num) { + if (this.length > num.length) return this.clone().iand(num); + return num.clone().iand(this); + }; + + BN.prototype.uand = function uand (num) { + if (this.length > num.length) return this.clone().iuand(num); + return num.clone().iuand(this); + }; + + // Xor `num` with `this` in-place + BN.prototype.iuxor = function iuxor (num) { + // a.length > b.length + var a; + var b; + if (this.length > num.length) { + a = this; + b = num; + } else { + a = num; + b = this; + } + + for (var i = 0; i < b.length; i++) { + this.words[i] = a.words[i] ^ b.words[i]; + } + + if (this !== a) { + for (; i < a.length; i++) { + this.words[i] = a.words[i]; + } + } + + this.length = a.length; + + return this.strip(); + }; + + BN.prototype.ixor = function ixor (num) { + assert((this.negative | num.negative) === 0); + return this.iuxor(num); + }; + + // Xor `num` with `this` + BN.prototype.xor = function xor (num) { + if (this.length > num.length) return this.clone().ixor(num); + return num.clone().ixor(this); + }; + + BN.prototype.uxor = function uxor (num) { + if (this.length > num.length) return this.clone().iuxor(num); + return num.clone().iuxor(this); + }; + + // Not ``this`` with ``width`` bitwidth + BN.prototype.inotn = function inotn (width) { + assert(typeof width === 'number' && width >= 0); + + var bytesNeeded = Math.ceil(width / 26) | 0; + var bitsLeft = width % 26; + + // Extend the buffer with leading zeroes + this._expand(bytesNeeded); + + if (bitsLeft > 0) { + bytesNeeded--; + } + + // Handle complete words + for (var i = 0; i < bytesNeeded; i++) { + this.words[i] = ~this.words[i] & 0x3ffffff; + } + + // Handle the residue + if (bitsLeft > 0) { + this.words[i] = ~this.words[i] & (0x3ffffff >> (26 - bitsLeft)); + } + + // And remove leading zeroes + return this.strip(); + }; + + BN.prototype.notn = function notn (width) { + return this.clone().inotn(width); + }; + + // Set `bit` of `this` + BN.prototype.setn = function setn (bit, val) { + assert(typeof bit === 'number' && bit >= 0); + + var off = (bit / 26) | 0; + var wbit = bit % 26; + + this._expand(off + 1); + + if (val) { + this.words[off] = this.words[off] | (1 << wbit); + } else { + this.words[off] = this.words[off] & ~(1 << wbit); + } + + return this.strip(); + }; + + // Add `num` to `this` in-place + BN.prototype.iadd = function iadd (num) { + var r; + + // negative + positive + if (this.negative !== 0 && num.negative === 0) { + this.negative = 0; + r = this.isub(num); + this.negative ^= 1; + return this._normSign(); + + // positive + negative + } else if (this.negative === 0 && num.negative !== 0) { + num.negative = 0; + r = this.isub(num); + num.negative = 1; + return r._normSign(); + } + + // a.length > b.length + var a, b; + if (this.length > num.length) { + a = this; + b = num; + } else { + a = num; + b = this; + } + + var carry = 0; + for (var i = 0; i < b.length; i++) { + r = (a.words[i] | 0) + (b.words[i] | 0) + carry; + this.words[i] = r & 0x3ffffff; + carry = r >>> 26; + } + for (; carry !== 0 && i < a.length; i++) { + r = (a.words[i] | 0) + carry; + this.words[i] = r & 0x3ffffff; + carry = r >>> 26; + } + + this.length = a.length; + if (carry !== 0) { + this.words[this.length] = carry; + this.length++; + // Copy the rest of the words + } else if (a !== this) { + for (; i < a.length; i++) { + this.words[i] = a.words[i]; + } + } + + return this; + }; + + // Add `num` to `this` + BN.prototype.add = function add (num) { + var res; + if (num.negative !== 0 && this.negative === 0) { + num.negative = 0; + res = this.sub(num); + num.negative ^= 1; + return res; + } else if (num.negative === 0 && this.negative !== 0) { + this.negative = 0; + res = num.sub(this); + this.negative = 1; + return res; + } + + if (this.length > num.length) return this.clone().iadd(num); + + return num.clone().iadd(this); + }; + + // Subtract `num` from `this` in-place + BN.prototype.isub = function isub (num) { + // this - (-num) = this + num + if (num.negative !== 0) { + num.negative = 0; + var r = this.iadd(num); + num.negative = 1; + return r._normSign(); + + // -this - num = -(this + num) + } else if (this.negative !== 0) { + this.negative = 0; + this.iadd(num); + this.negative = 1; + return this._normSign(); + } + + // At this point both numbers are positive + var cmp = this.cmp(num); + + // Optimization - zeroify + if (cmp === 0) { + this.negative = 0; + this.length = 1; + this.words[0] = 0; + return this; + } + + // a > b + var a, b; + if (cmp > 0) { + a = this; + b = num; + } else { + a = num; + b = this; + } + + var carry = 0; + for (var i = 0; i < b.length; i++) { + r = (a.words[i] | 0) - (b.words[i] | 0) + carry; + carry = r >> 26; + this.words[i] = r & 0x3ffffff; + } + for (; carry !== 0 && i < a.length; i++) { + r = (a.words[i] | 0) + carry; + carry = r >> 26; + this.words[i] = r & 0x3ffffff; + } + + // Copy rest of the words + if (carry === 0 && i < a.length && a !== this) { + for (; i < a.length; i++) { + this.words[i] = a.words[i]; + } + } + + this.length = Math.max(this.length, i); + + if (a !== this) { + this.negative = 1; + } + + return this.strip(); + }; + + // Subtract `num` from `this` + BN.prototype.sub = function sub (num) { + return this.clone().isub(num); + }; + + function smallMulTo (self, num, out) { + out.negative = num.negative ^ self.negative; + var len = (self.length + num.length) | 0; + out.length = len; + len = (len - 1) | 0; + + // Peel one iteration (compiler can't do it, because of code complexity) + var a = self.words[0] | 0; + var b = num.words[0] | 0; + var r = a * b; + + var lo = r & 0x3ffffff; + var carry = (r / 0x4000000) | 0; + out.words[0] = lo; + + for (var k = 1; k < len; k++) { + // Sum all words with the same `i + j = k` and accumulate `ncarry`, + // note that ncarry could be >= 0x3ffffff + var ncarry = carry >>> 26; + var rword = carry & 0x3ffffff; + var maxJ = Math.min(k, num.length - 1); + for (var j = Math.max(0, k - self.length + 1); j <= maxJ; j++) { + var i = (k - j) | 0; + a = self.words[i] | 0; + b = num.words[j] | 0; + r = a * b + rword; + ncarry += (r / 0x4000000) | 0; + rword = r & 0x3ffffff; + } + out.words[k] = rword | 0; + carry = ncarry | 0; + } + if (carry !== 0) { + out.words[k] = carry | 0; + } else { + out.length--; + } + + return out.strip(); + } + + // TODO(indutny): it may be reasonable to omit it for users who don't need + // to work with 256-bit numbers, otherwise it gives 20% improvement for 256-bit + // multiplication (like elliptic secp256k1). + var comb10MulTo = function comb10MulTo (self, num, out) { + var a = self.words; + var b = num.words; + var o = out.words; + var c = 0; + var lo; + var mid; + var hi; + var a0 = a[0] | 0; + var al0 = a0 & 0x1fff; + var ah0 = a0 >>> 13; + var a1 = a[1] | 0; + var al1 = a1 & 0x1fff; + var ah1 = a1 >>> 13; + var a2 = a[2] | 0; + var al2 = a2 & 0x1fff; + var ah2 = a2 >>> 13; + var a3 = a[3] | 0; + var al3 = a3 & 0x1fff; + var ah3 = a3 >>> 13; + var a4 = a[4] | 0; + var al4 = a4 & 0x1fff; + var ah4 = a4 >>> 13; + var a5 = a[5] | 0; + var al5 = a5 & 0x1fff; + var ah5 = a5 >>> 13; + var a6 = a[6] | 0; + var al6 = a6 & 0x1fff; + var ah6 = a6 >>> 13; + var a7 = a[7] | 0; + var al7 = a7 & 0x1fff; + var ah7 = a7 >>> 13; + var a8 = a[8] | 0; + var al8 = a8 & 0x1fff; + var ah8 = a8 >>> 13; + var a9 = a[9] | 0; + var al9 = a9 & 0x1fff; + var ah9 = a9 >>> 13; + var b0 = b[0] | 0; + var bl0 = b0 & 0x1fff; + var bh0 = b0 >>> 13; + var b1 = b[1] | 0; + var bl1 = b1 & 0x1fff; + var bh1 = b1 >>> 13; + var b2 = b[2] | 0; + var bl2 = b2 & 0x1fff; + var bh2 = b2 >>> 13; + var b3 = b[3] | 0; + var bl3 = b3 & 0x1fff; + var bh3 = b3 >>> 13; + var b4 = b[4] | 0; + var bl4 = b4 & 0x1fff; + var bh4 = b4 >>> 13; + var b5 = b[5] | 0; + var bl5 = b5 & 0x1fff; + var bh5 = b5 >>> 13; + var b6 = b[6] | 0; + var bl6 = b6 & 0x1fff; + var bh6 = b6 >>> 13; + var b7 = b[7] | 0; + var bl7 = b7 & 0x1fff; + var bh7 = b7 >>> 13; + var b8 = b[8] | 0; + var bl8 = b8 & 0x1fff; + var bh8 = b8 >>> 13; + var b9 = b[9] | 0; + var bl9 = b9 & 0x1fff; + var bh9 = b9 >>> 13; + + out.negative = self.negative ^ num.negative; + out.length = 19; + /* k = 0 */ + lo = Math.imul(al0, bl0); + mid = Math.imul(al0, bh0); + mid = (mid + Math.imul(ah0, bl0)) | 0; + hi = Math.imul(ah0, bh0); + var w0 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w0 >>> 26)) | 0; + w0 &= 0x3ffffff; + /* k = 1 */ + lo = Math.imul(al1, bl0); + mid = Math.imul(al1, bh0); + mid = (mid + Math.imul(ah1, bl0)) | 0; + hi = Math.imul(ah1, bh0); + lo = (lo + Math.imul(al0, bl1)) | 0; + mid = (mid + Math.imul(al0, bh1)) | 0; + mid = (mid + Math.imul(ah0, bl1)) | 0; + hi = (hi + Math.imul(ah0, bh1)) | 0; + var w1 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w1 >>> 26)) | 0; + w1 &= 0x3ffffff; + /* k = 2 */ + lo = Math.imul(al2, bl0); + mid = Math.imul(al2, bh0); + mid = (mid + Math.imul(ah2, bl0)) | 0; + hi = Math.imul(ah2, bh0); + lo = (lo + Math.imul(al1, bl1)) | 0; + mid = (mid + Math.imul(al1, bh1)) | 0; + mid = (mid + Math.imul(ah1, bl1)) | 0; + hi = (hi + Math.imul(ah1, bh1)) | 0; + lo = (lo + Math.imul(al0, bl2)) | 0; + mid = (mid + Math.imul(al0, bh2)) | 0; + mid = (mid + Math.imul(ah0, bl2)) | 0; + hi = (hi + Math.imul(ah0, bh2)) | 0; + var w2 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w2 >>> 26)) | 0; + w2 &= 0x3ffffff; + /* k = 3 */ + lo = Math.imul(al3, bl0); + mid = Math.imul(al3, bh0); + mid = (mid + Math.imul(ah3, bl0)) | 0; + hi = Math.imul(ah3, bh0); + lo = (lo + Math.imul(al2, bl1)) | 0; + mid = (mid + Math.imul(al2, bh1)) | 0; + mid = (mid + Math.imul(ah2, bl1)) | 0; + hi = (hi + Math.imul(ah2, bh1)) | 0; + lo = (lo + Math.imul(al1, bl2)) | 0; + mid = (mid + Math.imul(al1, bh2)) | 0; + mid = (mid + Math.imul(ah1, bl2)) | 0; + hi = (hi + Math.imul(ah1, bh2)) | 0; + lo = (lo + Math.imul(al0, bl3)) | 0; + mid = (mid + Math.imul(al0, bh3)) | 0; + mid = (mid + Math.imul(ah0, bl3)) | 0; + hi = (hi + Math.imul(ah0, bh3)) | 0; + var w3 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w3 >>> 26)) | 0; + w3 &= 0x3ffffff; + /* k = 4 */ + lo = Math.imul(al4, bl0); + mid = Math.imul(al4, bh0); + mid = (mid + Math.imul(ah4, bl0)) | 0; + hi = Math.imul(ah4, bh0); + lo = (lo + Math.imul(al3, bl1)) | 0; + mid = (mid + Math.imul(al3, bh1)) | 0; + mid = (mid + Math.imul(ah3, bl1)) | 0; + hi = (hi + Math.imul(ah3, bh1)) | 0; + lo = (lo + Math.imul(al2, bl2)) | 0; + mid = (mid + Math.imul(al2, bh2)) | 0; + mid = (mid + Math.imul(ah2, bl2)) | 0; + hi = (hi + Math.imul(ah2, bh2)) | 0; + lo = (lo + Math.imul(al1, bl3)) | 0; + mid = (mid + Math.imul(al1, bh3)) | 0; + mid = (mid + Math.imul(ah1, bl3)) | 0; + hi = (hi + Math.imul(ah1, bh3)) | 0; + lo = (lo + Math.imul(al0, bl4)) | 0; + mid = (mid + Math.imul(al0, bh4)) | 0; + mid = (mid + Math.imul(ah0, bl4)) | 0; + hi = (hi + Math.imul(ah0, bh4)) | 0; + var w4 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w4 >>> 26)) | 0; + w4 &= 0x3ffffff; + /* k = 5 */ + lo = Math.imul(al5, bl0); + mid = Math.imul(al5, bh0); + mid = (mid + Math.imul(ah5, bl0)) | 0; + hi = Math.imul(ah5, bh0); + lo = (lo + Math.imul(al4, bl1)) | 0; + mid = (mid + Math.imul(al4, bh1)) | 0; + mid = (mid + Math.imul(ah4, bl1)) | 0; + hi = (hi + Math.imul(ah4, bh1)) | 0; + lo = (lo + Math.imul(al3, bl2)) | 0; + mid = (mid + Math.imul(al3, bh2)) | 0; + mid = (mid + Math.imul(ah3, bl2)) | 0; + hi = (hi + Math.imul(ah3, bh2)) | 0; + lo = (lo + Math.imul(al2, bl3)) | 0; + mid = (mid + Math.imul(al2, bh3)) | 0; + mid = (mid + Math.imul(ah2, bl3)) | 0; + hi = (hi + Math.imul(ah2, bh3)) | 0; + lo = (lo + Math.imul(al1, bl4)) | 0; + mid = (mid + Math.imul(al1, bh4)) | 0; + mid = (mid + Math.imul(ah1, bl4)) | 0; + hi = (hi + Math.imul(ah1, bh4)) | 0; + lo = (lo + Math.imul(al0, bl5)) | 0; + mid = (mid + Math.imul(al0, bh5)) | 0; + mid = (mid + Math.imul(ah0, bl5)) | 0; + hi = (hi + Math.imul(ah0, bh5)) | 0; + var w5 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w5 >>> 26)) | 0; + w5 &= 0x3ffffff; + /* k = 6 */ + lo = Math.imul(al6, bl0); + mid = Math.imul(al6, bh0); + mid = (mid + Math.imul(ah6, bl0)) | 0; + hi = Math.imul(ah6, bh0); + lo = (lo + Math.imul(al5, bl1)) | 0; + mid = (mid + Math.imul(al5, bh1)) | 0; + mid = (mid + Math.imul(ah5, bl1)) | 0; + hi = (hi + Math.imul(ah5, bh1)) | 0; + lo = (lo + Math.imul(al4, bl2)) | 0; + mid = (mid + Math.imul(al4, bh2)) | 0; + mid = (mid + Math.imul(ah4, bl2)) | 0; + hi = (hi + Math.imul(ah4, bh2)) | 0; + lo = (lo + Math.imul(al3, bl3)) | 0; + mid = (mid + Math.imul(al3, bh3)) | 0; + mid = (mid + Math.imul(ah3, bl3)) | 0; + hi = (hi + Math.imul(ah3, bh3)) | 0; + lo = (lo + Math.imul(al2, bl4)) | 0; + mid = (mid + Math.imul(al2, bh4)) | 0; + mid = (mid + Math.imul(ah2, bl4)) | 0; + hi = (hi + Math.imul(ah2, bh4)) | 0; + lo = (lo + Math.imul(al1, bl5)) | 0; + mid = (mid + Math.imul(al1, bh5)) | 0; + mid = (mid + Math.imul(ah1, bl5)) | 0; + hi = (hi + Math.imul(ah1, bh5)) | 0; + lo = (lo + Math.imul(al0, bl6)) | 0; + mid = (mid + Math.imul(al0, bh6)) | 0; + mid = (mid + Math.imul(ah0, bl6)) | 0; + hi = (hi + Math.imul(ah0, bh6)) | 0; + var w6 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w6 >>> 26)) | 0; + w6 &= 0x3ffffff; + /* k = 7 */ + lo = Math.imul(al7, bl0); + mid = Math.imul(al7, bh0); + mid = (mid + Math.imul(ah7, bl0)) | 0; + hi = Math.imul(ah7, bh0); + lo = (lo + Math.imul(al6, bl1)) | 0; + mid = (mid + Math.imul(al6, bh1)) | 0; + mid = (mid + Math.imul(ah6, bl1)) | 0; + hi = (hi + Math.imul(ah6, bh1)) | 0; + lo = (lo + Math.imul(al5, bl2)) | 0; + mid = (mid + Math.imul(al5, bh2)) | 0; + mid = (mid + Math.imul(ah5, bl2)) | 0; + hi = (hi + Math.imul(ah5, bh2)) | 0; + lo = (lo + Math.imul(al4, bl3)) | 0; + mid = (mid + Math.imul(al4, bh3)) | 0; + mid = (mid + Math.imul(ah4, bl3)) | 0; + hi = (hi + Math.imul(ah4, bh3)) | 0; + lo = (lo + Math.imul(al3, bl4)) | 0; + mid = (mid + Math.imul(al3, bh4)) | 0; + mid = (mid + Math.imul(ah3, bl4)) | 0; + hi = (hi + Math.imul(ah3, bh4)) | 0; + lo = (lo + Math.imul(al2, bl5)) | 0; + mid = (mid + Math.imul(al2, bh5)) | 0; + mid = (mid + Math.imul(ah2, bl5)) | 0; + hi = (hi + Math.imul(ah2, bh5)) | 0; + lo = (lo + Math.imul(al1, bl6)) | 0; + mid = (mid + Math.imul(al1, bh6)) | 0; + mid = (mid + Math.imul(ah1, bl6)) | 0; + hi = (hi + Math.imul(ah1, bh6)) | 0; + lo = (lo + Math.imul(al0, bl7)) | 0; + mid = (mid + Math.imul(al0, bh7)) | 0; + mid = (mid + Math.imul(ah0, bl7)) | 0; + hi = (hi + Math.imul(ah0, bh7)) | 0; + var w7 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w7 >>> 26)) | 0; + w7 &= 0x3ffffff; + /* k = 8 */ + lo = Math.imul(al8, bl0); + mid = Math.imul(al8, bh0); + mid = (mid + Math.imul(ah8, bl0)) | 0; + hi = Math.imul(ah8, bh0); + lo = (lo + Math.imul(al7, bl1)) | 0; + mid = (mid + Math.imul(al7, bh1)) | 0; + mid = (mid + Math.imul(ah7, bl1)) | 0; + hi = (hi + Math.imul(ah7, bh1)) | 0; + lo = (lo + Math.imul(al6, bl2)) | 0; + mid = (mid + Math.imul(al6, bh2)) | 0; + mid = (mid + Math.imul(ah6, bl2)) | 0; + hi = (hi + Math.imul(ah6, bh2)) | 0; + lo = (lo + Math.imul(al5, bl3)) | 0; + mid = (mid + Math.imul(al5, bh3)) | 0; + mid = (mid + Math.imul(ah5, bl3)) | 0; + hi = (hi + Math.imul(ah5, bh3)) | 0; + lo = (lo + Math.imul(al4, bl4)) | 0; + mid = (mid + Math.imul(al4, bh4)) | 0; + mid = (mid + Math.imul(ah4, bl4)) | 0; + hi = (hi + Math.imul(ah4, bh4)) | 0; + lo = (lo + Math.imul(al3, bl5)) | 0; + mid = (mid + Math.imul(al3, bh5)) | 0; + mid = (mid + Math.imul(ah3, bl5)) | 0; + hi = (hi + Math.imul(ah3, bh5)) | 0; + lo = (lo + Math.imul(al2, bl6)) | 0; + mid = (mid + Math.imul(al2, bh6)) | 0; + mid = (mid + Math.imul(ah2, bl6)) | 0; + hi = (hi + Math.imul(ah2, bh6)) | 0; + lo = (lo + Math.imul(al1, bl7)) | 0; + mid = (mid + Math.imul(al1, bh7)) | 0; + mid = (mid + Math.imul(ah1, bl7)) | 0; + hi = (hi + Math.imul(ah1, bh7)) | 0; + lo = (lo + Math.imul(al0, bl8)) | 0; + mid = (mid + Math.imul(al0, bh8)) | 0; + mid = (mid + Math.imul(ah0, bl8)) | 0; + hi = (hi + Math.imul(ah0, bh8)) | 0; + var w8 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w8 >>> 26)) | 0; + w8 &= 0x3ffffff; + /* k = 9 */ + lo = Math.imul(al9, bl0); + mid = Math.imul(al9, bh0); + mid = (mid + Math.imul(ah9, bl0)) | 0; + hi = Math.imul(ah9, bh0); + lo = (lo + Math.imul(al8, bl1)) | 0; + mid = (mid + Math.imul(al8, bh1)) | 0; + mid = (mid + Math.imul(ah8, bl1)) | 0; + hi = (hi + Math.imul(ah8, bh1)) | 0; + lo = (lo + Math.imul(al7, bl2)) | 0; + mid = (mid + Math.imul(al7, bh2)) | 0; + mid = (mid + Math.imul(ah7, bl2)) | 0; + hi = (hi + Math.imul(ah7, bh2)) | 0; + lo = (lo + Math.imul(al6, bl3)) | 0; + mid = (mid + Math.imul(al6, bh3)) | 0; + mid = (mid + Math.imul(ah6, bl3)) | 0; + hi = (hi + Math.imul(ah6, bh3)) | 0; + lo = (lo + Math.imul(al5, bl4)) | 0; + mid = (mid + Math.imul(al5, bh4)) | 0; + mid = (mid + Math.imul(ah5, bl4)) | 0; + hi = (hi + Math.imul(ah5, bh4)) | 0; + lo = (lo + Math.imul(al4, bl5)) | 0; + mid = (mid + Math.imul(al4, bh5)) | 0; + mid = (mid + Math.imul(ah4, bl5)) | 0; + hi = (hi + Math.imul(ah4, bh5)) | 0; + lo = (lo + Math.imul(al3, bl6)) | 0; + mid = (mid + Math.imul(al3, bh6)) | 0; + mid = (mid + Math.imul(ah3, bl6)) | 0; + hi = (hi + Math.imul(ah3, bh6)) | 0; + lo = (lo + Math.imul(al2, bl7)) | 0; + mid = (mid + Math.imul(al2, bh7)) | 0; + mid = (mid + Math.imul(ah2, bl7)) | 0; + hi = (hi + Math.imul(ah2, bh7)) | 0; + lo = (lo + Math.imul(al1, bl8)) | 0; + mid = (mid + Math.imul(al1, bh8)) | 0; + mid = (mid + Math.imul(ah1, bl8)) | 0; + hi = (hi + Math.imul(ah1, bh8)) | 0; + lo = (lo + Math.imul(al0, bl9)) | 0; + mid = (mid + Math.imul(al0, bh9)) | 0; + mid = (mid + Math.imul(ah0, bl9)) | 0; + hi = (hi + Math.imul(ah0, bh9)) | 0; + var w9 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w9 >>> 26)) | 0; + w9 &= 0x3ffffff; + /* k = 10 */ + lo = Math.imul(al9, bl1); + mid = Math.imul(al9, bh1); + mid = (mid + Math.imul(ah9, bl1)) | 0; + hi = Math.imul(ah9, bh1); + lo = (lo + Math.imul(al8, bl2)) | 0; + mid = (mid + Math.imul(al8, bh2)) | 0; + mid = (mid + Math.imul(ah8, bl2)) | 0; + hi = (hi + Math.imul(ah8, bh2)) | 0; + lo = (lo + Math.imul(al7, bl3)) | 0; + mid = (mid + Math.imul(al7, bh3)) | 0; + mid = (mid + Math.imul(ah7, bl3)) | 0; + hi = (hi + Math.imul(ah7, bh3)) | 0; + lo = (lo + Math.imul(al6, bl4)) | 0; + mid = (mid + Math.imul(al6, bh4)) | 0; + mid = (mid + Math.imul(ah6, bl4)) | 0; + hi = (hi + Math.imul(ah6, bh4)) | 0; + lo = (lo + Math.imul(al5, bl5)) | 0; + mid = (mid + Math.imul(al5, bh5)) | 0; + mid = (mid + Math.imul(ah5, bl5)) | 0; + hi = (hi + Math.imul(ah5, bh5)) | 0; + lo = (lo + Math.imul(al4, bl6)) | 0; + mid = (mid + Math.imul(al4, bh6)) | 0; + mid = (mid + Math.imul(ah4, bl6)) | 0; + hi = (hi + Math.imul(ah4, bh6)) | 0; + lo = (lo + Math.imul(al3, bl7)) | 0; + mid = (mid + Math.imul(al3, bh7)) | 0; + mid = (mid + Math.imul(ah3, bl7)) | 0; + hi = (hi + Math.imul(ah3, bh7)) | 0; + lo = (lo + Math.imul(al2, bl8)) | 0; + mid = (mid + Math.imul(al2, bh8)) | 0; + mid = (mid + Math.imul(ah2, bl8)) | 0; + hi = (hi + Math.imul(ah2, bh8)) | 0; + lo = (lo + Math.imul(al1, bl9)) | 0; + mid = (mid + Math.imul(al1, bh9)) | 0; + mid = (mid + Math.imul(ah1, bl9)) | 0; + hi = (hi + Math.imul(ah1, bh9)) | 0; + var w10 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w10 >>> 26)) | 0; + w10 &= 0x3ffffff; + /* k = 11 */ + lo = Math.imul(al9, bl2); + mid = Math.imul(al9, bh2); + mid = (mid + Math.imul(ah9, bl2)) | 0; + hi = Math.imul(ah9, bh2); + lo = (lo + Math.imul(al8, bl3)) | 0; + mid = (mid + Math.imul(al8, bh3)) | 0; + mid = (mid + Math.imul(ah8, bl3)) | 0; + hi = (hi + Math.imul(ah8, bh3)) | 0; + lo = (lo + Math.imul(al7, bl4)) | 0; + mid = (mid + Math.imul(al7, bh4)) | 0; + mid = (mid + Math.imul(ah7, bl4)) | 0; + hi = (hi + Math.imul(ah7, bh4)) | 0; + lo = (lo + Math.imul(al6, bl5)) | 0; + mid = (mid + Math.imul(al6, bh5)) | 0; + mid = (mid + Math.imul(ah6, bl5)) | 0; + hi = (hi + Math.imul(ah6, bh5)) | 0; + lo = (lo + Math.imul(al5, bl6)) | 0; + mid = (mid + Math.imul(al5, bh6)) | 0; + mid = (mid + Math.imul(ah5, bl6)) | 0; + hi = (hi + Math.imul(ah5, bh6)) | 0; + lo = (lo + Math.imul(al4, bl7)) | 0; + mid = (mid + Math.imul(al4, bh7)) | 0; + mid = (mid + Math.imul(ah4, bl7)) | 0; + hi = (hi + Math.imul(ah4, bh7)) | 0; + lo = (lo + Math.imul(al3, bl8)) | 0; + mid = (mid + Math.imul(al3, bh8)) | 0; + mid = (mid + Math.imul(ah3, bl8)) | 0; + hi = (hi + Math.imul(ah3, bh8)) | 0; + lo = (lo + Math.imul(al2, bl9)) | 0; + mid = (mid + Math.imul(al2, bh9)) | 0; + mid = (mid + Math.imul(ah2, bl9)) | 0; + hi = (hi + Math.imul(ah2, bh9)) | 0; + var w11 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w11 >>> 26)) | 0; + w11 &= 0x3ffffff; + /* k = 12 */ + lo = Math.imul(al9, bl3); + mid = Math.imul(al9, bh3); + mid = (mid + Math.imul(ah9, bl3)) | 0; + hi = Math.imul(ah9, bh3); + lo = (lo + Math.imul(al8, bl4)) | 0; + mid = (mid + Math.imul(al8, bh4)) | 0; + mid = (mid + Math.imul(ah8, bl4)) | 0; + hi = (hi + Math.imul(ah8, bh4)) | 0; + lo = (lo + Math.imul(al7, bl5)) | 0; + mid = (mid + Math.imul(al7, bh5)) | 0; + mid = (mid + Math.imul(ah7, bl5)) | 0; + hi = (hi + Math.imul(ah7, bh5)) | 0; + lo = (lo + Math.imul(al6, bl6)) | 0; + mid = (mid + Math.imul(al6, bh6)) | 0; + mid = (mid + Math.imul(ah6, bl6)) | 0; + hi = (hi + Math.imul(ah6, bh6)) | 0; + lo = (lo + Math.imul(al5, bl7)) | 0; + mid = (mid + Math.imul(al5, bh7)) | 0; + mid = (mid + Math.imul(ah5, bl7)) | 0; + hi = (hi + Math.imul(ah5, bh7)) | 0; + lo = (lo + Math.imul(al4, bl8)) | 0; + mid = (mid + Math.imul(al4, bh8)) | 0; + mid = (mid + Math.imul(ah4, bl8)) | 0; + hi = (hi + Math.imul(ah4, bh8)) | 0; + lo = (lo + Math.imul(al3, bl9)) | 0; + mid = (mid + Math.imul(al3, bh9)) | 0; + mid = (mid + Math.imul(ah3, bl9)) | 0; + hi = (hi + Math.imul(ah3, bh9)) | 0; + var w12 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w12 >>> 26)) | 0; + w12 &= 0x3ffffff; + /* k = 13 */ + lo = Math.imul(al9, bl4); + mid = Math.imul(al9, bh4); + mid = (mid + Math.imul(ah9, bl4)) | 0; + hi = Math.imul(ah9, bh4); + lo = (lo + Math.imul(al8, bl5)) | 0; + mid = (mid + Math.imul(al8, bh5)) | 0; + mid = (mid + Math.imul(ah8, bl5)) | 0; + hi = (hi + Math.imul(ah8, bh5)) | 0; + lo = (lo + Math.imul(al7, bl6)) | 0; + mid = (mid + Math.imul(al7, bh6)) | 0; + mid = (mid + Math.imul(ah7, bl6)) | 0; + hi = (hi + Math.imul(ah7, bh6)) | 0; + lo = (lo + Math.imul(al6, bl7)) | 0; + mid = (mid + Math.imul(al6, bh7)) | 0; + mid = (mid + Math.imul(ah6, bl7)) | 0; + hi = (hi + Math.imul(ah6, bh7)) | 0; + lo = (lo + Math.imul(al5, bl8)) | 0; + mid = (mid + Math.imul(al5, bh8)) | 0; + mid = (mid + Math.imul(ah5, bl8)) | 0; + hi = (hi + Math.imul(ah5, bh8)) | 0; + lo = (lo + Math.imul(al4, bl9)) | 0; + mid = (mid + Math.imul(al4, bh9)) | 0; + mid = (mid + Math.imul(ah4, bl9)) | 0; + hi = (hi + Math.imul(ah4, bh9)) | 0; + var w13 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w13 >>> 26)) | 0; + w13 &= 0x3ffffff; + /* k = 14 */ + lo = Math.imul(al9, bl5); + mid = Math.imul(al9, bh5); + mid = (mid + Math.imul(ah9, bl5)) | 0; + hi = Math.imul(ah9, bh5); + lo = (lo + Math.imul(al8, bl6)) | 0; + mid = (mid + Math.imul(al8, bh6)) | 0; + mid = (mid + Math.imul(ah8, bl6)) | 0; + hi = (hi + Math.imul(ah8, bh6)) | 0; + lo = (lo + Math.imul(al7, bl7)) | 0; + mid = (mid + Math.imul(al7, bh7)) | 0; + mid = (mid + Math.imul(ah7, bl7)) | 0; + hi = (hi + Math.imul(ah7, bh7)) | 0; + lo = (lo + Math.imul(al6, bl8)) | 0; + mid = (mid + Math.imul(al6, bh8)) | 0; + mid = (mid + Math.imul(ah6, bl8)) | 0; + hi = (hi + Math.imul(ah6, bh8)) | 0; + lo = (lo + Math.imul(al5, bl9)) | 0; + mid = (mid + Math.imul(al5, bh9)) | 0; + mid = (mid + Math.imul(ah5, bl9)) | 0; + hi = (hi + Math.imul(ah5, bh9)) | 0; + var w14 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w14 >>> 26)) | 0; + w14 &= 0x3ffffff; + /* k = 15 */ + lo = Math.imul(al9, bl6); + mid = Math.imul(al9, bh6); + mid = (mid + Math.imul(ah9, bl6)) | 0; + hi = Math.imul(ah9, bh6); + lo = (lo + Math.imul(al8, bl7)) | 0; + mid = (mid + Math.imul(al8, bh7)) | 0; + mid = (mid + Math.imul(ah8, bl7)) | 0; + hi = (hi + Math.imul(ah8, bh7)) | 0; + lo = (lo + Math.imul(al7, bl8)) | 0; + mid = (mid + Math.imul(al7, bh8)) | 0; + mid = (mid + Math.imul(ah7, bl8)) | 0; + hi = (hi + Math.imul(ah7, bh8)) | 0; + lo = (lo + Math.imul(al6, bl9)) | 0; + mid = (mid + Math.imul(al6, bh9)) | 0; + mid = (mid + Math.imul(ah6, bl9)) | 0; + hi = (hi + Math.imul(ah6, bh9)) | 0; + var w15 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w15 >>> 26)) | 0; + w15 &= 0x3ffffff; + /* k = 16 */ + lo = Math.imul(al9, bl7); + mid = Math.imul(al9, bh7); + mid = (mid + Math.imul(ah9, bl7)) | 0; + hi = Math.imul(ah9, bh7); + lo = (lo + Math.imul(al8, bl8)) | 0; + mid = (mid + Math.imul(al8, bh8)) | 0; + mid = (mid + Math.imul(ah8, bl8)) | 0; + hi = (hi + Math.imul(ah8, bh8)) | 0; + lo = (lo + Math.imul(al7, bl9)) | 0; + mid = (mid + Math.imul(al7, bh9)) | 0; + mid = (mid + Math.imul(ah7, bl9)) | 0; + hi = (hi + Math.imul(ah7, bh9)) | 0; + var w16 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w16 >>> 26)) | 0; + w16 &= 0x3ffffff; + /* k = 17 */ + lo = Math.imul(al9, bl8); + mid = Math.imul(al9, bh8); + mid = (mid + Math.imul(ah9, bl8)) | 0; + hi = Math.imul(ah9, bh8); + lo = (lo + Math.imul(al8, bl9)) | 0; + mid = (mid + Math.imul(al8, bh9)) | 0; + mid = (mid + Math.imul(ah8, bl9)) | 0; + hi = (hi + Math.imul(ah8, bh9)) | 0; + var w17 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w17 >>> 26)) | 0; + w17 &= 0x3ffffff; + /* k = 18 */ + lo = Math.imul(al9, bl9); + mid = Math.imul(al9, bh9); + mid = (mid + Math.imul(ah9, bl9)) | 0; + hi = Math.imul(ah9, bh9); + var w18 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w18 >>> 26)) | 0; + w18 &= 0x3ffffff; + o[0] = w0; + o[1] = w1; + o[2] = w2; + o[3] = w3; + o[4] = w4; + o[5] = w5; + o[6] = w6; + o[7] = w7; + o[8] = w8; + o[9] = w9; + o[10] = w10; + o[11] = w11; + o[12] = w12; + o[13] = w13; + o[14] = w14; + o[15] = w15; + o[16] = w16; + o[17] = w17; + o[18] = w18; + if (c !== 0) { + o[19] = c; + out.length++; + } + return out; + }; + + // Polyfill comb + if (!Math.imul) { + comb10MulTo = smallMulTo; + } + + function bigMulTo (self, num, out) { + out.negative = num.negative ^ self.negative; + out.length = self.length + num.length; + + var carry = 0; + var hncarry = 0; + for (var k = 0; k < out.length - 1; k++) { + // Sum all words with the same `i + j = k` and accumulate `ncarry`, + // note that ncarry could be >= 0x3ffffff + var ncarry = hncarry; + hncarry = 0; + var rword = carry & 0x3ffffff; + var maxJ = Math.min(k, num.length - 1); + for (var j = Math.max(0, k - self.length + 1); j <= maxJ; j++) { + var i = k - j; + var a = self.words[i] | 0; + var b = num.words[j] | 0; + var r = a * b; + + var lo = r & 0x3ffffff; + ncarry = (ncarry + ((r / 0x4000000) | 0)) | 0; + lo = (lo + rword) | 0; + rword = lo & 0x3ffffff; + ncarry = (ncarry + (lo >>> 26)) | 0; + + hncarry += ncarry >>> 26; + ncarry &= 0x3ffffff; + } + out.words[k] = rword; + carry = ncarry; + ncarry = hncarry; + } + if (carry !== 0) { + out.words[k] = carry; + } else { + out.length--; + } + + return out.strip(); + } + + function jumboMulTo (self, num, out) { + var fftm = new FFTM(); + return fftm.mulp(self, num, out); + } + + BN.prototype.mulTo = function mulTo (num, out) { + var res; + var len = this.length + num.length; + if (this.length === 10 && num.length === 10) { + res = comb10MulTo(this, num, out); + } else if (len < 63) { + res = smallMulTo(this, num, out); + } else if (len < 1024) { + res = bigMulTo(this, num, out); + } else { + res = jumboMulTo(this, num, out); + } + + return res; + }; + + // Cooley-Tukey algorithm for FFT + // slightly revisited to rely on looping instead of recursion + + function FFTM (x, y) { + this.x = x; + this.y = y; + } + + FFTM.prototype.makeRBT = function makeRBT (N) { + var t = new Array(N); + var l = BN.prototype._countBits(N) - 1; + for (var i = 0; i < N; i++) { + t[i] = this.revBin(i, l, N); + } + + return t; + }; + + // Returns binary-reversed representation of `x` + FFTM.prototype.revBin = function revBin (x, l, N) { + if (x === 0 || x === N - 1) return x; + + var rb = 0; + for (var i = 0; i < l; i++) { + rb |= (x & 1) << (l - i - 1); + x >>= 1; + } + + return rb; + }; + + // Performs "tweedling" phase, therefore 'emulating' + // behaviour of the recursive algorithm + FFTM.prototype.permute = function permute (rbt, rws, iws, rtws, itws, N) { + for (var i = 0; i < N; i++) { + rtws[i] = rws[rbt[i]]; + itws[i] = iws[rbt[i]]; + } + }; + + FFTM.prototype.transform = function transform (rws, iws, rtws, itws, N, rbt) { + this.permute(rbt, rws, iws, rtws, itws, N); + + for (var s = 1; s < N; s <<= 1) { + var l = s << 1; + + var rtwdf = Math.cos(2 * Math.PI / l); + var itwdf = Math.sin(2 * Math.PI / l); + + for (var p = 0; p < N; p += l) { + var rtwdf_ = rtwdf; + var itwdf_ = itwdf; + + for (var j = 0; j < s; j++) { + var re = rtws[p + j]; + var ie = itws[p + j]; + + var ro = rtws[p + j + s]; + var io = itws[p + j + s]; + + var rx = rtwdf_ * ro - itwdf_ * io; + + io = rtwdf_ * io + itwdf_ * ro; + ro = rx; + + rtws[p + j] = re + ro; + itws[p + j] = ie + io; + + rtws[p + j + s] = re - ro; + itws[p + j + s] = ie - io; + + /* jshint maxdepth : false */ + if (j !== l) { + rx = rtwdf * rtwdf_ - itwdf * itwdf_; + + itwdf_ = rtwdf * itwdf_ + itwdf * rtwdf_; + rtwdf_ = rx; + } + } + } + } + }; + + FFTM.prototype.guessLen13b = function guessLen13b (n, m) { + var N = Math.max(m, n) | 1; + var odd = N & 1; + var i = 0; + for (N = N / 2 | 0; N; N = N >>> 1) { + i++; + } + + return 1 << i + 1 + odd; + }; + + FFTM.prototype.conjugate = function conjugate (rws, iws, N) { + if (N <= 1) return; + + for (var i = 0; i < N / 2; i++) { + var t = rws[i]; + + rws[i] = rws[N - i - 1]; + rws[N - i - 1] = t; + + t = iws[i]; + + iws[i] = -iws[N - i - 1]; + iws[N - i - 1] = -t; + } + }; + + FFTM.prototype.normalize13b = function normalize13b (ws, N) { + var carry = 0; + for (var i = 0; i < N / 2; i++) { + var w = Math.round(ws[2 * i + 1] / N) * 0x2000 + + Math.round(ws[2 * i] / N) + + carry; + + ws[i] = w & 0x3ffffff; + + if (w < 0x4000000) { + carry = 0; + } else { + carry = w / 0x4000000 | 0; + } + } + + return ws; + }; + + FFTM.prototype.convert13b = function convert13b (ws, len, rws, N) { + var carry = 0; + for (var i = 0; i < len; i++) { + carry = carry + (ws[i] | 0); + + rws[2 * i] = carry & 0x1fff; carry = carry >>> 13; + rws[2 * i + 1] = carry & 0x1fff; carry = carry >>> 13; + } + + // Pad with zeroes + for (i = 2 * len; i < N; ++i) { + rws[i] = 0; + } + + assert(carry === 0); + assert((carry & ~0x1fff) === 0); + }; + + FFTM.prototype.stub = function stub (N) { + var ph = new Array(N); + for (var i = 0; i < N; i++) { + ph[i] = 0; + } + + return ph; + }; + + FFTM.prototype.mulp = function mulp (x, y, out) { + var N = 2 * this.guessLen13b(x.length, y.length); + + var rbt = this.makeRBT(N); + + var _ = this.stub(N); + + var rws = new Array(N); + var rwst = new Array(N); + var iwst = new Array(N); + + var nrws = new Array(N); + var nrwst = new Array(N); + var niwst = new Array(N); + + var rmws = out.words; + rmws.length = N; + + this.convert13b(x.words, x.length, rws, N); + this.convert13b(y.words, y.length, nrws, N); + + this.transform(rws, _, rwst, iwst, N, rbt); + this.transform(nrws, _, nrwst, niwst, N, rbt); + + for (var i = 0; i < N; i++) { + var rx = rwst[i] * nrwst[i] - iwst[i] * niwst[i]; + iwst[i] = rwst[i] * niwst[i] + iwst[i] * nrwst[i]; + rwst[i] = rx; + } + + this.conjugate(rwst, iwst, N); + this.transform(rwst, iwst, rmws, _, N, rbt); + this.conjugate(rmws, _, N); + this.normalize13b(rmws, N); + + out.negative = x.negative ^ y.negative; + out.length = x.length + y.length; + return out.strip(); + }; + + // Multiply `this` by `num` + BN.prototype.mul = function mul (num) { + var out = new BN(null); + out.words = new Array(this.length + num.length); + return this.mulTo(num, out); + }; + + // Multiply employing FFT + BN.prototype.mulf = function mulf (num) { + var out = new BN(null); + out.words = new Array(this.length + num.length); + return jumboMulTo(this, num, out); + }; + + // In-place Multiplication + BN.prototype.imul = function imul (num) { + return this.clone().mulTo(num, this); + }; + + BN.prototype.imuln = function imuln (num) { + assert(typeof num === 'number'); + assert(num < 0x4000000); + + // Carry + var carry = 0; + for (var i = 0; i < this.length; i++) { + var w = (this.words[i] | 0) * num; + var lo = (w & 0x3ffffff) + (carry & 0x3ffffff); + carry >>= 26; + carry += (w / 0x4000000) | 0; + // NOTE: lo is 27bit maximum + carry += lo >>> 26; + this.words[i] = lo & 0x3ffffff; + } + + if (carry !== 0) { + this.words[i] = carry; + this.length++; + } + + return this; + }; + + BN.prototype.muln = function muln (num) { + return this.clone().imuln(num); + }; + + // `this` * `this` + BN.prototype.sqr = function sqr () { + return this.mul(this); + }; + + // `this` * `this` in-place + BN.prototype.isqr = function isqr () { + return this.imul(this.clone()); + }; + + // Math.pow(`this`, `num`) + BN.prototype.pow = function pow (num) { + var w = toBitArray(num); + if (w.length === 0) return new BN(1); + + // Skip leading zeroes + var res = this; + for (var i = 0; i < w.length; i++, res = res.sqr()) { + if (w[i] !== 0) break; + } + + if (++i < w.length) { + for (var q = res.sqr(); i < w.length; i++, q = q.sqr()) { + if (w[i] === 0) continue; + + res = res.mul(q); + } + } + + return res; + }; + + // Shift-left in-place + BN.prototype.iushln = function iushln (bits) { + assert(typeof bits === 'number' && bits >= 0); + var r = bits % 26; + var s = (bits - r) / 26; + var carryMask = (0x3ffffff >>> (26 - r)) << (26 - r); + var i; + + if (r !== 0) { + var carry = 0; + + for (i = 0; i < this.length; i++) { + var newCarry = this.words[i] & carryMask; + var c = ((this.words[i] | 0) - newCarry) << r; + this.words[i] = c | carry; + carry = newCarry >>> (26 - r); + } + + if (carry) { + this.words[i] = carry; + this.length++; + } + } + + if (s !== 0) { + for (i = this.length - 1; i >= 0; i--) { + this.words[i + s] = this.words[i]; + } + + for (i = 0; i < s; i++) { + this.words[i] = 0; + } + + this.length += s; + } + + return this.strip(); + }; + + BN.prototype.ishln = function ishln (bits) { + // TODO(indutny): implement me + assert(this.negative === 0); + return this.iushln(bits); + }; + + // Shift-right in-place + // NOTE: `hint` is a lowest bit before trailing zeroes + // NOTE: if `extended` is present - it will be filled with destroyed bits + BN.prototype.iushrn = function iushrn (bits, hint, extended) { + assert(typeof bits === 'number' && bits >= 0); + var h; + if (hint) { + h = (hint - (hint % 26)) / 26; + } else { + h = 0; + } + + var r = bits % 26; + var s = Math.min((bits - r) / 26, this.length); + var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r); + var maskedWords = extended; + + h -= s; + h = Math.max(0, h); + + // Extended mode, copy masked part + if (maskedWords) { + for (var i = 0; i < s; i++) { + maskedWords.words[i] = this.words[i]; + } + maskedWords.length = s; + } + + if (s === 0) { + // No-op, we should not move anything at all + } else if (this.length > s) { + this.length -= s; + for (i = 0; i < this.length; i++) { + this.words[i] = this.words[i + s]; + } + } else { + this.words[0] = 0; + this.length = 1; + } + + var carry = 0; + for (i = this.length - 1; i >= 0 && (carry !== 0 || i >= h); i--) { + var word = this.words[i] | 0; + this.words[i] = (carry << (26 - r)) | (word >>> r); + carry = word & mask; + } + + // Push carried bits as a mask + if (maskedWords && carry !== 0) { + maskedWords.words[maskedWords.length++] = carry; + } + + if (this.length === 0) { + this.words[0] = 0; + this.length = 1; + } + + return this.strip(); + }; + + BN.prototype.ishrn = function ishrn (bits, hint, extended) { + // TODO(indutny): implement me + assert(this.negative === 0); + return this.iushrn(bits, hint, extended); + }; + + // Shift-left + BN.prototype.shln = function shln (bits) { + return this.clone().ishln(bits); + }; + + BN.prototype.ushln = function ushln (bits) { + return this.clone().iushln(bits); + }; + + // Shift-right + BN.prototype.shrn = function shrn (bits) { + return this.clone().ishrn(bits); + }; + + BN.prototype.ushrn = function ushrn (bits) { + return this.clone().iushrn(bits); + }; + + // Test if n bit is set + BN.prototype.testn = function testn (bit) { + assert(typeof bit === 'number' && bit >= 0); + var r = bit % 26; + var s = (bit - r) / 26; + var q = 1 << r; + + // Fast case: bit is much higher than all existing words + if (this.length <= s) return false; + + // Check bit and return + var w = this.words[s]; + + return !!(w & q); + }; + + // Return only lowers bits of number (in-place) + BN.prototype.imaskn = function imaskn (bits) { + assert(typeof bits === 'number' && bits >= 0); + var r = bits % 26; + var s = (bits - r) / 26; + + assert(this.negative === 0, 'imaskn works only with positive numbers'); + + if (this.length <= s) { + return this; + } + + if (r !== 0) { + s++; + } + this.length = Math.min(s, this.length); + + if (r !== 0) { + var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r); + this.words[this.length - 1] &= mask; + } + + return this.strip(); + }; + + // Return only lowers bits of number + BN.prototype.maskn = function maskn (bits) { + return this.clone().imaskn(bits); + }; + + // Add plain number `num` to `this` + BN.prototype.iaddn = function iaddn (num) { + assert(typeof num === 'number'); + assert(num < 0x4000000); + if (num < 0) return this.isubn(-num); + + // Possible sign change + if (this.negative !== 0) { + if (this.length === 1 && (this.words[0] | 0) < num) { + this.words[0] = num - (this.words[0] | 0); + this.negative = 0; + return this; + } + + this.negative = 0; + this.isubn(num); + this.negative = 1; + return this; + } + + // Add without checks + return this._iaddn(num); + }; + + BN.prototype._iaddn = function _iaddn (num) { + this.words[0] += num; + + // Carry + for (var i = 0; i < this.length && this.words[i] >= 0x4000000; i++) { + this.words[i] -= 0x4000000; + if (i === this.length - 1) { + this.words[i + 1] = 1; + } else { + this.words[i + 1]++; + } + } + this.length = Math.max(this.length, i + 1); + + return this; + }; + + // Subtract plain number `num` from `this` + BN.prototype.isubn = function isubn (num) { + assert(typeof num === 'number'); + assert(num < 0x4000000); + if (num < 0) return this.iaddn(-num); + + if (this.negative !== 0) { + this.negative = 0; + this.iaddn(num); + this.negative = 1; + return this; + } + + this.words[0] -= num; + + if (this.length === 1 && this.words[0] < 0) { + this.words[0] = -this.words[0]; + this.negative = 1; + } else { + // Carry + for (var i = 0; i < this.length && this.words[i] < 0; i++) { + this.words[i] += 0x4000000; + this.words[i + 1] -= 1; + } + } + + return this.strip(); + }; + + BN.prototype.addn = function addn (num) { + return this.clone().iaddn(num); + }; + + BN.prototype.subn = function subn (num) { + return this.clone().isubn(num); + }; + + BN.prototype.iabs = function iabs () { + this.negative = 0; + + return this; + }; + + BN.prototype.abs = function abs () { + return this.clone().iabs(); + }; + + BN.prototype._ishlnsubmul = function _ishlnsubmul (num, mul, shift) { + var len = num.length + shift; + var i; + + this._expand(len); + + var w; + var carry = 0; + for (i = 0; i < num.length; i++) { + w = (this.words[i + shift] | 0) + carry; + var right = (num.words[i] | 0) * mul; + w -= right & 0x3ffffff; + carry = (w >> 26) - ((right / 0x4000000) | 0); + this.words[i + shift] = w & 0x3ffffff; + } + for (; i < this.length - shift; i++) { + w = (this.words[i + shift] | 0) + carry; + carry = w >> 26; + this.words[i + shift] = w & 0x3ffffff; + } + + if (carry === 0) return this.strip(); + + // Subtraction overflow + assert(carry === -1); + carry = 0; + for (i = 0; i < this.length; i++) { + w = -(this.words[i] | 0) + carry; + carry = w >> 26; + this.words[i] = w & 0x3ffffff; + } + this.negative = 1; + + return this.strip(); + }; + + BN.prototype._wordDiv = function _wordDiv (num, mode) { + var shift = this.length - num.length; + + var a = this.clone(); + var b = num; + + // Normalize + var bhi = b.words[b.length - 1] | 0; + var bhiBits = this._countBits(bhi); + shift = 26 - bhiBits; + if (shift !== 0) { + b = b.ushln(shift); + a.iushln(shift); + bhi = b.words[b.length - 1] | 0; + } + + // Initialize quotient + var m = a.length - b.length; + var q; + + if (mode !== 'mod') { + q = new BN(null); + q.length = m + 1; + q.words = new Array(q.length); + for (var i = 0; i < q.length; i++) { + q.words[i] = 0; + } + } + + var diff = a.clone()._ishlnsubmul(b, 1, m); + if (diff.negative === 0) { + a = diff; + if (q) { + q.words[m] = 1; + } + } + + for (var j = m - 1; j >= 0; j--) { + var qj = (a.words[b.length + j] | 0) * 0x4000000 + + (a.words[b.length + j - 1] | 0); + + // NOTE: (qj / bhi) is (0x3ffffff * 0x4000000 + 0x3ffffff) / 0x2000000 max + // (0x7ffffff) + qj = Math.min((qj / bhi) | 0, 0x3ffffff); + + a._ishlnsubmul(b, qj, j); + while (a.negative !== 0) { + qj--; + a.negative = 0; + a._ishlnsubmul(b, 1, j); + if (!a.isZero()) { + a.negative ^= 1; + } + } + if (q) { + q.words[j] = qj; + } + } + if (q) { + q.strip(); + } + a.strip(); + + // Denormalize + if (mode !== 'div' && shift !== 0) { + a.iushrn(shift); + } + + return { + div: q || null, + mod: a + }; + }; + + // NOTE: 1) `mode` can be set to `mod` to request mod only, + // to `div` to request div only, or be absent to + // request both div & mod + // 2) `positive` is true if unsigned mod is requested + BN.prototype.divmod = function divmod (num, mode, positive) { + assert(!num.isZero()); + + if (this.isZero()) { + return { + div: new BN(0), + mod: new BN(0) + }; + } + + var div, mod, res; + if (this.negative !== 0 && num.negative === 0) { + res = this.neg().divmod(num, mode); + + if (mode !== 'mod') { + div = res.div.neg(); + } + + if (mode !== 'div') { + mod = res.mod.neg(); + if (positive && mod.negative !== 0) { + mod.iadd(num); + } + } + + return { + div: div, + mod: mod + }; + } + + if (this.negative === 0 && num.negative !== 0) { + res = this.divmod(num.neg(), mode); + + if (mode !== 'mod') { + div = res.div.neg(); + } + + return { + div: div, + mod: res.mod + }; + } + + if ((this.negative & num.negative) !== 0) { + res = this.neg().divmod(num.neg(), mode); + + if (mode !== 'div') { + mod = res.mod.neg(); + if (positive && mod.negative !== 0) { + mod.isub(num); + } + } + + return { + div: res.div, + mod: mod + }; + } + + // Both numbers are positive at this point + + // Strip both numbers to approximate shift value + if (num.length > this.length || this.cmp(num) < 0) { + return { + div: new BN(0), + mod: this + }; + } + + // Very short reduction + if (num.length === 1) { + if (mode === 'div') { + return { + div: this.divn(num.words[0]), + mod: null + }; + } + + if (mode === 'mod') { + return { + div: null, + mod: new BN(this.modn(num.words[0])) + }; + } + + return { + div: this.divn(num.words[0]), + mod: new BN(this.modn(num.words[0])) + }; + } + + return this._wordDiv(num, mode); + }; + + // Find `this` / `num` + BN.prototype.div = function div (num) { + return this.divmod(num, 'div', false).div; + }; + + // Find `this` % `num` + BN.prototype.mod = function mod (num) { + return this.divmod(num, 'mod', false).mod; + }; + + BN.prototype.umod = function umod (num) { + return this.divmod(num, 'mod', true).mod; + }; + + // Find Round(`this` / `num`) + BN.prototype.divRound = function divRound (num) { + var dm = this.divmod(num); + + // Fast case - exact division + if (dm.mod.isZero()) return dm.div; + + var mod = dm.div.negative !== 0 ? dm.mod.isub(num) : dm.mod; + + var half = num.ushrn(1); + var r2 = num.andln(1); + var cmp = mod.cmp(half); + + // Round down + if (cmp < 0 || r2 === 1 && cmp === 0) return dm.div; + + // Round up + return dm.div.negative !== 0 ? dm.div.isubn(1) : dm.div.iaddn(1); + }; + + BN.prototype.modn = function modn (num) { + assert(num <= 0x3ffffff); + var p = (1 << 26) % num; + + var acc = 0; + for (var i = this.length - 1; i >= 0; i--) { + acc = (p * acc + (this.words[i] | 0)) % num; + } + + return acc; + }; + + // In-place division by number + BN.prototype.idivn = function idivn (num) { + assert(num <= 0x3ffffff); + + var carry = 0; + for (var i = this.length - 1; i >= 0; i--) { + var w = (this.words[i] | 0) + carry * 0x4000000; + this.words[i] = (w / num) | 0; + carry = w % num; + } + + return this.strip(); + }; + + BN.prototype.divn = function divn (num) { + return this.clone().idivn(num); + }; + + BN.prototype.egcd = function egcd (p) { + assert(p.negative === 0); + assert(!p.isZero()); + + var x = this; + var y = p.clone(); + + if (x.negative !== 0) { + x = x.umod(p); + } else { + x = x.clone(); + } + + // A * x + B * y = x + var A = new BN(1); + var B = new BN(0); + + // C * x + D * y = y + var C = new BN(0); + var D = new BN(1); + + var g = 0; + + while (x.isEven() && y.isEven()) { + x.iushrn(1); + y.iushrn(1); + ++g; + } + + var yp = y.clone(); + var xp = x.clone(); + + while (!x.isZero()) { + for (var i = 0, im = 1; (x.words[0] & im) === 0 && i < 26; ++i, im <<= 1); + if (i > 0) { + x.iushrn(i); + while (i-- > 0) { + if (A.isOdd() || B.isOdd()) { + A.iadd(yp); + B.isub(xp); + } + + A.iushrn(1); + B.iushrn(1); + } + } + + for (var j = 0, jm = 1; (y.words[0] & jm) === 0 && j < 26; ++j, jm <<= 1); + if (j > 0) { + y.iushrn(j); + while (j-- > 0) { + if (C.isOdd() || D.isOdd()) { + C.iadd(yp); + D.isub(xp); + } + + C.iushrn(1); + D.iushrn(1); + } + } + + if (x.cmp(y) >= 0) { + x.isub(y); + A.isub(C); + B.isub(D); + } else { + y.isub(x); + C.isub(A); + D.isub(B); + } + } + + return { + a: C, + b: D, + gcd: y.iushln(g) + }; + }; + + // This is reduced incarnation of the binary EEA + // above, designated to invert members of the + // _prime_ fields F(p) at a maximal speed + BN.prototype._invmp = function _invmp (p) { + assert(p.negative === 0); + assert(!p.isZero()); + + var a = this; + var b = p.clone(); + + if (a.negative !== 0) { + a = a.umod(p); + } else { + a = a.clone(); + } + + var x1 = new BN(1); + var x2 = new BN(0); + + var delta = b.clone(); + + while (a.cmpn(1) > 0 && b.cmpn(1) > 0) { + for (var i = 0, im = 1; (a.words[0] & im) === 0 && i < 26; ++i, im <<= 1); + if (i > 0) { + a.iushrn(i); + while (i-- > 0) { + if (x1.isOdd()) { + x1.iadd(delta); + } + + x1.iushrn(1); + } + } + + for (var j = 0, jm = 1; (b.words[0] & jm) === 0 && j < 26; ++j, jm <<= 1); + if (j > 0) { + b.iushrn(j); + while (j-- > 0) { + if (x2.isOdd()) { + x2.iadd(delta); + } + + x2.iushrn(1); + } + } + + if (a.cmp(b) >= 0) { + a.isub(b); + x1.isub(x2); + } else { + b.isub(a); + x2.isub(x1); + } + } + + var res; + if (a.cmpn(1) === 0) { + res = x1; + } else { + res = x2; + } + + if (res.cmpn(0) < 0) { + res.iadd(p); + } + + return res; + }; + + BN.prototype.gcd = function gcd (num) { + if (this.isZero()) return num.abs(); + if (num.isZero()) return this.abs(); + + var a = this.clone(); + var b = num.clone(); + a.negative = 0; + b.negative = 0; + + // Remove common factor of two + for (var shift = 0; a.isEven() && b.isEven(); shift++) { + a.iushrn(1); + b.iushrn(1); + } + + do { + while (a.isEven()) { + a.iushrn(1); + } + while (b.isEven()) { + b.iushrn(1); + } + + var r = a.cmp(b); + if (r < 0) { + // Swap `a` and `b` to make `a` always bigger than `b` + var t = a; + a = b; + b = t; + } else if (r === 0 || b.cmpn(1) === 0) { + break; + } + + a.isub(b); + } while (true); + + return b.iushln(shift); + }; + + // Invert number in the field F(num) + BN.prototype.invm = function invm (num) { + return this.egcd(num).a.umod(num); + }; + + BN.prototype.isEven = function isEven () { + return (this.words[0] & 1) === 0; + }; + + BN.prototype.isOdd = function isOdd () { + return (this.words[0] & 1) === 1; + }; + + // And first word and num + BN.prototype.andln = function andln (num) { + return this.words[0] & num; + }; + + // Increment at the bit position in-line + BN.prototype.bincn = function bincn (bit) { + assert(typeof bit === 'number'); + var r = bit % 26; + var s = (bit - r) / 26; + var q = 1 << r; + + // Fast case: bit is much higher than all existing words + if (this.length <= s) { + this._expand(s + 1); + this.words[s] |= q; + return this; + } + + // Add bit and propagate, if needed + var carry = q; + for (var i = s; carry !== 0 && i < this.length; i++) { + var w = this.words[i] | 0; + w += carry; + carry = w >>> 26; + w &= 0x3ffffff; + this.words[i] = w; + } + if (carry !== 0) { + this.words[i] = carry; + this.length++; + } + return this; + }; + + BN.prototype.isZero = function isZero () { + return this.length === 1 && this.words[0] === 0; + }; + + BN.prototype.cmpn = function cmpn (num) { + var negative = num < 0; + + if (this.negative !== 0 && !negative) return -1; + if (this.negative === 0 && negative) return 1; + + this.strip(); + + var res; + if (this.length > 1) { + res = 1; + } else { + if (negative) { + num = -num; + } + + assert(num <= 0x3ffffff, 'Number is too big'); + + var w = this.words[0] | 0; + res = w === num ? 0 : w < num ? -1 : 1; + } + if (this.negative !== 0) return -res | 0; + return res; + }; + + // Compare two numbers and return: + // 1 - if `this` > `num` + // 0 - if `this` == `num` + // -1 - if `this` < `num` + BN.prototype.cmp = function cmp (num) { + if (this.negative !== 0 && num.negative === 0) return -1; + if (this.negative === 0 && num.negative !== 0) return 1; + + var res = this.ucmp(num); + if (this.negative !== 0) return -res | 0; + return res; + }; + + // Unsigned comparison + BN.prototype.ucmp = function ucmp (num) { + // At this point both numbers have the same sign + if (this.length > num.length) return 1; + if (this.length < num.length) return -1; + + var res = 0; + for (var i = this.length - 1; i >= 0; i--) { + var a = this.words[i] | 0; + var b = num.words[i] | 0; + + if (a === b) continue; + if (a < b) { + res = -1; + } else if (a > b) { + res = 1; + } + break; + } + return res; + }; + + BN.prototype.gtn = function gtn (num) { + return this.cmpn(num) === 1; + }; + + BN.prototype.gt = function gt (num) { + return this.cmp(num) === 1; + }; + + BN.prototype.gten = function gten (num) { + return this.cmpn(num) >= 0; + }; + + BN.prototype.gte = function gte (num) { + return this.cmp(num) >= 0; + }; + + BN.prototype.ltn = function ltn (num) { + return this.cmpn(num) === -1; + }; + + BN.prototype.lt = function lt (num) { + return this.cmp(num) === -1; + }; + + BN.prototype.lten = function lten (num) { + return this.cmpn(num) <= 0; + }; + + BN.prototype.lte = function lte (num) { + return this.cmp(num) <= 0; + }; + + BN.prototype.eqn = function eqn (num) { + return this.cmpn(num) === 0; + }; + + BN.prototype.eq = function eq (num) { + return this.cmp(num) === 0; + }; + + // + // A reduce context, could be using montgomery or something better, depending + // on the `m` itself. + // + BN.red = function red (num) { + return new Red(num); + }; + + BN.prototype.toRed = function toRed (ctx) { + assert(!this.red, 'Already a number in reduction context'); + assert(this.negative === 0, 'red works only with positives'); + return ctx.convertTo(this)._forceRed(ctx); + }; + + BN.prototype.fromRed = function fromRed () { + assert(this.red, 'fromRed works only with numbers in reduction context'); + return this.red.convertFrom(this); + }; + + BN.prototype._forceRed = function _forceRed (ctx) { + this.red = ctx; + return this; + }; + + BN.prototype.forceRed = function forceRed (ctx) { + assert(!this.red, 'Already a number in reduction context'); + return this._forceRed(ctx); + }; + + BN.prototype.redAdd = function redAdd (num) { + assert(this.red, 'redAdd works only with red numbers'); + return this.red.add(this, num); + }; + + BN.prototype.redIAdd = function redIAdd (num) { + assert(this.red, 'redIAdd works only with red numbers'); + return this.red.iadd(this, num); + }; + + BN.prototype.redSub = function redSub (num) { + assert(this.red, 'redSub works only with red numbers'); + return this.red.sub(this, num); + }; + + BN.prototype.redISub = function redISub (num) { + assert(this.red, 'redISub works only with red numbers'); + return this.red.isub(this, num); + }; + + BN.prototype.redShl = function redShl (num) { + assert(this.red, 'redShl works only with red numbers'); + return this.red.shl(this, num); + }; + + BN.prototype.redMul = function redMul (num) { + assert(this.red, 'redMul works only with red numbers'); + this.red._verify2(this, num); + return this.red.mul(this, num); + }; + + BN.prototype.redIMul = function redIMul (num) { + assert(this.red, 'redMul works only with red numbers'); + this.red._verify2(this, num); + return this.red.imul(this, num); + }; + + BN.prototype.redSqr = function redSqr () { + assert(this.red, 'redSqr works only with red numbers'); + this.red._verify1(this); + return this.red.sqr(this); + }; + + BN.prototype.redISqr = function redISqr () { + assert(this.red, 'redISqr works only with red numbers'); + this.red._verify1(this); + return this.red.isqr(this); + }; + + // Square root over p + BN.prototype.redSqrt = function redSqrt () { + assert(this.red, 'redSqrt works only with red numbers'); + this.red._verify1(this); + return this.red.sqrt(this); + }; + + BN.prototype.redInvm = function redInvm () { + assert(this.red, 'redInvm works only with red numbers'); + this.red._verify1(this); + return this.red.invm(this); + }; + + // Return negative clone of `this` % `red modulo` + BN.prototype.redNeg = function redNeg () { + assert(this.red, 'redNeg works only with red numbers'); + this.red._verify1(this); + return this.red.neg(this); + }; + + BN.prototype.redPow = function redPow (num) { + assert(this.red && !num.red, 'redPow(normalNum)'); + this.red._verify1(this); + return this.red.pow(this, num); + }; + + // Prime numbers with efficient reduction + var primes = { + k256: null, + p224: null, + p192: null, + p25519: null + }; + + // Pseudo-Mersenne prime + function MPrime (name, p) { + // P = 2 ^ N - K + this.name = name; + this.p = new BN(p, 16); + this.n = this.p.bitLength(); + this.k = new BN(1).iushln(this.n).isub(this.p); + + this.tmp = this._tmp(); + } + + MPrime.prototype._tmp = function _tmp () { + var tmp = new BN(null); + tmp.words = new Array(Math.ceil(this.n / 13)); + return tmp; + }; + + MPrime.prototype.ireduce = function ireduce (num) { + // Assumes that `num` is less than `P^2` + // num = HI * (2 ^ N - K) + HI * K + LO = HI * K + LO (mod P) + var r = num; + var rlen; + + do { + this.split(r, this.tmp); + r = this.imulK(r); + r = r.iadd(this.tmp); + rlen = r.bitLength(); + } while (rlen > this.n); + + var cmp = rlen < this.n ? -1 : r.ucmp(this.p); + if (cmp === 0) { + r.words[0] = 0; + r.length = 1; + } else if (cmp > 0) { + r.isub(this.p); + } else { + if (r.strip !== undefined) { + // r is BN v4 instance + r.strip(); + } else { + // r is BN v5 instance + r._strip(); + } + } + + return r; + }; + + MPrime.prototype.split = function split (input, out) { + input.iushrn(this.n, 0, out); + }; + + MPrime.prototype.imulK = function imulK (num) { + return num.imul(this.k); + }; + + function K256 () { + MPrime.call( + this, + 'k256', + 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff fffffffe fffffc2f'); + } + inherits(K256, MPrime); + + K256.prototype.split = function split (input, output) { + // 256 = 9 * 26 + 22 + var mask = 0x3fffff; + + var outLen = Math.min(input.length, 9); + for (var i = 0; i < outLen; i++) { + output.words[i] = input.words[i]; + } + output.length = outLen; + + if (input.length <= 9) { + input.words[0] = 0; + input.length = 1; + return; + } + + // Shift by 9 limbs + var prev = input.words[9]; + output.words[output.length++] = prev & mask; + + for (i = 10; i < input.length; i++) { + var next = input.words[i] | 0; + input.words[i - 10] = ((next & mask) << 4) | (prev >>> 22); + prev = next; + } + prev >>>= 22; + input.words[i - 10] = prev; + if (prev === 0 && input.length > 10) { + input.length -= 10; + } else { + input.length -= 9; + } + }; + + K256.prototype.imulK = function imulK (num) { + // K = 0x1000003d1 = [ 0x40, 0x3d1 ] + num.words[num.length] = 0; + num.words[num.length + 1] = 0; + num.length += 2; + + // bounded at: 0x40 * 0x3ffffff + 0x3d0 = 0x100000390 + var lo = 0; + for (var i = 0; i < num.length; i++) { + var w = num.words[i] | 0; + lo += w * 0x3d1; + num.words[i] = lo & 0x3ffffff; + lo = w * 0x40 + ((lo / 0x4000000) | 0); + } + + // Fast length reduction + if (num.words[num.length - 1] === 0) { + num.length--; + if (num.words[num.length - 1] === 0) { + num.length--; + } + } + return num; + }; + + function P224 () { + MPrime.call( + this, + 'p224', + 'ffffffff ffffffff ffffffff ffffffff 00000000 00000000 00000001'); + } + inherits(P224, MPrime); + + function P192 () { + MPrime.call( + this, + 'p192', + 'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff'); + } + inherits(P192, MPrime); + + function P25519 () { + // 2 ^ 255 - 19 + MPrime.call( + this, + '25519', + '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed'); + } + inherits(P25519, MPrime); + + P25519.prototype.imulK = function imulK (num) { + // K = 0x13 + var carry = 0; + for (var i = 0; i < num.length; i++) { + var hi = (num.words[i] | 0) * 0x13 + carry; + var lo = hi & 0x3ffffff; + hi >>>= 26; + + num.words[i] = lo; + carry = hi; + } + if (carry !== 0) { + num.words[num.length++] = carry; + } + return num; + }; + + // Exported mostly for testing purposes, use plain name instead + BN._prime = function prime (name) { + // Cached version of prime + if (primes[name]) return primes[name]; + + var prime; + if (name === 'k256') { + prime = new K256(); + } else if (name === 'p224') { + prime = new P224(); + } else if (name === 'p192') { + prime = new P192(); + } else if (name === 'p25519') { + prime = new P25519(); + } else { + throw new Error('Unknown prime ' + name); + } + primes[name] = prime; + + return prime; + }; + + // + // Base reduction engine + // + function Red (m) { + if (typeof m === 'string') { + var prime = BN._prime(m); + this.m = prime.p; + this.prime = prime; + } else { + assert(m.gtn(1), 'modulus must be greater than 1'); + this.m = m; + this.prime = null; + } + } + + Red.prototype._verify1 = function _verify1 (a) { + assert(a.negative === 0, 'red works only with positives'); + assert(a.red, 'red works only with red numbers'); + }; + + Red.prototype._verify2 = function _verify2 (a, b) { + assert((a.negative | b.negative) === 0, 'red works only with positives'); + assert(a.red && a.red === b.red, + 'red works only with red numbers'); + }; + + Red.prototype.imod = function imod (a) { + if (this.prime) return this.prime.ireduce(a)._forceRed(this); + return a.umod(this.m)._forceRed(this); + }; + + Red.prototype.neg = function neg (a) { + if (a.isZero()) { + return a.clone(); + } + + return this.m.sub(a)._forceRed(this); + }; + + Red.prototype.add = function add (a, b) { + this._verify2(a, b); + + var res = a.add(b); + if (res.cmp(this.m) >= 0) { + res.isub(this.m); + } + return res._forceRed(this); + }; + + Red.prototype.iadd = function iadd (a, b) { + this._verify2(a, b); + + var res = a.iadd(b); + if (res.cmp(this.m) >= 0) { + res.isub(this.m); + } + return res; + }; + + Red.prototype.sub = function sub (a, b) { + this._verify2(a, b); + + var res = a.sub(b); + if (res.cmpn(0) < 0) { + res.iadd(this.m); + } + return res._forceRed(this); + }; + + Red.prototype.isub = function isub (a, b) { + this._verify2(a, b); + + var res = a.isub(b); + if (res.cmpn(0) < 0) { + res.iadd(this.m); + } + return res; + }; + + Red.prototype.shl = function shl (a, num) { + this._verify1(a); + return this.imod(a.ushln(num)); + }; + + Red.prototype.imul = function imul (a, b) { + this._verify2(a, b); + return this.imod(a.imul(b)); + }; + + Red.prototype.mul = function mul (a, b) { + this._verify2(a, b); + return this.imod(a.mul(b)); + }; + + Red.prototype.isqr = function isqr (a) { + return this.imul(a, a.clone()); + }; + + Red.prototype.sqr = function sqr (a) { + return this.mul(a, a); + }; + + Red.prototype.sqrt = function sqrt (a) { + if (a.isZero()) return a.clone(); + + var mod3 = this.m.andln(3); + assert(mod3 % 2 === 1); + + // Fast case + if (mod3 === 3) { + var pow = this.m.add(new BN(1)).iushrn(2); + return this.pow(a, pow); + } + + // Tonelli-Shanks algorithm (Totally unoptimized and slow) + // + // Find Q and S, that Q * 2 ^ S = (P - 1) + var q = this.m.subn(1); + var s = 0; + while (!q.isZero() && q.andln(1) === 0) { + s++; + q.iushrn(1); + } + assert(!q.isZero()); + + var one = new BN(1).toRed(this); + var nOne = one.redNeg(); + + // Find quadratic non-residue + // NOTE: Max is such because of generalized Riemann hypothesis. + var lpow = this.m.subn(1).iushrn(1); + var z = this.m.bitLength(); + z = new BN(2 * z * z).toRed(this); + + while (this.pow(z, lpow).cmp(nOne) !== 0) { + z.redIAdd(nOne); + } + + var c = this.pow(z, q); + var r = this.pow(a, q.addn(1).iushrn(1)); + var t = this.pow(a, q); + var m = s; + while (t.cmp(one) !== 0) { + var tmp = t; + for (var i = 0; tmp.cmp(one) !== 0; i++) { + tmp = tmp.redSqr(); + } + assert(i < m); + var b = this.pow(c, new BN(1).iushln(m - i - 1)); + + r = r.redMul(b); + c = b.redSqr(); + t = t.redMul(c); + m = i; + } + + return r; + }; + + Red.prototype.invm = function invm (a) { + var inv = a._invmp(this.m); + if (inv.negative !== 0) { + inv.negative = 0; + return this.imod(inv).redNeg(); + } else { + return this.imod(inv); + } + }; + + Red.prototype.pow = function pow (a, num) { + if (num.isZero()) return new BN(1).toRed(this); + if (num.cmpn(1) === 0) return a.clone(); + + var windowSize = 4; + var wnd = new Array(1 << windowSize); + wnd[0] = new BN(1).toRed(this); + wnd[1] = a; + for (var i = 2; i < wnd.length; i++) { + wnd[i] = this.mul(wnd[i - 1], a); + } + + var res = wnd[0]; + var current = 0; + var currentLen = 0; + var start = num.bitLength() % 26; + if (start === 0) { + start = 26; + } + + for (i = num.length - 1; i >= 0; i--) { + var word = num.words[i]; + for (var j = start - 1; j >= 0; j--) { + var bit = (word >> j) & 1; + if (res !== wnd[0]) { + res = this.sqr(res); + } + + if (bit === 0 && current === 0) { + currentLen = 0; + continue; + } + + current <<= 1; + current |= bit; + currentLen++; + if (currentLen !== windowSize && (i !== 0 || j !== 0)) continue; + + res = this.mul(res, wnd[current]); + currentLen = 0; + current = 0; + } + start = 26; + } + + return res; + }; + + Red.prototype.convertTo = function convertTo (num) { + var r = num.umod(this.m); + + return r === num ? r.clone() : r; + }; + + Red.prototype.convertFrom = function convertFrom (num) { + var res = num.clone(); + res.red = null; + return res; + }; + + // + // Montgomery method engine + // + + BN.mont = function mont (num) { + return new Mont(num); + }; + + function Mont (m) { + Red.call(this, m); + + this.shift = this.m.bitLength(); + if (this.shift % 26 !== 0) { + this.shift += 26 - (this.shift % 26); + } + + this.r = new BN(1).iushln(this.shift); + this.r2 = this.imod(this.r.sqr()); + this.rinv = this.r._invmp(this.m); + + this.minv = this.rinv.mul(this.r).isubn(1).div(this.m); + this.minv = this.minv.umod(this.r); + this.minv = this.r.sub(this.minv); + } + inherits(Mont, Red); + + Mont.prototype.convertTo = function convertTo (num) { + return this.imod(num.ushln(this.shift)); + }; + + Mont.prototype.convertFrom = function convertFrom (num) { + var r = this.imod(num.mul(this.rinv)); + r.red = null; + return r; + }; + + Mont.prototype.imul = function imul (a, b) { + if (a.isZero() || b.isZero()) { + a.words[0] = 0; + a.length = 1; + return a; + } + + var t = a.imul(b); + var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m); + var u = t.isub(c).iushrn(this.shift); + var res = u; + + if (u.cmp(this.m) >= 0) { + res = u.isub(this.m); + } else if (u.cmpn(0) < 0) { + res = u.iadd(this.m); + } + + return res._forceRed(this); + }; + + Mont.prototype.mul = function mul (a, b) { + if (a.isZero() || b.isZero()) return new BN(0)._forceRed(this); + + var t = a.mul(b); + var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m); + var u = t.isub(c).iushrn(this.shift); + var res = u; + if (u.cmp(this.m) >= 0) { + res = u.isub(this.m); + } else if (u.cmpn(0) < 0) { + res = u.iadd(this.m); + } + + return res._forceRed(this); + }; + + Mont.prototype.invm = function invm (a) { + // (AR)^-1 * R^2 = (A^-1 * R^-1) * R^2 = A^-1 * R + var res = this.imod(a._invmp(this.m).mul(this.r2)); + return res._forceRed(this); + }; +})(typeof module === 'undefined' || module, this); diff --git a/node_modules/bn.js/package.json b/node_modules/bn.js/package.json new file mode 100644 index 0000000..098b805 --- /dev/null +++ b/node_modules/bn.js/package.json @@ -0,0 +1,36 @@ +{ + "name": "bn.js", + "version": "4.12.0", + "description": "Big number implementation in pure javascript", + "main": "lib/bn.js", + "scripts": { + "lint": "semistandard", + "unit": "mocha --reporter=spec test/*-test.js", + "test": "npm run lint && npm run unit" + }, + "repository": { + "type": "git", + "url": "git@github.com:indutny/bn.js" + }, + "keywords": [ + "BN", + "BigNum", + "Big number", + "Modulo", + "Montgomery" + ], + "author": "Fedor Indutny ", + "license": "MIT", + "bugs": { + "url": "https://github.com/indutny/bn.js/issues" + }, + "homepage": "https://github.com/indutny/bn.js", + "browser": { + "buffer": false + }, + "devDependencies": { + "istanbul": "^0.3.5", + "mocha": "^2.1.0", + "semistandard": "^7.0.4" + } +} diff --git a/node_modules/brorand/.npmignore b/node_modules/brorand/.npmignore new file mode 100644 index 0000000..1ca9571 --- /dev/null +++ b/node_modules/brorand/.npmignore @@ -0,0 +1,2 @@ +node_modules/ +npm-debug.log diff --git a/node_modules/brorand/README.md b/node_modules/brorand/README.md new file mode 100644 index 0000000..f80437d --- /dev/null +++ b/node_modules/brorand/README.md @@ -0,0 +1,26 @@ +# Brorand + +#### LICENSE + +This software is licensed under the MIT License. + +Copyright Fedor Indutny, 2014. + +Permission is hereby granted, free of charge, to any person obtaining a +copy of this software and associated documentation files (the +"Software"), to deal in the Software without restriction, including +without limitation the rights to use, copy, modify, merge, publish, +distribute, sublicense, and/or sell copies of the Software, and to permit +persons to whom the Software is furnished to do so, subject to the +following conditions: + +The above copyright notice and this permission notice shall be included +in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN +NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, +DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR +OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE +USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/node_modules/brorand/index.js b/node_modules/brorand/index.js new file mode 100644 index 0000000..9a0fff4 --- /dev/null +++ b/node_modules/brorand/index.js @@ -0,0 +1,65 @@ +var r; + +module.exports = function rand(len) { + if (!r) + r = new Rand(null); + + return r.generate(len); +}; + +function Rand(rand) { + this.rand = rand; +} +module.exports.Rand = Rand; + +Rand.prototype.generate = function generate(len) { + return this._rand(len); +}; + +// Emulate crypto API using randy +Rand.prototype._rand = function _rand(n) { + if (this.rand.getBytes) + return this.rand.getBytes(n); + + var res = new Uint8Array(n); + for (var i = 0; i < res.length; i++) + res[i] = this.rand.getByte(); + return res; +}; + +if (typeof self === 'object') { + if (self.crypto && self.crypto.getRandomValues) { + // Modern browsers + Rand.prototype._rand = function _rand(n) { + var arr = new Uint8Array(n); + self.crypto.getRandomValues(arr); + return arr; + }; + } else if (self.msCrypto && self.msCrypto.getRandomValues) { + // IE + Rand.prototype._rand = function _rand(n) { + var arr = new Uint8Array(n); + self.msCrypto.getRandomValues(arr); + return arr; + }; + + // Safari's WebWorkers do not have `crypto` + } else if (typeof window === 'object') { + // Old junk + Rand.prototype._rand = function() { + throw new Error('Not implemented yet'); + }; + } +} else { + // Node.js or Web worker with no crypto support + try { + var crypto = require('crypto'); + if (typeof crypto.randomBytes !== 'function') + throw new Error('Not supported'); + + Rand.prototype._rand = function _rand(n) { + return crypto.randomBytes(n); + }; + } catch (e) { + } +} diff --git a/node_modules/brorand/package.json b/node_modules/brorand/package.json new file mode 100644 index 0000000..f213c9d --- /dev/null +++ b/node_modules/brorand/package.json @@ -0,0 +1,31 @@ +{ + "name": "brorand", + "version": "1.1.0", + "description": "Random number generator for browsers and node.js", + "main": "index.js", + "scripts": { + "test": "mocha --reporter=spec test/**/*-test.js" + }, + "repository": { + "type": "git", + "url": "git@github.com:indutny/brorand" + }, + "keywords": [ + "Random", + "RNG", + "browser", + "crypto" + ], + "author": "Fedor Indutny ", + "license": "MIT", + "bugs": { + "url": "https://github.com/indutny/brorand/issues" + }, + "homepage": "https://github.com/indutny/brorand", + "devDependencies": { + "mocha": "^2.0.1" + }, + "browser": { + "crypto": false + } +} diff --git a/node_modules/brorand/test/api-test.js b/node_modules/brorand/test/api-test.js new file mode 100644 index 0000000..b6c876d --- /dev/null +++ b/node_modules/brorand/test/api-test.js @@ -0,0 +1,8 @@ +var brorand = require('../'); +var assert = require('assert'); + +describe('Brorand', function() { + it('should generate random numbers', function() { + assert.equal(brorand(100).length, 100); + }); +}); diff --git a/node_modules/elliptic/README.md b/node_modules/elliptic/README.md new file mode 100644 index 0000000..96219e5 --- /dev/null +++ b/node_modules/elliptic/README.md @@ -0,0 +1,238 @@ +# Elliptic [![Build Status](https://secure.travis-ci.org/indutny/elliptic.png)](http://travis-ci.org/indutny/elliptic) [![Coverage Status](https://coveralls.io/repos/indutny/elliptic/badge.svg?branch=master&service=github)](https://coveralls.io/github/indutny/elliptic?branch=master) [![Code Climate](https://codeclimate.com/github/indutny/elliptic/badges/gpa.svg)](https://codeclimate.com/github/indutny/elliptic) + +[![Saucelabs Test Status](https://saucelabs.com/browser-matrix/gh-indutny-elliptic.svg)](https://saucelabs.com/u/gh-indutny-elliptic) + +Fast elliptic-curve cryptography in a plain javascript implementation. + +NOTE: Please take a look at http://safecurves.cr.yp.to/ before choosing a curve +for your cryptography operations. + +## Incentive + +ECC is much slower than regular RSA cryptography, the JS implementations are +even more slower. + +## Benchmarks + +```bash +$ node benchmarks/index.js +Benchmarking: sign +elliptic#sign x 262 ops/sec ±0.51% (177 runs sampled) +eccjs#sign x 55.91 ops/sec ±0.90% (144 runs sampled) +------------------------ +Fastest is elliptic#sign +======================== +Benchmarking: verify +elliptic#verify x 113 ops/sec ±0.50% (166 runs sampled) +eccjs#verify x 48.56 ops/sec ±0.36% (125 runs sampled) +------------------------ +Fastest is elliptic#verify +======================== +Benchmarking: gen +elliptic#gen x 294 ops/sec ±0.43% (176 runs sampled) +eccjs#gen x 62.25 ops/sec ±0.63% (129 runs sampled) +------------------------ +Fastest is elliptic#gen +======================== +Benchmarking: ecdh +elliptic#ecdh x 136 ops/sec ±0.85% (156 runs sampled) +------------------------ +Fastest is elliptic#ecdh +======================== +``` + +## API + +### ECDSA + +```javascript +var EC = require('elliptic').ec; + +// Create and initialize EC context +// (better do it once and reuse it) +var ec = new EC('secp256k1'); + +// Generate keys +var key = ec.genKeyPair(); + +// Sign the message's hash (input must be an array, or a hex-string) +var msgHash = [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 ]; +var signature = key.sign(msgHash); + +// Export DER encoded signature in Array +var derSign = signature.toDER(); + +// Verify signature +console.log(key.verify(msgHash, derSign)); + +// CHECK WITH NO PRIVATE KEY + +var pubPoint = key.getPublic(); +var x = pubPoint.getX(); +var y = pubPoint.getY(); + +// Public Key MUST be either: +// 1) '04' + hex string of x + hex string of y; or +// 2) object with two hex string properties (x and y); or +// 3) object with two buffer properties (x and y) +var pub = pubPoint.encode('hex'); // case 1 +var pub = { x: x.toString('hex'), y: y.toString('hex') }; // case 2 +var pub = { x: x.toBuffer(), y: y.toBuffer() }; // case 3 +var pub = { x: x.toArrayLike(Buffer), y: y.toArrayLike(Buffer) }; // case 3 + +// Import public key +var key = ec.keyFromPublic(pub, 'hex'); + +// Signature MUST be either: +// 1) DER-encoded signature as hex-string; or +// 2) DER-encoded signature as buffer; or +// 3) object with two hex-string properties (r and s); or +// 4) object with two buffer properties (r and s) + +var signature = '3046022100...'; // case 1 +var signature = new Buffer('...'); // case 2 +var signature = { r: 'b1fc...', s: '9c42...' }; // case 3 + +// Verify signature +console.log(key.verify(msgHash, signature)); +``` + +### EdDSA + +```javascript +var EdDSA = require('elliptic').eddsa; + +// Create and initialize EdDSA context +// (better do it once and reuse it) +var ec = new EdDSA('ed25519'); + +// Create key pair from secret +var key = ec.keyFromSecret('693e3c...'); // hex string, array or Buffer + +// Sign the message's hash (input must be an array, or a hex-string) +var msgHash = [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 ]; +var signature = key.sign(msgHash).toHex(); + +// Verify signature +console.log(key.verify(msgHash, signature)); + +// CHECK WITH NO PRIVATE KEY + +// Import public key +var pub = '0a1af638...'; +var key = ec.keyFromPublic(pub, 'hex'); + +// Verify signature +var signature = '70bed1...'; +console.log(key.verify(msgHash, signature)); +``` + +### ECDH + +```javascript +var EC = require('elliptic').ec; +var ec = new EC('curve25519'); + +// Generate keys +var key1 = ec.genKeyPair(); +var key2 = ec.genKeyPair(); + +var shared1 = key1.derive(key2.getPublic()); +var shared2 = key2.derive(key1.getPublic()); + +console.log('Both shared secrets are BN instances'); +console.log(shared1.toString(16)); +console.log(shared2.toString(16)); +``` + +three and more members: +```javascript +var EC = require('elliptic').ec; +var ec = new EC('curve25519'); + +var A = ec.genKeyPair(); +var B = ec.genKeyPair(); +var C = ec.genKeyPair(); + +var AB = A.getPublic().mul(B.getPrivate()) +var BC = B.getPublic().mul(C.getPrivate()) +var CA = C.getPublic().mul(A.getPrivate()) + +var ABC = AB.mul(C.getPrivate()) +var BCA = BC.mul(A.getPrivate()) +var CAB = CA.mul(B.getPrivate()) + +console.log(ABC.getX().toString(16)) +console.log(BCA.getX().toString(16)) +console.log(CAB.getX().toString(16)) +``` + +NOTE: `.derive()` returns a [BN][1] instance. + +## Supported curves + +Elliptic.js support following curve types: + +* Short Weierstrass +* Montgomery +* Edwards +* Twisted Edwards + +Following curve 'presets' are embedded into the library: + +* `secp256k1` +* `p192` +* `p224` +* `p256` +* `p384` +* `p521` +* `curve25519` +* `ed25519` + +NOTE: That `curve25519` could not be used for ECDSA, use `ed25519` instead. + +### Implementation details + +ECDSA is using deterministic `k` value generation as per [RFC6979][0]. Most of +the curve operations are performed on non-affine coordinates (either projective +or extended), various windowing techniques are used for different cases. + +All operations are performed in reduction context using [bn.js][1], hashing is +provided by [hash.js][2] + +### Related projects + +* [eccrypto][3]: isomorphic implementation of ECDSA, ECDH and ECIES for both + browserify and node (uses `elliptic` for browser and [secp256k1-node][4] for + node) + +#### LICENSE + +This software is licensed under the MIT License. + +Copyright Fedor Indutny, 2014. + +Permission is hereby granted, free of charge, to any person obtaining a +copy of this software and associated documentation files (the +"Software"), to deal in the Software without restriction, including +without limitation the rights to use, copy, modify, merge, publish, +distribute, sublicense, and/or sell copies of the Software, and to permit +persons to whom the Software is furnished to do so, subject to the +following conditions: + +The above copyright notice and this permission notice shall be included +in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN +NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, +DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR +OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE +USE OR OTHER DEALINGS IN THE SOFTWARE. + +[0]: http://tools.ietf.org/html/rfc6979 +[1]: https://github.com/indutny/bn.js +[2]: https://github.com/indutny/hash.js +[3]: https://github.com/bitchan/eccrypto +[4]: https://github.com/wanderer/secp256k1-node diff --git a/node_modules/elliptic/lib/elliptic.js b/node_modules/elliptic/lib/elliptic.js new file mode 100644 index 0000000..dfe2fe7 --- /dev/null +++ b/node_modules/elliptic/lib/elliptic.js @@ -0,0 +1,13 @@ +'use strict'; + +var elliptic = exports; + +elliptic.version = require('../package.json').version; +elliptic.utils = require('./elliptic/utils'); +elliptic.rand = require('brorand'); +elliptic.curve = require('./elliptic/curve'); +elliptic.curves = require('./elliptic/curves'); + +// Protocols +elliptic.ec = require('./elliptic/ec'); +elliptic.eddsa = require('./elliptic/eddsa'); diff --git a/node_modules/elliptic/lib/elliptic/curve/base.js b/node_modules/elliptic/lib/elliptic/curve/base.js new file mode 100644 index 0000000..8543fa8 --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/curve/base.js @@ -0,0 +1,381 @@ +'use strict'; + +var BN = require('bn.js'); +var utils = require('../utils'); +var getNAF = utils.getNAF; +var getJSF = utils.getJSF; +var assert = utils.assert; + +function BaseCurve(type, conf) { + this.type = type; + this.p = new BN(conf.p, 16); + + // Use Montgomery, when there is no fast reduction for the prime + this.red = conf.prime ? BN.red(conf.prime) : BN.mont(this.p); + + // Useful for many curves + this.zero = new BN(0).toRed(this.red); + this.one = new BN(1).toRed(this.red); + this.two = new BN(2).toRed(this.red); + + // Curve configuration, optional + this.n = conf.n && new BN(conf.n, 16); + this.g = conf.g && this.pointFromJSON(conf.g, conf.gRed); + + // Temporary arrays + this._wnafT1 = new Array(4); + this._wnafT2 = new Array(4); + this._wnafT3 = new Array(4); + this._wnafT4 = new Array(4); + + this._bitLength = this.n ? this.n.bitLength() : 0; + + // Generalized Greg Maxwell's trick + var adjustCount = this.n && this.p.div(this.n); + if (!adjustCount || adjustCount.cmpn(100) > 0) { + this.redN = null; + } else { + this._maxwellTrick = true; + this.redN = this.n.toRed(this.red); + } +} +module.exports = BaseCurve; + +BaseCurve.prototype.point = function point() { + throw new Error('Not implemented'); +}; + +BaseCurve.prototype.validate = function validate() { + throw new Error('Not implemented'); +}; + +BaseCurve.prototype._fixedNafMul = function _fixedNafMul(p, k) { + assert(p.precomputed); + var doubles = p._getDoubles(); + + var naf = getNAF(k, 1, this._bitLength); + var I = (1 << (doubles.step + 1)) - (doubles.step % 2 === 0 ? 2 : 1); + I /= 3; + + // Translate into more windowed form + var repr = []; + var j; + var nafW; + for (j = 0; j < naf.length; j += doubles.step) { + nafW = 0; + for (var l = j + doubles.step - 1; l >= j; l--) + nafW = (nafW << 1) + naf[l]; + repr.push(nafW); + } + + var a = this.jpoint(null, null, null); + var b = this.jpoint(null, null, null); + for (var i = I; i > 0; i--) { + for (j = 0; j < repr.length; j++) { + nafW = repr[j]; + if (nafW === i) + b = b.mixedAdd(doubles.points[j]); + else if (nafW === -i) + b = b.mixedAdd(doubles.points[j].neg()); + } + a = a.add(b); + } + return a.toP(); +}; + +BaseCurve.prototype._wnafMul = function _wnafMul(p, k) { + var w = 4; + + // Precompute window + var nafPoints = p._getNAFPoints(w); + w = nafPoints.wnd; + var wnd = nafPoints.points; + + // Get NAF form + var naf = getNAF(k, w, this._bitLength); + + // Add `this`*(N+1) for every w-NAF index + var acc = this.jpoint(null, null, null); + for (var i = naf.length - 1; i >= 0; i--) { + // Count zeroes + for (var l = 0; i >= 0 && naf[i] === 0; i--) + l++; + if (i >= 0) + l++; + acc = acc.dblp(l); + + if (i < 0) + break; + var z = naf[i]; + assert(z !== 0); + if (p.type === 'affine') { + // J +- P + if (z > 0) + acc = acc.mixedAdd(wnd[(z - 1) >> 1]); + else + acc = acc.mixedAdd(wnd[(-z - 1) >> 1].neg()); + } else { + // J +- J + if (z > 0) + acc = acc.add(wnd[(z - 1) >> 1]); + else + acc = acc.add(wnd[(-z - 1) >> 1].neg()); + } + } + return p.type === 'affine' ? acc.toP() : acc; +}; + +BaseCurve.prototype._wnafMulAdd = function _wnafMulAdd(defW, + points, + coeffs, + len, + jacobianResult) { + var wndWidth = this._wnafT1; + var wnd = this._wnafT2; + var naf = this._wnafT3; + + // Fill all arrays + var max = 0; + var i; + var j; + var p; + for (i = 0; i < len; i++) { + p = points[i]; + var nafPoints = p._getNAFPoints(defW); + wndWidth[i] = nafPoints.wnd; + wnd[i] = nafPoints.points; + } + + // Comb small window NAFs + for (i = len - 1; i >= 1; i -= 2) { + var a = i - 1; + var b = i; + if (wndWidth[a] !== 1 || wndWidth[b] !== 1) { + naf[a] = getNAF(coeffs[a], wndWidth[a], this._bitLength); + naf[b] = getNAF(coeffs[b], wndWidth[b], this._bitLength); + max = Math.max(naf[a].length, max); + max = Math.max(naf[b].length, max); + continue; + } + + var comb = [ + points[a], /* 1 */ + null, /* 3 */ + null, /* 5 */ + points[b], /* 7 */ + ]; + + // Try to avoid Projective points, if possible + if (points[a].y.cmp(points[b].y) === 0) { + comb[1] = points[a].add(points[b]); + comb[2] = points[a].toJ().mixedAdd(points[b].neg()); + } else if (points[a].y.cmp(points[b].y.redNeg()) === 0) { + comb[1] = points[a].toJ().mixedAdd(points[b]); + comb[2] = points[a].add(points[b].neg()); + } else { + comb[1] = points[a].toJ().mixedAdd(points[b]); + comb[2] = points[a].toJ().mixedAdd(points[b].neg()); + } + + var index = [ + -3, /* -1 -1 */ + -1, /* -1 0 */ + -5, /* -1 1 */ + -7, /* 0 -1 */ + 0, /* 0 0 */ + 7, /* 0 1 */ + 5, /* 1 -1 */ + 1, /* 1 0 */ + 3, /* 1 1 */ + ]; + + var jsf = getJSF(coeffs[a], coeffs[b]); + max = Math.max(jsf[0].length, max); + naf[a] = new Array(max); + naf[b] = new Array(max); + for (j = 0; j < max; j++) { + var ja = jsf[0][j] | 0; + var jb = jsf[1][j] | 0; + + naf[a][j] = index[(ja + 1) * 3 + (jb + 1)]; + naf[b][j] = 0; + wnd[a] = comb; + } + } + + var acc = this.jpoint(null, null, null); + var tmp = this._wnafT4; + for (i = max; i >= 0; i--) { + var k = 0; + + while (i >= 0) { + var zero = true; + for (j = 0; j < len; j++) { + tmp[j] = naf[j][i] | 0; + if (tmp[j] !== 0) + zero = false; + } + if (!zero) + break; + k++; + i--; + } + if (i >= 0) + k++; + acc = acc.dblp(k); + if (i < 0) + break; + + for (j = 0; j < len; j++) { + var z = tmp[j]; + p; + if (z === 0) + continue; + else if (z > 0) + p = wnd[j][(z - 1) >> 1]; + else if (z < 0) + p = wnd[j][(-z - 1) >> 1].neg(); + + if (p.type === 'affine') + acc = acc.mixedAdd(p); + else + acc = acc.add(p); + } + } + // Zeroify references + for (i = 0; i < len; i++) + wnd[i] = null; + + if (jacobianResult) + return acc; + else + return acc.toP(); +}; + +function BasePoint(curve, type) { + this.curve = curve; + this.type = type; + this.precomputed = null; +} +BaseCurve.BasePoint = BasePoint; + +BasePoint.prototype.eq = function eq(/*other*/) { + throw new Error('Not implemented'); +}; + +BasePoint.prototype.validate = function validate() { + return this.curve.validate(this); +}; + +BaseCurve.prototype.decodePoint = function decodePoint(bytes, enc) { + bytes = utils.toArray(bytes, enc); + + var len = this.p.byteLength(); + + // uncompressed, hybrid-odd, hybrid-even + if ((bytes[0] === 0x04 || bytes[0] === 0x06 || bytes[0] === 0x07) && + bytes.length - 1 === 2 * len) { + if (bytes[0] === 0x06) + assert(bytes[bytes.length - 1] % 2 === 0); + else if (bytes[0] === 0x07) + assert(bytes[bytes.length - 1] % 2 === 1); + + var res = this.point(bytes.slice(1, 1 + len), + bytes.slice(1 + len, 1 + 2 * len)); + + return res; + } else if ((bytes[0] === 0x02 || bytes[0] === 0x03) && + bytes.length - 1 === len) { + return this.pointFromX(bytes.slice(1, 1 + len), bytes[0] === 0x03); + } + throw new Error('Unknown point format'); +}; + +BasePoint.prototype.encodeCompressed = function encodeCompressed(enc) { + return this.encode(enc, true); +}; + +BasePoint.prototype._encode = function _encode(compact) { + var len = this.curve.p.byteLength(); + var x = this.getX().toArray('be', len); + + if (compact) + return [ this.getY().isEven() ? 0x02 : 0x03 ].concat(x); + + return [ 0x04 ].concat(x, this.getY().toArray('be', len)); +}; + +BasePoint.prototype.encode = function encode(enc, compact) { + return utils.encode(this._encode(compact), enc); +}; + +BasePoint.prototype.precompute = function precompute(power) { + if (this.precomputed) + return this; + + var precomputed = { + doubles: null, + naf: null, + beta: null, + }; + precomputed.naf = this._getNAFPoints(8); + precomputed.doubles = this._getDoubles(4, power); + precomputed.beta = this._getBeta(); + this.precomputed = precomputed; + + return this; +}; + +BasePoint.prototype._hasDoubles = function _hasDoubles(k) { + if (!this.precomputed) + return false; + + var doubles = this.precomputed.doubles; + if (!doubles) + return false; + + return doubles.points.length >= Math.ceil((k.bitLength() + 1) / doubles.step); +}; + +BasePoint.prototype._getDoubles = function _getDoubles(step, power) { + if (this.precomputed && this.precomputed.doubles) + return this.precomputed.doubles; + + var doubles = [ this ]; + var acc = this; + for (var i = 0; i < power; i += step) { + for (var j = 0; j < step; j++) + acc = acc.dbl(); + doubles.push(acc); + } + return { + step: step, + points: doubles, + }; +}; + +BasePoint.prototype._getNAFPoints = function _getNAFPoints(wnd) { + if (this.precomputed && this.precomputed.naf) + return this.precomputed.naf; + + var res = [ this ]; + var max = (1 << wnd) - 1; + var dbl = max === 1 ? null : this.dbl(); + for (var i = 1; i < max; i++) + res[i] = res[i - 1].add(dbl); + return { + wnd: wnd, + points: res, + }; +}; + +BasePoint.prototype._getBeta = function _getBeta() { + return null; +}; + +BasePoint.prototype.dblp = function dblp(k) { + var r = this; + for (var i = 0; i < k; i++) + r = r.dbl(); + return r; +}; diff --git a/node_modules/elliptic/lib/elliptic/curve/edwards.js b/node_modules/elliptic/lib/elliptic/curve/edwards.js new file mode 100644 index 0000000..6e757c6 --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/curve/edwards.js @@ -0,0 +1,435 @@ +'use strict'; + +var utils = require('../utils'); +var BN = require('bn.js'); +var inherits = require('inherits'); +var Base = require('./base'); + +var assert = utils.assert; + +function EdwardsCurve(conf) { + // NOTE: Important as we are creating point in Base.call() + this.twisted = (conf.a | 0) !== 1; + this.mOneA = this.twisted && (conf.a | 0) === -1; + this.extended = this.mOneA; + + Base.call(this, 'edwards', conf); + + this.a = new BN(conf.a, 16).umod(this.red.m); + this.a = this.a.toRed(this.red); + this.c = new BN(conf.c, 16).toRed(this.red); + this.c2 = this.c.redSqr(); + this.d = new BN(conf.d, 16).toRed(this.red); + this.dd = this.d.redAdd(this.d); + + assert(!this.twisted || this.c.fromRed().cmpn(1) === 0); + this.oneC = (conf.c | 0) === 1; +} +inherits(EdwardsCurve, Base); +module.exports = EdwardsCurve; + +EdwardsCurve.prototype._mulA = function _mulA(num) { + if (this.mOneA) + return num.redNeg(); + else + return this.a.redMul(num); +}; + +EdwardsCurve.prototype._mulC = function _mulC(num) { + if (this.oneC) + return num; + else + return this.c.redMul(num); +}; + +// Just for compatibility with Short curve +EdwardsCurve.prototype.jpoint = function jpoint(x, y, z, t) { + return this.point(x, y, z, t); +}; + +EdwardsCurve.prototype.pointFromX = function pointFromX(x, odd) { + x = new BN(x, 16); + if (!x.red) + x = x.toRed(this.red); + + var x2 = x.redSqr(); + var rhs = this.c2.redSub(this.a.redMul(x2)); + var lhs = this.one.redSub(this.c2.redMul(this.d).redMul(x2)); + + var y2 = rhs.redMul(lhs.redInvm()); + var y = y2.redSqrt(); + if (y.redSqr().redSub(y2).cmp(this.zero) !== 0) + throw new Error('invalid point'); + + var isOdd = y.fromRed().isOdd(); + if (odd && !isOdd || !odd && isOdd) + y = y.redNeg(); + + return this.point(x, y); +}; + +EdwardsCurve.prototype.pointFromY = function pointFromY(y, odd) { + y = new BN(y, 16); + if (!y.red) + y = y.toRed(this.red); + + // x^2 = (y^2 - c^2) / (c^2 d y^2 - a) + var y2 = y.redSqr(); + var lhs = y2.redSub(this.c2); + var rhs = y2.redMul(this.d).redMul(this.c2).redSub(this.a); + var x2 = lhs.redMul(rhs.redInvm()); + + if (x2.cmp(this.zero) === 0) { + if (odd) + throw new Error('invalid point'); + else + return this.point(this.zero, y); + } + + var x = x2.redSqrt(); + if (x.redSqr().redSub(x2).cmp(this.zero) !== 0) + throw new Error('invalid point'); + + if (x.fromRed().isOdd() !== odd) + x = x.redNeg(); + + return this.point(x, y); +}; + +EdwardsCurve.prototype.validate = function validate(point) { + if (point.isInfinity()) + return true; + + // Curve: A * X^2 + Y^2 = C^2 * (1 + D * X^2 * Y^2) + point.normalize(); + + var x2 = point.x.redSqr(); + var y2 = point.y.redSqr(); + var lhs = x2.redMul(this.a).redAdd(y2); + var rhs = this.c2.redMul(this.one.redAdd(this.d.redMul(x2).redMul(y2))); + + return lhs.cmp(rhs) === 0; +}; + +function Point(curve, x, y, z, t) { + Base.BasePoint.call(this, curve, 'projective'); + if (x === null && y === null && z === null) { + this.x = this.curve.zero; + this.y = this.curve.one; + this.z = this.curve.one; + this.t = this.curve.zero; + this.zOne = true; + } else { + this.x = new BN(x, 16); + this.y = new BN(y, 16); + this.z = z ? new BN(z, 16) : this.curve.one; + this.t = t && new BN(t, 16); + if (!this.x.red) + this.x = this.x.toRed(this.curve.red); + if (!this.y.red) + this.y = this.y.toRed(this.curve.red); + if (!this.z.red) + this.z = this.z.toRed(this.curve.red); + if (this.t && !this.t.red) + this.t = this.t.toRed(this.curve.red); + this.zOne = this.z === this.curve.one; + + // Use extended coordinates + if (this.curve.extended && !this.t) { + this.t = this.x.redMul(this.y); + if (!this.zOne) + this.t = this.t.redMul(this.z.redInvm()); + } + } +} +inherits(Point, Base.BasePoint); + +EdwardsCurve.prototype.pointFromJSON = function pointFromJSON(obj) { + return Point.fromJSON(this, obj); +}; + +EdwardsCurve.prototype.point = function point(x, y, z, t) { + return new Point(this, x, y, z, t); +}; + +Point.fromJSON = function fromJSON(curve, obj) { + return new Point(curve, obj[0], obj[1], obj[2]); +}; + +Point.prototype.inspect = function inspect() { + if (this.isInfinity()) + return ''; + return ''; +}; + +Point.prototype.isInfinity = function isInfinity() { + // XXX This code assumes that zero is always zero in red + return this.x.cmpn(0) === 0 && + (this.y.cmp(this.z) === 0 || + (this.zOne && this.y.cmp(this.curve.c) === 0)); +}; + +Point.prototype._extDbl = function _extDbl() { + // hyperelliptic.org/EFD/g1p/auto-twisted-extended-1.html + // #doubling-dbl-2008-hwcd + // 4M + 4S + + // A = X1^2 + var a = this.x.redSqr(); + // B = Y1^2 + var b = this.y.redSqr(); + // C = 2 * Z1^2 + var c = this.z.redSqr(); + c = c.redIAdd(c); + // D = a * A + var d = this.curve._mulA(a); + // E = (X1 + Y1)^2 - A - B + var e = this.x.redAdd(this.y).redSqr().redISub(a).redISub(b); + // G = D + B + var g = d.redAdd(b); + // F = G - C + var f = g.redSub(c); + // H = D - B + var h = d.redSub(b); + // X3 = E * F + var nx = e.redMul(f); + // Y3 = G * H + var ny = g.redMul(h); + // T3 = E * H + var nt = e.redMul(h); + // Z3 = F * G + var nz = f.redMul(g); + return this.curve.point(nx, ny, nz, nt); +}; + +Point.prototype._projDbl = function _projDbl() { + // hyperelliptic.org/EFD/g1p/auto-twisted-projective.html + // #doubling-dbl-2008-bbjlp + // #doubling-dbl-2007-bl + // and others + // Generally 3M + 4S or 2M + 4S + + // B = (X1 + Y1)^2 + var b = this.x.redAdd(this.y).redSqr(); + // C = X1^2 + var c = this.x.redSqr(); + // D = Y1^2 + var d = this.y.redSqr(); + + var nx; + var ny; + var nz; + var e; + var h; + var j; + if (this.curve.twisted) { + // E = a * C + e = this.curve._mulA(c); + // F = E + D + var f = e.redAdd(d); + if (this.zOne) { + // X3 = (B - C - D) * (F - 2) + nx = b.redSub(c).redSub(d).redMul(f.redSub(this.curve.two)); + // Y3 = F * (E - D) + ny = f.redMul(e.redSub(d)); + // Z3 = F^2 - 2 * F + nz = f.redSqr().redSub(f).redSub(f); + } else { + // H = Z1^2 + h = this.z.redSqr(); + // J = F - 2 * H + j = f.redSub(h).redISub(h); + // X3 = (B-C-D)*J + nx = b.redSub(c).redISub(d).redMul(j); + // Y3 = F * (E - D) + ny = f.redMul(e.redSub(d)); + // Z3 = F * J + nz = f.redMul(j); + } + } else { + // E = C + D + e = c.redAdd(d); + // H = (c * Z1)^2 + h = this.curve._mulC(this.z).redSqr(); + // J = E - 2 * H + j = e.redSub(h).redSub(h); + // X3 = c * (B - E) * J + nx = this.curve._mulC(b.redISub(e)).redMul(j); + // Y3 = c * E * (C - D) + ny = this.curve._mulC(e).redMul(c.redISub(d)); + // Z3 = E * J + nz = e.redMul(j); + } + return this.curve.point(nx, ny, nz); +}; + +Point.prototype.dbl = function dbl() { + if (this.isInfinity()) + return this; + + // Double in extended coordinates + if (this.curve.extended) + return this._extDbl(); + else + return this._projDbl(); +}; + +Point.prototype._extAdd = function _extAdd(p) { + // hyperelliptic.org/EFD/g1p/auto-twisted-extended-1.html + // #addition-add-2008-hwcd-3 + // 8M + + // A = (Y1 - X1) * (Y2 - X2) + var a = this.y.redSub(this.x).redMul(p.y.redSub(p.x)); + // B = (Y1 + X1) * (Y2 + X2) + var b = this.y.redAdd(this.x).redMul(p.y.redAdd(p.x)); + // C = T1 * k * T2 + var c = this.t.redMul(this.curve.dd).redMul(p.t); + // D = Z1 * 2 * Z2 + var d = this.z.redMul(p.z.redAdd(p.z)); + // E = B - A + var e = b.redSub(a); + // F = D - C + var f = d.redSub(c); + // G = D + C + var g = d.redAdd(c); + // H = B + A + var h = b.redAdd(a); + // X3 = E * F + var nx = e.redMul(f); + // Y3 = G * H + var ny = g.redMul(h); + // T3 = E * H + var nt = e.redMul(h); + // Z3 = F * G + var nz = f.redMul(g); + return this.curve.point(nx, ny, nz, nt); +}; + +Point.prototype._projAdd = function _projAdd(p) { + // hyperelliptic.org/EFD/g1p/auto-twisted-projective.html + // #addition-add-2008-bbjlp + // #addition-add-2007-bl + // 10M + 1S + + // A = Z1 * Z2 + var a = this.z.redMul(p.z); + // B = A^2 + var b = a.redSqr(); + // C = X1 * X2 + var c = this.x.redMul(p.x); + // D = Y1 * Y2 + var d = this.y.redMul(p.y); + // E = d * C * D + var e = this.curve.d.redMul(c).redMul(d); + // F = B - E + var f = b.redSub(e); + // G = B + E + var g = b.redAdd(e); + // X3 = A * F * ((X1 + Y1) * (X2 + Y2) - C - D) + var tmp = this.x.redAdd(this.y).redMul(p.x.redAdd(p.y)).redISub(c).redISub(d); + var nx = a.redMul(f).redMul(tmp); + var ny; + var nz; + if (this.curve.twisted) { + // Y3 = A * G * (D - a * C) + ny = a.redMul(g).redMul(d.redSub(this.curve._mulA(c))); + // Z3 = F * G + nz = f.redMul(g); + } else { + // Y3 = A * G * (D - C) + ny = a.redMul(g).redMul(d.redSub(c)); + // Z3 = c * F * G + nz = this.curve._mulC(f).redMul(g); + } + return this.curve.point(nx, ny, nz); +}; + +Point.prototype.add = function add(p) { + if (this.isInfinity()) + return p; + if (p.isInfinity()) + return this; + + if (this.curve.extended) + return this._extAdd(p); + else + return this._projAdd(p); +}; + +Point.prototype.mul = function mul(k) { + if (this._hasDoubles(k)) + return this.curve._fixedNafMul(this, k); + else + return this.curve._wnafMul(this, k); +}; + +Point.prototype.mulAdd = function mulAdd(k1, p, k2) { + return this.curve._wnafMulAdd(1, [ this, p ], [ k1, k2 ], 2, false); +}; + +Point.prototype.jmulAdd = function jmulAdd(k1, p, k2) { + return this.curve._wnafMulAdd(1, [ this, p ], [ k1, k2 ], 2, true); +}; + +Point.prototype.normalize = function normalize() { + if (this.zOne) + return this; + + // Normalize coordinates + var zi = this.z.redInvm(); + this.x = this.x.redMul(zi); + this.y = this.y.redMul(zi); + if (this.t) + this.t = this.t.redMul(zi); + this.z = this.curve.one; + this.zOne = true; + return this; +}; + +Point.prototype.neg = function neg() { + return this.curve.point(this.x.redNeg(), + this.y, + this.z, + this.t && this.t.redNeg()); +}; + +Point.prototype.getX = function getX() { + this.normalize(); + return this.x.fromRed(); +}; + +Point.prototype.getY = function getY() { + this.normalize(); + return this.y.fromRed(); +}; + +Point.prototype.eq = function eq(other) { + return this === other || + this.getX().cmp(other.getX()) === 0 && + this.getY().cmp(other.getY()) === 0; +}; + +Point.prototype.eqXToP = function eqXToP(x) { + var rx = x.toRed(this.curve.red).redMul(this.z); + if (this.x.cmp(rx) === 0) + return true; + + var xc = x.clone(); + var t = this.curve.redN.redMul(this.z); + for (;;) { + xc.iadd(this.curve.n); + if (xc.cmp(this.curve.p) >= 0) + return false; + + rx.redIAdd(t); + if (this.x.cmp(rx) === 0) + return true; + } +}; + +// Compatibility with BaseCurve +Point.prototype.toP = Point.prototype.normalize; +Point.prototype.mixedAdd = Point.prototype.add; diff --git a/node_modules/elliptic/lib/elliptic/curve/index.js b/node_modules/elliptic/lib/elliptic/curve/index.js new file mode 100644 index 0000000..c589281 --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/curve/index.js @@ -0,0 +1,8 @@ +'use strict'; + +var curve = exports; + +curve.base = require('./base'); +curve.short = require('./short'); +curve.mont = require('./mont'); +curve.edwards = require('./edwards'); diff --git a/node_modules/elliptic/lib/elliptic/curve/mont.js b/node_modules/elliptic/lib/elliptic/curve/mont.js new file mode 100644 index 0000000..4b9f80f --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/curve/mont.js @@ -0,0 +1,178 @@ +'use strict'; + +var BN = require('bn.js'); +var inherits = require('inherits'); +var Base = require('./base'); + +var utils = require('../utils'); + +function MontCurve(conf) { + Base.call(this, 'mont', conf); + + this.a = new BN(conf.a, 16).toRed(this.red); + this.b = new BN(conf.b, 16).toRed(this.red); + this.i4 = new BN(4).toRed(this.red).redInvm(); + this.two = new BN(2).toRed(this.red); + this.a24 = this.i4.redMul(this.a.redAdd(this.two)); +} +inherits(MontCurve, Base); +module.exports = MontCurve; + +MontCurve.prototype.validate = function validate(point) { + var x = point.normalize().x; + var x2 = x.redSqr(); + var rhs = x2.redMul(x).redAdd(x2.redMul(this.a)).redAdd(x); + var y = rhs.redSqrt(); + + return y.redSqr().cmp(rhs) === 0; +}; + +function Point(curve, x, z) { + Base.BasePoint.call(this, curve, 'projective'); + if (x === null && z === null) { + this.x = this.curve.one; + this.z = this.curve.zero; + } else { + this.x = new BN(x, 16); + this.z = new BN(z, 16); + if (!this.x.red) + this.x = this.x.toRed(this.curve.red); + if (!this.z.red) + this.z = this.z.toRed(this.curve.red); + } +} +inherits(Point, Base.BasePoint); + +MontCurve.prototype.decodePoint = function decodePoint(bytes, enc) { + return this.point(utils.toArray(bytes, enc), 1); +}; + +MontCurve.prototype.point = function point(x, z) { + return new Point(this, x, z); +}; + +MontCurve.prototype.pointFromJSON = function pointFromJSON(obj) { + return Point.fromJSON(this, obj); +}; + +Point.prototype.precompute = function precompute() { + // No-op +}; + +Point.prototype._encode = function _encode() { + return this.getX().toArray('be', this.curve.p.byteLength()); +}; + +Point.fromJSON = function fromJSON(curve, obj) { + return new Point(curve, obj[0], obj[1] || curve.one); +}; + +Point.prototype.inspect = function inspect() { + if (this.isInfinity()) + return ''; + return ''; +}; + +Point.prototype.isInfinity = function isInfinity() { + // XXX This code assumes that zero is always zero in red + return this.z.cmpn(0) === 0; +}; + +Point.prototype.dbl = function dbl() { + // http://hyperelliptic.org/EFD/g1p/auto-montgom-xz.html#doubling-dbl-1987-m-3 + // 2M + 2S + 4A + + // A = X1 + Z1 + var a = this.x.redAdd(this.z); + // AA = A^2 + var aa = a.redSqr(); + // B = X1 - Z1 + var b = this.x.redSub(this.z); + // BB = B^2 + var bb = b.redSqr(); + // C = AA - BB + var c = aa.redSub(bb); + // X3 = AA * BB + var nx = aa.redMul(bb); + // Z3 = C * (BB + A24 * C) + var nz = c.redMul(bb.redAdd(this.curve.a24.redMul(c))); + return this.curve.point(nx, nz); +}; + +Point.prototype.add = function add() { + throw new Error('Not supported on Montgomery curve'); +}; + +Point.prototype.diffAdd = function diffAdd(p, diff) { + // http://hyperelliptic.org/EFD/g1p/auto-montgom-xz.html#diffadd-dadd-1987-m-3 + // 4M + 2S + 6A + + // A = X2 + Z2 + var a = this.x.redAdd(this.z); + // B = X2 - Z2 + var b = this.x.redSub(this.z); + // C = X3 + Z3 + var c = p.x.redAdd(p.z); + // D = X3 - Z3 + var d = p.x.redSub(p.z); + // DA = D * A + var da = d.redMul(a); + // CB = C * B + var cb = c.redMul(b); + // X5 = Z1 * (DA + CB)^2 + var nx = diff.z.redMul(da.redAdd(cb).redSqr()); + // Z5 = X1 * (DA - CB)^2 + var nz = diff.x.redMul(da.redISub(cb).redSqr()); + return this.curve.point(nx, nz); +}; + +Point.prototype.mul = function mul(k) { + var t = k.clone(); + var a = this; // (N / 2) * Q + Q + var b = this.curve.point(null, null); // (N / 2) * Q + var c = this; // Q + + for (var bits = []; t.cmpn(0) !== 0; t.iushrn(1)) + bits.push(t.andln(1)); + + for (var i = bits.length - 1; i >= 0; i--) { + if (bits[i] === 0) { + // N * Q + Q = ((N / 2) * Q + Q)) + (N / 2) * Q + a = a.diffAdd(b, c); + // N * Q = 2 * ((N / 2) * Q + Q)) + b = b.dbl(); + } else { + // N * Q = ((N / 2) * Q + Q) + ((N / 2) * Q) + b = a.diffAdd(b, c); + // N * Q + Q = 2 * ((N / 2) * Q + Q) + a = a.dbl(); + } + } + return b; +}; + +Point.prototype.mulAdd = function mulAdd() { + throw new Error('Not supported on Montgomery curve'); +}; + +Point.prototype.jumlAdd = function jumlAdd() { + throw new Error('Not supported on Montgomery curve'); +}; + +Point.prototype.eq = function eq(other) { + return this.getX().cmp(other.getX()) === 0; +}; + +Point.prototype.normalize = function normalize() { + this.x = this.x.redMul(this.z.redInvm()); + this.z = this.curve.one; + return this; +}; + +Point.prototype.getX = function getX() { + // Normalize coordinates + this.normalize(); + + return this.x.fromRed(); +}; diff --git a/node_modules/elliptic/lib/elliptic/curve/short.js b/node_modules/elliptic/lib/elliptic/curve/short.js new file mode 100644 index 0000000..eec36ec --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/curve/short.js @@ -0,0 +1,938 @@ +'use strict'; + +var utils = require('../utils'); +var BN = require('bn.js'); +var inherits = require('inherits'); +var Base = require('./base'); + +var assert = utils.assert; + +function ShortCurve(conf) { + Base.call(this, 'short', conf); + + this.a = new BN(conf.a, 16).toRed(this.red); + this.b = new BN(conf.b, 16).toRed(this.red); + this.tinv = this.two.redInvm(); + + this.zeroA = this.a.fromRed().cmpn(0) === 0; + this.threeA = this.a.fromRed().sub(this.p).cmpn(-3) === 0; + + // If the curve is endomorphic, precalculate beta and lambda + this.endo = this._getEndomorphism(conf); + this._endoWnafT1 = new Array(4); + this._endoWnafT2 = new Array(4); +} +inherits(ShortCurve, Base); +module.exports = ShortCurve; + +ShortCurve.prototype._getEndomorphism = function _getEndomorphism(conf) { + // No efficient endomorphism + if (!this.zeroA || !this.g || !this.n || this.p.modn(3) !== 1) + return; + + // Compute beta and lambda, that lambda * P = (beta * Px; Py) + var beta; + var lambda; + if (conf.beta) { + beta = new BN(conf.beta, 16).toRed(this.red); + } else { + var betas = this._getEndoRoots(this.p); + // Choose the smallest beta + beta = betas[0].cmp(betas[1]) < 0 ? betas[0] : betas[1]; + beta = beta.toRed(this.red); + } + if (conf.lambda) { + lambda = new BN(conf.lambda, 16); + } else { + // Choose the lambda that is matching selected beta + var lambdas = this._getEndoRoots(this.n); + if (this.g.mul(lambdas[0]).x.cmp(this.g.x.redMul(beta)) === 0) { + lambda = lambdas[0]; + } else { + lambda = lambdas[1]; + assert(this.g.mul(lambda).x.cmp(this.g.x.redMul(beta)) === 0); + } + } + + // Get basis vectors, used for balanced length-two representation + var basis; + if (conf.basis) { + basis = conf.basis.map(function(vec) { + return { + a: new BN(vec.a, 16), + b: new BN(vec.b, 16), + }; + }); + } else { + basis = this._getEndoBasis(lambda); + } + + return { + beta: beta, + lambda: lambda, + basis: basis, + }; +}; + +ShortCurve.prototype._getEndoRoots = function _getEndoRoots(num) { + // Find roots of for x^2 + x + 1 in F + // Root = (-1 +- Sqrt(-3)) / 2 + // + var red = num === this.p ? this.red : BN.mont(num); + var tinv = new BN(2).toRed(red).redInvm(); + var ntinv = tinv.redNeg(); + + var s = new BN(3).toRed(red).redNeg().redSqrt().redMul(tinv); + + var l1 = ntinv.redAdd(s).fromRed(); + var l2 = ntinv.redSub(s).fromRed(); + return [ l1, l2 ]; +}; + +ShortCurve.prototype._getEndoBasis = function _getEndoBasis(lambda) { + // aprxSqrt >= sqrt(this.n) + var aprxSqrt = this.n.ushrn(Math.floor(this.n.bitLength() / 2)); + + // 3.74 + // Run EGCD, until r(L + 1) < aprxSqrt + var u = lambda; + var v = this.n.clone(); + var x1 = new BN(1); + var y1 = new BN(0); + var x2 = new BN(0); + var y2 = new BN(1); + + // NOTE: all vectors are roots of: a + b * lambda = 0 (mod n) + var a0; + var b0; + // First vector + var a1; + var b1; + // Second vector + var a2; + var b2; + + var prevR; + var i = 0; + var r; + var x; + while (u.cmpn(0) !== 0) { + var q = v.div(u); + r = v.sub(q.mul(u)); + x = x2.sub(q.mul(x1)); + var y = y2.sub(q.mul(y1)); + + if (!a1 && r.cmp(aprxSqrt) < 0) { + a0 = prevR.neg(); + b0 = x1; + a1 = r.neg(); + b1 = x; + } else if (a1 && ++i === 2) { + break; + } + prevR = r; + + v = u; + u = r; + x2 = x1; + x1 = x; + y2 = y1; + y1 = y; + } + a2 = r.neg(); + b2 = x; + + var len1 = a1.sqr().add(b1.sqr()); + var len2 = a2.sqr().add(b2.sqr()); + if (len2.cmp(len1) >= 0) { + a2 = a0; + b2 = b0; + } + + // Normalize signs + if (a1.negative) { + a1 = a1.neg(); + b1 = b1.neg(); + } + if (a2.negative) { + a2 = a2.neg(); + b2 = b2.neg(); + } + + return [ + { a: a1, b: b1 }, + { a: a2, b: b2 }, + ]; +}; + +ShortCurve.prototype._endoSplit = function _endoSplit(k) { + var basis = this.endo.basis; + var v1 = basis[0]; + var v2 = basis[1]; + + var c1 = v2.b.mul(k).divRound(this.n); + var c2 = v1.b.neg().mul(k).divRound(this.n); + + var p1 = c1.mul(v1.a); + var p2 = c2.mul(v2.a); + var q1 = c1.mul(v1.b); + var q2 = c2.mul(v2.b); + + // Calculate answer + var k1 = k.sub(p1).sub(p2); + var k2 = q1.add(q2).neg(); + return { k1: k1, k2: k2 }; +}; + +ShortCurve.prototype.pointFromX = function pointFromX(x, odd) { + x = new BN(x, 16); + if (!x.red) + x = x.toRed(this.red); + + var y2 = x.redSqr().redMul(x).redIAdd(x.redMul(this.a)).redIAdd(this.b); + var y = y2.redSqrt(); + if (y.redSqr().redSub(y2).cmp(this.zero) !== 0) + throw new Error('invalid point'); + + // XXX Is there any way to tell if the number is odd without converting it + // to non-red form? + var isOdd = y.fromRed().isOdd(); + if (odd && !isOdd || !odd && isOdd) + y = y.redNeg(); + + return this.point(x, y); +}; + +ShortCurve.prototype.validate = function validate(point) { + if (point.inf) + return true; + + var x = point.x; + var y = point.y; + + var ax = this.a.redMul(x); + var rhs = x.redSqr().redMul(x).redIAdd(ax).redIAdd(this.b); + return y.redSqr().redISub(rhs).cmpn(0) === 0; +}; + +ShortCurve.prototype._endoWnafMulAdd = + function _endoWnafMulAdd(points, coeffs, jacobianResult) { + var npoints = this._endoWnafT1; + var ncoeffs = this._endoWnafT2; + for (var i = 0; i < points.length; i++) { + var split = this._endoSplit(coeffs[i]); + var p = points[i]; + var beta = p._getBeta(); + + if (split.k1.negative) { + split.k1.ineg(); + p = p.neg(true); + } + if (split.k2.negative) { + split.k2.ineg(); + beta = beta.neg(true); + } + + npoints[i * 2] = p; + npoints[i * 2 + 1] = beta; + ncoeffs[i * 2] = split.k1; + ncoeffs[i * 2 + 1] = split.k2; + } + var res = this._wnafMulAdd(1, npoints, ncoeffs, i * 2, jacobianResult); + + // Clean-up references to points and coefficients + for (var j = 0; j < i * 2; j++) { + npoints[j] = null; + ncoeffs[j] = null; + } + return res; + }; + +function Point(curve, x, y, isRed) { + Base.BasePoint.call(this, curve, 'affine'); + if (x === null && y === null) { + this.x = null; + this.y = null; + this.inf = true; + } else { + this.x = new BN(x, 16); + this.y = new BN(y, 16); + // Force redgomery representation when loading from JSON + if (isRed) { + this.x.forceRed(this.curve.red); + this.y.forceRed(this.curve.red); + } + if (!this.x.red) + this.x = this.x.toRed(this.curve.red); + if (!this.y.red) + this.y = this.y.toRed(this.curve.red); + this.inf = false; + } +} +inherits(Point, Base.BasePoint); + +ShortCurve.prototype.point = function point(x, y, isRed) { + return new Point(this, x, y, isRed); +}; + +ShortCurve.prototype.pointFromJSON = function pointFromJSON(obj, red) { + return Point.fromJSON(this, obj, red); +}; + +Point.prototype._getBeta = function _getBeta() { + if (!this.curve.endo) + return; + + var pre = this.precomputed; + if (pre && pre.beta) + return pre.beta; + + var beta = this.curve.point(this.x.redMul(this.curve.endo.beta), this.y); + if (pre) { + var curve = this.curve; + var endoMul = function(p) { + return curve.point(p.x.redMul(curve.endo.beta), p.y); + }; + pre.beta = beta; + beta.precomputed = { + beta: null, + naf: pre.naf && { + wnd: pre.naf.wnd, + points: pre.naf.points.map(endoMul), + }, + doubles: pre.doubles && { + step: pre.doubles.step, + points: pre.doubles.points.map(endoMul), + }, + }; + } + return beta; +}; + +Point.prototype.toJSON = function toJSON() { + if (!this.precomputed) + return [ this.x, this.y ]; + + return [ this.x, this.y, this.precomputed && { + doubles: this.precomputed.doubles && { + step: this.precomputed.doubles.step, + points: this.precomputed.doubles.points.slice(1), + }, + naf: this.precomputed.naf && { + wnd: this.precomputed.naf.wnd, + points: this.precomputed.naf.points.slice(1), + }, + } ]; +}; + +Point.fromJSON = function fromJSON(curve, obj, red) { + if (typeof obj === 'string') + obj = JSON.parse(obj); + var res = curve.point(obj[0], obj[1], red); + if (!obj[2]) + return res; + + function obj2point(obj) { + return curve.point(obj[0], obj[1], red); + } + + var pre = obj[2]; + res.precomputed = { + beta: null, + doubles: pre.doubles && { + step: pre.doubles.step, + points: [ res ].concat(pre.doubles.points.map(obj2point)), + }, + naf: pre.naf && { + wnd: pre.naf.wnd, + points: [ res ].concat(pre.naf.points.map(obj2point)), + }, + }; + return res; +}; + +Point.prototype.inspect = function inspect() { + if (this.isInfinity()) + return ''; + return ''; +}; + +Point.prototype.isInfinity = function isInfinity() { + return this.inf; +}; + +Point.prototype.add = function add(p) { + // O + P = P + if (this.inf) + return p; + + // P + O = P + if (p.inf) + return this; + + // P + P = 2P + if (this.eq(p)) + return this.dbl(); + + // P + (-P) = O + if (this.neg().eq(p)) + return this.curve.point(null, null); + + // P + Q = O + if (this.x.cmp(p.x) === 0) + return this.curve.point(null, null); + + var c = this.y.redSub(p.y); + if (c.cmpn(0) !== 0) + c = c.redMul(this.x.redSub(p.x).redInvm()); + var nx = c.redSqr().redISub(this.x).redISub(p.x); + var ny = c.redMul(this.x.redSub(nx)).redISub(this.y); + return this.curve.point(nx, ny); +}; + +Point.prototype.dbl = function dbl() { + if (this.inf) + return this; + + // 2P = O + var ys1 = this.y.redAdd(this.y); + if (ys1.cmpn(0) === 0) + return this.curve.point(null, null); + + var a = this.curve.a; + + var x2 = this.x.redSqr(); + var dyinv = ys1.redInvm(); + var c = x2.redAdd(x2).redIAdd(x2).redIAdd(a).redMul(dyinv); + + var nx = c.redSqr().redISub(this.x.redAdd(this.x)); + var ny = c.redMul(this.x.redSub(nx)).redISub(this.y); + return this.curve.point(nx, ny); +}; + +Point.prototype.getX = function getX() { + return this.x.fromRed(); +}; + +Point.prototype.getY = function getY() { + return this.y.fromRed(); +}; + +Point.prototype.mul = function mul(k) { + k = new BN(k, 16); + if (this.isInfinity()) + return this; + else if (this._hasDoubles(k)) + return this.curve._fixedNafMul(this, k); + else if (this.curve.endo) + return this.curve._endoWnafMulAdd([ this ], [ k ]); + else + return this.curve._wnafMul(this, k); +}; + +Point.prototype.mulAdd = function mulAdd(k1, p2, k2) { + var points = [ this, p2 ]; + var coeffs = [ k1, k2 ]; + if (this.curve.endo) + return this.curve._endoWnafMulAdd(points, coeffs); + else + return this.curve._wnafMulAdd(1, points, coeffs, 2); +}; + +Point.prototype.jmulAdd = function jmulAdd(k1, p2, k2) { + var points = [ this, p2 ]; + var coeffs = [ k1, k2 ]; + if (this.curve.endo) + return this.curve._endoWnafMulAdd(points, coeffs, true); + else + return this.curve._wnafMulAdd(1, points, coeffs, 2, true); +}; + +Point.prototype.eq = function eq(p) { + return this === p || + this.inf === p.inf && + (this.inf || this.x.cmp(p.x) === 0 && this.y.cmp(p.y) === 0); +}; + +Point.prototype.neg = function neg(_precompute) { + if (this.inf) + return this; + + var res = this.curve.point(this.x, this.y.redNeg()); + if (_precompute && this.precomputed) { + var pre = this.precomputed; + var negate = function(p) { + return p.neg(); + }; + res.precomputed = { + naf: pre.naf && { + wnd: pre.naf.wnd, + points: pre.naf.points.map(negate), + }, + doubles: pre.doubles && { + step: pre.doubles.step, + points: pre.doubles.points.map(negate), + }, + }; + } + return res; +}; + +Point.prototype.toJ = function toJ() { + if (this.inf) + return this.curve.jpoint(null, null, null); + + var res = this.curve.jpoint(this.x, this.y, this.curve.one); + return res; +}; + +function JPoint(curve, x, y, z) { + Base.BasePoint.call(this, curve, 'jacobian'); + if (x === null && y === null && z === null) { + this.x = this.curve.one; + this.y = this.curve.one; + this.z = new BN(0); + } else { + this.x = new BN(x, 16); + this.y = new BN(y, 16); + this.z = new BN(z, 16); + } + if (!this.x.red) + this.x = this.x.toRed(this.curve.red); + if (!this.y.red) + this.y = this.y.toRed(this.curve.red); + if (!this.z.red) + this.z = this.z.toRed(this.curve.red); + + this.zOne = this.z === this.curve.one; +} +inherits(JPoint, Base.BasePoint); + +ShortCurve.prototype.jpoint = function jpoint(x, y, z) { + return new JPoint(this, x, y, z); +}; + +JPoint.prototype.toP = function toP() { + if (this.isInfinity()) + return this.curve.point(null, null); + + var zinv = this.z.redInvm(); + var zinv2 = zinv.redSqr(); + var ax = this.x.redMul(zinv2); + var ay = this.y.redMul(zinv2).redMul(zinv); + + return this.curve.point(ax, ay); +}; + +JPoint.prototype.neg = function neg() { + return this.curve.jpoint(this.x, this.y.redNeg(), this.z); +}; + +JPoint.prototype.add = function add(p) { + // O + P = P + if (this.isInfinity()) + return p; + + // P + O = P + if (p.isInfinity()) + return this; + + // 12M + 4S + 7A + var pz2 = p.z.redSqr(); + var z2 = this.z.redSqr(); + var u1 = this.x.redMul(pz2); + var u2 = p.x.redMul(z2); + var s1 = this.y.redMul(pz2.redMul(p.z)); + var s2 = p.y.redMul(z2.redMul(this.z)); + + var h = u1.redSub(u2); + var r = s1.redSub(s2); + if (h.cmpn(0) === 0) { + if (r.cmpn(0) !== 0) + return this.curve.jpoint(null, null, null); + else + return this.dbl(); + } + + var h2 = h.redSqr(); + var h3 = h2.redMul(h); + var v = u1.redMul(h2); + + var nx = r.redSqr().redIAdd(h3).redISub(v).redISub(v); + var ny = r.redMul(v.redISub(nx)).redISub(s1.redMul(h3)); + var nz = this.z.redMul(p.z).redMul(h); + + return this.curve.jpoint(nx, ny, nz); +}; + +JPoint.prototype.mixedAdd = function mixedAdd(p) { + // O + P = P + if (this.isInfinity()) + return p.toJ(); + + // P + O = P + if (p.isInfinity()) + return this; + + // 8M + 3S + 7A + var z2 = this.z.redSqr(); + var u1 = this.x; + var u2 = p.x.redMul(z2); + var s1 = this.y; + var s2 = p.y.redMul(z2).redMul(this.z); + + var h = u1.redSub(u2); + var r = s1.redSub(s2); + if (h.cmpn(0) === 0) { + if (r.cmpn(0) !== 0) + return this.curve.jpoint(null, null, null); + else + return this.dbl(); + } + + var h2 = h.redSqr(); + var h3 = h2.redMul(h); + var v = u1.redMul(h2); + + var nx = r.redSqr().redIAdd(h3).redISub(v).redISub(v); + var ny = r.redMul(v.redISub(nx)).redISub(s1.redMul(h3)); + var nz = this.z.redMul(h); + + return this.curve.jpoint(nx, ny, nz); +}; + +JPoint.prototype.dblp = function dblp(pow) { + if (pow === 0) + return this; + if (this.isInfinity()) + return this; + if (!pow) + return this.dbl(); + + var i; + if (this.curve.zeroA || this.curve.threeA) { + var r = this; + for (i = 0; i < pow; i++) + r = r.dbl(); + return r; + } + + // 1M + 2S + 1A + N * (4S + 5M + 8A) + // N = 1 => 6M + 6S + 9A + var a = this.curve.a; + var tinv = this.curve.tinv; + + var jx = this.x; + var jy = this.y; + var jz = this.z; + var jz4 = jz.redSqr().redSqr(); + + // Reuse results + var jyd = jy.redAdd(jy); + for (i = 0; i < pow; i++) { + var jx2 = jx.redSqr(); + var jyd2 = jyd.redSqr(); + var jyd4 = jyd2.redSqr(); + var c = jx2.redAdd(jx2).redIAdd(jx2).redIAdd(a.redMul(jz4)); + + var t1 = jx.redMul(jyd2); + var nx = c.redSqr().redISub(t1.redAdd(t1)); + var t2 = t1.redISub(nx); + var dny = c.redMul(t2); + dny = dny.redIAdd(dny).redISub(jyd4); + var nz = jyd.redMul(jz); + if (i + 1 < pow) + jz4 = jz4.redMul(jyd4); + + jx = nx; + jz = nz; + jyd = dny; + } + + return this.curve.jpoint(jx, jyd.redMul(tinv), jz); +}; + +JPoint.prototype.dbl = function dbl() { + if (this.isInfinity()) + return this; + + if (this.curve.zeroA) + return this._zeroDbl(); + else if (this.curve.threeA) + return this._threeDbl(); + else + return this._dbl(); +}; + +JPoint.prototype._zeroDbl = function _zeroDbl() { + var nx; + var ny; + var nz; + // Z = 1 + if (this.zOne) { + // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-0.html + // #doubling-mdbl-2007-bl + // 1M + 5S + 14A + + // XX = X1^2 + var xx = this.x.redSqr(); + // YY = Y1^2 + var yy = this.y.redSqr(); + // YYYY = YY^2 + var yyyy = yy.redSqr(); + // S = 2 * ((X1 + YY)^2 - XX - YYYY) + var s = this.x.redAdd(yy).redSqr().redISub(xx).redISub(yyyy); + s = s.redIAdd(s); + // M = 3 * XX + a; a = 0 + var m = xx.redAdd(xx).redIAdd(xx); + // T = M ^ 2 - 2*S + var t = m.redSqr().redISub(s).redISub(s); + + // 8 * YYYY + var yyyy8 = yyyy.redIAdd(yyyy); + yyyy8 = yyyy8.redIAdd(yyyy8); + yyyy8 = yyyy8.redIAdd(yyyy8); + + // X3 = T + nx = t; + // Y3 = M * (S - T) - 8 * YYYY + ny = m.redMul(s.redISub(t)).redISub(yyyy8); + // Z3 = 2*Y1 + nz = this.y.redAdd(this.y); + } else { + // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-0.html + // #doubling-dbl-2009-l + // 2M + 5S + 13A + + // A = X1^2 + var a = this.x.redSqr(); + // B = Y1^2 + var b = this.y.redSqr(); + // C = B^2 + var c = b.redSqr(); + // D = 2 * ((X1 + B)^2 - A - C) + var d = this.x.redAdd(b).redSqr().redISub(a).redISub(c); + d = d.redIAdd(d); + // E = 3 * A + var e = a.redAdd(a).redIAdd(a); + // F = E^2 + var f = e.redSqr(); + + // 8 * C + var c8 = c.redIAdd(c); + c8 = c8.redIAdd(c8); + c8 = c8.redIAdd(c8); + + // X3 = F - 2 * D + nx = f.redISub(d).redISub(d); + // Y3 = E * (D - X3) - 8 * C + ny = e.redMul(d.redISub(nx)).redISub(c8); + // Z3 = 2 * Y1 * Z1 + nz = this.y.redMul(this.z); + nz = nz.redIAdd(nz); + } + + return this.curve.jpoint(nx, ny, nz); +}; + +JPoint.prototype._threeDbl = function _threeDbl() { + var nx; + var ny; + var nz; + // Z = 1 + if (this.zOne) { + // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-3.html + // #doubling-mdbl-2007-bl + // 1M + 5S + 15A + + // XX = X1^2 + var xx = this.x.redSqr(); + // YY = Y1^2 + var yy = this.y.redSqr(); + // YYYY = YY^2 + var yyyy = yy.redSqr(); + // S = 2 * ((X1 + YY)^2 - XX - YYYY) + var s = this.x.redAdd(yy).redSqr().redISub(xx).redISub(yyyy); + s = s.redIAdd(s); + // M = 3 * XX + a + var m = xx.redAdd(xx).redIAdd(xx).redIAdd(this.curve.a); + // T = M^2 - 2 * S + var t = m.redSqr().redISub(s).redISub(s); + // X3 = T + nx = t; + // Y3 = M * (S - T) - 8 * YYYY + var yyyy8 = yyyy.redIAdd(yyyy); + yyyy8 = yyyy8.redIAdd(yyyy8); + yyyy8 = yyyy8.redIAdd(yyyy8); + ny = m.redMul(s.redISub(t)).redISub(yyyy8); + // Z3 = 2 * Y1 + nz = this.y.redAdd(this.y); + } else { + // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-3.html#doubling-dbl-2001-b + // 3M + 5S + + // delta = Z1^2 + var delta = this.z.redSqr(); + // gamma = Y1^2 + var gamma = this.y.redSqr(); + // beta = X1 * gamma + var beta = this.x.redMul(gamma); + // alpha = 3 * (X1 - delta) * (X1 + delta) + var alpha = this.x.redSub(delta).redMul(this.x.redAdd(delta)); + alpha = alpha.redAdd(alpha).redIAdd(alpha); + // X3 = alpha^2 - 8 * beta + var beta4 = beta.redIAdd(beta); + beta4 = beta4.redIAdd(beta4); + var beta8 = beta4.redAdd(beta4); + nx = alpha.redSqr().redISub(beta8); + // Z3 = (Y1 + Z1)^2 - gamma - delta + nz = this.y.redAdd(this.z).redSqr().redISub(gamma).redISub(delta); + // Y3 = alpha * (4 * beta - X3) - 8 * gamma^2 + var ggamma8 = gamma.redSqr(); + ggamma8 = ggamma8.redIAdd(ggamma8); + ggamma8 = ggamma8.redIAdd(ggamma8); + ggamma8 = ggamma8.redIAdd(ggamma8); + ny = alpha.redMul(beta4.redISub(nx)).redISub(ggamma8); + } + + return this.curve.jpoint(nx, ny, nz); +}; + +JPoint.prototype._dbl = function _dbl() { + var a = this.curve.a; + + // 4M + 6S + 10A + var jx = this.x; + var jy = this.y; + var jz = this.z; + var jz4 = jz.redSqr().redSqr(); + + var jx2 = jx.redSqr(); + var jy2 = jy.redSqr(); + + var c = jx2.redAdd(jx2).redIAdd(jx2).redIAdd(a.redMul(jz4)); + + var jxd4 = jx.redAdd(jx); + jxd4 = jxd4.redIAdd(jxd4); + var t1 = jxd4.redMul(jy2); + var nx = c.redSqr().redISub(t1.redAdd(t1)); + var t2 = t1.redISub(nx); + + var jyd8 = jy2.redSqr(); + jyd8 = jyd8.redIAdd(jyd8); + jyd8 = jyd8.redIAdd(jyd8); + jyd8 = jyd8.redIAdd(jyd8); + var ny = c.redMul(t2).redISub(jyd8); + var nz = jy.redAdd(jy).redMul(jz); + + return this.curve.jpoint(nx, ny, nz); +}; + +JPoint.prototype.trpl = function trpl() { + if (!this.curve.zeroA) + return this.dbl().add(this); + + // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-0.html#tripling-tpl-2007-bl + // 5M + 10S + ... + + // XX = X1^2 + var xx = this.x.redSqr(); + // YY = Y1^2 + var yy = this.y.redSqr(); + // ZZ = Z1^2 + var zz = this.z.redSqr(); + // YYYY = YY^2 + var yyyy = yy.redSqr(); + // M = 3 * XX + a * ZZ2; a = 0 + var m = xx.redAdd(xx).redIAdd(xx); + // MM = M^2 + var mm = m.redSqr(); + // E = 6 * ((X1 + YY)^2 - XX - YYYY) - MM + var e = this.x.redAdd(yy).redSqr().redISub(xx).redISub(yyyy); + e = e.redIAdd(e); + e = e.redAdd(e).redIAdd(e); + e = e.redISub(mm); + // EE = E^2 + var ee = e.redSqr(); + // T = 16*YYYY + var t = yyyy.redIAdd(yyyy); + t = t.redIAdd(t); + t = t.redIAdd(t); + t = t.redIAdd(t); + // U = (M + E)^2 - MM - EE - T + var u = m.redIAdd(e).redSqr().redISub(mm).redISub(ee).redISub(t); + // X3 = 4 * (X1 * EE - 4 * YY * U) + var yyu4 = yy.redMul(u); + yyu4 = yyu4.redIAdd(yyu4); + yyu4 = yyu4.redIAdd(yyu4); + var nx = this.x.redMul(ee).redISub(yyu4); + nx = nx.redIAdd(nx); + nx = nx.redIAdd(nx); + // Y3 = 8 * Y1 * (U * (T - U) - E * EE) + var ny = this.y.redMul(u.redMul(t.redISub(u)).redISub(e.redMul(ee))); + ny = ny.redIAdd(ny); + ny = ny.redIAdd(ny); + ny = ny.redIAdd(ny); + // Z3 = (Z1 + E)^2 - ZZ - EE + var nz = this.z.redAdd(e).redSqr().redISub(zz).redISub(ee); + + return this.curve.jpoint(nx, ny, nz); +}; + +JPoint.prototype.mul = function mul(k, kbase) { + k = new BN(k, kbase); + + return this.curve._wnafMul(this, k); +}; + +JPoint.prototype.eq = function eq(p) { + if (p.type === 'affine') + return this.eq(p.toJ()); + + if (this === p) + return true; + + // x1 * z2^2 == x2 * z1^2 + var z2 = this.z.redSqr(); + var pz2 = p.z.redSqr(); + if (this.x.redMul(pz2).redISub(p.x.redMul(z2)).cmpn(0) !== 0) + return false; + + // y1 * z2^3 == y2 * z1^3 + var z3 = z2.redMul(this.z); + var pz3 = pz2.redMul(p.z); + return this.y.redMul(pz3).redISub(p.y.redMul(z3)).cmpn(0) === 0; +}; + +JPoint.prototype.eqXToP = function eqXToP(x) { + var zs = this.z.redSqr(); + var rx = x.toRed(this.curve.red).redMul(zs); + if (this.x.cmp(rx) === 0) + return true; + + var xc = x.clone(); + var t = this.curve.redN.redMul(zs); + for (;;) { + xc.iadd(this.curve.n); + if (xc.cmp(this.curve.p) >= 0) + return false; + + rx.redIAdd(t); + if (this.x.cmp(rx) === 0) + return true; + } +}; + +JPoint.prototype.inspect = function inspect() { + if (this.isInfinity()) + return ''; + return ''; +}; + +JPoint.prototype.isInfinity = function isInfinity() { + // XXX This code assumes that zero is always zero in red + return this.z.cmpn(0) === 0; +}; diff --git a/node_modules/elliptic/lib/elliptic/curves.js b/node_modules/elliptic/lib/elliptic/curves.js new file mode 100644 index 0000000..6c36e03 --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/curves.js @@ -0,0 +1,206 @@ +'use strict'; + +var curves = exports; + +var hash = require('hash.js'); +var curve = require('./curve'); +var utils = require('./utils'); + +var assert = utils.assert; + +function PresetCurve(options) { + if (options.type === 'short') + this.curve = new curve.short(options); + else if (options.type === 'edwards') + this.curve = new curve.edwards(options); + else + this.curve = new curve.mont(options); + this.g = this.curve.g; + this.n = this.curve.n; + this.hash = options.hash; + + assert(this.g.validate(), 'Invalid curve'); + assert(this.g.mul(this.n).isInfinity(), 'Invalid curve, G*N != O'); +} +curves.PresetCurve = PresetCurve; + +function defineCurve(name, options) { + Object.defineProperty(curves, name, { + configurable: true, + enumerable: true, + get: function() { + var curve = new PresetCurve(options); + Object.defineProperty(curves, name, { + configurable: true, + enumerable: true, + value: curve, + }); + return curve; + }, + }); +} + +defineCurve('p192', { + type: 'short', + prime: 'p192', + p: 'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff', + a: 'ffffffff ffffffff ffffffff fffffffe ffffffff fffffffc', + b: '64210519 e59c80e7 0fa7e9ab 72243049 feb8deec c146b9b1', + n: 'ffffffff ffffffff ffffffff 99def836 146bc9b1 b4d22831', + hash: hash.sha256, + gRed: false, + g: [ + '188da80e b03090f6 7cbf20eb 43a18800 f4ff0afd 82ff1012', + '07192b95 ffc8da78 631011ed 6b24cdd5 73f977a1 1e794811', + ], +}); + +defineCurve('p224', { + type: 'short', + prime: 'p224', + p: 'ffffffff ffffffff ffffffff ffffffff 00000000 00000000 00000001', + a: 'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff fffffffe', + b: 'b4050a85 0c04b3ab f5413256 5044b0b7 d7bfd8ba 270b3943 2355ffb4', + n: 'ffffffff ffffffff ffffffff ffff16a2 e0b8f03e 13dd2945 5c5c2a3d', + hash: hash.sha256, + gRed: false, + g: [ + 'b70e0cbd 6bb4bf7f 321390b9 4a03c1d3 56c21122 343280d6 115c1d21', + 'bd376388 b5f723fb 4c22dfe6 cd4375a0 5a074764 44d58199 85007e34', + ], +}); + +defineCurve('p256', { + type: 'short', + prime: null, + p: 'ffffffff 00000001 00000000 00000000 00000000 ffffffff ffffffff ffffffff', + a: 'ffffffff 00000001 00000000 00000000 00000000 ffffffff ffffffff fffffffc', + b: '5ac635d8 aa3a93e7 b3ebbd55 769886bc 651d06b0 cc53b0f6 3bce3c3e 27d2604b', + n: 'ffffffff 00000000 ffffffff ffffffff bce6faad a7179e84 f3b9cac2 fc632551', + hash: hash.sha256, + gRed: false, + g: [ + '6b17d1f2 e12c4247 f8bce6e5 63a440f2 77037d81 2deb33a0 f4a13945 d898c296', + '4fe342e2 fe1a7f9b 8ee7eb4a 7c0f9e16 2bce3357 6b315ece cbb64068 37bf51f5', + ], +}); + +defineCurve('p384', { + type: 'short', + prime: null, + p: 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'fffffffe ffffffff 00000000 00000000 ffffffff', + a: 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'fffffffe ffffffff 00000000 00000000 fffffffc', + b: 'b3312fa7 e23ee7e4 988e056b e3f82d19 181d9c6e fe814112 0314088f ' + + '5013875a c656398d 8a2ed19d 2a85c8ed d3ec2aef', + n: 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff c7634d81 ' + + 'f4372ddf 581a0db2 48b0a77a ecec196a ccc52973', + hash: hash.sha384, + gRed: false, + g: [ + 'aa87ca22 be8b0537 8eb1c71e f320ad74 6e1d3b62 8ba79b98 59f741e0 82542a38 ' + + '5502f25d bf55296c 3a545e38 72760ab7', + '3617de4a 96262c6f 5d9e98bf 9292dc29 f8f41dbd 289a147c e9da3113 b5f0b8c0 ' + + '0a60b1ce 1d7e819d 7a431d7c 90ea0e5f', + ], +}); + +defineCurve('p521', { + type: 'short', + prime: null, + p: '000001ff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'ffffffff ffffffff ffffffff ffffffff ffffffff', + a: '000001ff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'ffffffff ffffffff ffffffff ffffffff fffffffc', + b: '00000051 953eb961 8e1c9a1f 929a21a0 b68540ee a2da725b ' + + '99b315f3 b8b48991 8ef109e1 56193951 ec7e937b 1652c0bd ' + + '3bb1bf07 3573df88 3d2c34f1 ef451fd4 6b503f00', + n: '000001ff ffffffff ffffffff ffffffff ffffffff ffffffff ' + + 'ffffffff ffffffff fffffffa 51868783 bf2f966b 7fcc0148 ' + + 'f709a5d0 3bb5c9b8 899c47ae bb6fb71e 91386409', + hash: hash.sha512, + gRed: false, + g: [ + '000000c6 858e06b7 0404e9cd 9e3ecb66 2395b442 9c648139 ' + + '053fb521 f828af60 6b4d3dba a14b5e77 efe75928 fe1dc127 ' + + 'a2ffa8de 3348b3c1 856a429b f97e7e31 c2e5bd66', + '00000118 39296a78 9a3bc004 5c8a5fb4 2c7d1bd9 98f54449 ' + + '579b4468 17afbd17 273e662c 97ee7299 5ef42640 c550b901 ' + + '3fad0761 353c7086 a272c240 88be9476 9fd16650', + ], +}); + +defineCurve('curve25519', { + type: 'mont', + prime: 'p25519', + p: '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed', + a: '76d06', + b: '1', + n: '1000000000000000 0000000000000000 14def9dea2f79cd6 5812631a5cf5d3ed', + hash: hash.sha256, + gRed: false, + g: [ + '9', + ], +}); + +defineCurve('ed25519', { + type: 'edwards', + prime: 'p25519', + p: '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed', + a: '-1', + c: '1', + // -121665 * (121666^(-1)) (mod P) + d: '52036cee2b6ffe73 8cc740797779e898 00700a4d4141d8ab 75eb4dca135978a3', + n: '1000000000000000 0000000000000000 14def9dea2f79cd6 5812631a5cf5d3ed', + hash: hash.sha256, + gRed: false, + g: [ + '216936d3cd6e53fec0a4e231fdd6dc5c692cc7609525a7b2c9562d608f25d51a', + + // 4/5 + '6666666666666666666666666666666666666666666666666666666666666658', + ], +}); + +var pre; +try { + pre = require('./precomputed/secp256k1'); +} catch (e) { + pre = undefined; +} + +defineCurve('secp256k1', { + type: 'short', + prime: 'k256', + p: 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff fffffffe fffffc2f', + a: '0', + b: '7', + n: 'ffffffff ffffffff ffffffff fffffffe baaedce6 af48a03b bfd25e8c d0364141', + h: '1', + hash: hash.sha256, + + // Precomputed endomorphism + beta: '7ae96a2b657c07106e64479eac3434e99cf0497512f58995c1396c28719501ee', + lambda: '5363ad4cc05c30e0a5261c028812645a122e22ea20816678df02967c1b23bd72', + basis: [ + { + a: '3086d221a7d46bcde86c90e49284eb15', + b: '-e4437ed6010e88286f547fa90abfe4c3', + }, + { + a: '114ca50f7a8e2f3f657c1108d9d44cfd8', + b: '3086d221a7d46bcde86c90e49284eb15', + }, + ], + + gRed: false, + g: [ + '79be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798', + '483ada7726a3c4655da4fbfc0e1108a8fd17b448a68554199c47d08ffb10d4b8', + pre, + ], +}); diff --git a/node_modules/elliptic/lib/elliptic/ec/index.js b/node_modules/elliptic/lib/elliptic/ec/index.js new file mode 100644 index 0000000..8b58781 --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/ec/index.js @@ -0,0 +1,243 @@ +'use strict'; + +var BN = require('bn.js'); +var HmacDRBG = require('hmac-drbg'); +var utils = require('../utils'); +var curves = require('../curves'); +var rand = require('brorand'); +var assert = utils.assert; + +var KeyPair = require('./key'); +var Signature = require('./signature'); + +function EC(options) { + if (!(this instanceof EC)) + return new EC(options); + + // Shortcut `elliptic.ec(curve-name)` + if (typeof options === 'string') { + assert(Object.prototype.hasOwnProperty.call(curves, options), + 'Unknown curve ' + options); + + options = curves[options]; + } + + // Shortcut for `elliptic.ec(elliptic.curves.curveName)` + if (options instanceof curves.PresetCurve) + options = { curve: options }; + + this.curve = options.curve.curve; + this.n = this.curve.n; + this.nh = this.n.ushrn(1); + this.g = this.curve.g; + + // Point on curve + this.g = options.curve.g; + this.g.precompute(options.curve.n.bitLength() + 1); + + // Hash for function for DRBG + this.hash = options.hash || options.curve.hash; +} +module.exports = EC; + +EC.prototype.keyPair = function keyPair(options) { + return new KeyPair(this, options); +}; + +EC.prototype.keyFromPrivate = function keyFromPrivate(priv, enc) { + return KeyPair.fromPrivate(this, priv, enc); +}; + +EC.prototype.keyFromPublic = function keyFromPublic(pub, enc) { + return KeyPair.fromPublic(this, pub, enc); +}; + +EC.prototype.genKeyPair = function genKeyPair(options) { + if (!options) + options = {}; + + // Instantiate Hmac_DRBG + var drbg = new HmacDRBG({ + hash: this.hash, + pers: options.pers, + persEnc: options.persEnc || 'utf8', + entropy: options.entropy || rand(this.hash.hmacStrength), + entropyEnc: options.entropy && options.entropyEnc || 'utf8', + nonce: this.n.toArray(), + }); + + var bytes = this.n.byteLength(); + var ns2 = this.n.sub(new BN(2)); + for (;;) { + var priv = new BN(drbg.generate(bytes)); + if (priv.cmp(ns2) > 0) + continue; + + priv.iaddn(1); + return this.keyFromPrivate(priv); + } +}; + +EC.prototype._truncateToN = function _truncateToN(msg, truncOnly) { + var delta = msg.byteLength() * 8 - this.n.bitLength(); + if (delta > 0) + msg = msg.ushrn(delta); + if (!truncOnly && msg.cmp(this.n) >= 0) + return msg.sub(this.n); + else + return msg; +}; + +EC.prototype.sign = function sign(msg, key, enc, options) { + if (typeof enc === 'object') { + options = enc; + enc = null; + } + if (!options) + options = {}; + + key = this.keyFromPrivate(key, enc); + msg = this._truncateToN(new BN(msg, 16)); + + // Zero-extend key to provide enough entropy + var bytes = this.n.byteLength(); + var bkey = key.getPrivate().toArray('be', bytes); + + // Zero-extend nonce to have the same byte size as N + var nonce = msg.toArray('be', bytes); + + // Instantiate Hmac_DRBG + var drbg = new HmacDRBG({ + hash: this.hash, + entropy: bkey, + nonce: nonce, + pers: options.pers, + persEnc: options.persEnc || 'utf8', + }); + + // Number of bytes to generate + var ns1 = this.n.sub(new BN(1)); + + for (var iter = 0; ; iter++) { + var k = options.k ? + options.k(iter) : + new BN(drbg.generate(this.n.byteLength())); + k = this._truncateToN(k, true); + if (k.cmpn(1) <= 0 || k.cmp(ns1) >= 0) + continue; + + var kp = this.g.mul(k); + if (kp.isInfinity()) + continue; + + var kpX = kp.getX(); + var r = kpX.umod(this.n); + if (r.cmpn(0) === 0) + continue; + + var s = k.invm(this.n).mul(r.mul(key.getPrivate()).iadd(msg)); + s = s.umod(this.n); + if (s.cmpn(0) === 0) + continue; + + var recoveryParam = (kp.getY().isOdd() ? 1 : 0) | + (kpX.cmp(r) !== 0 ? 2 : 0); + + // Use complement of `s`, if it is > `n / 2` + if (options.canonical && s.cmp(this.nh) > 0) { + s = this.n.sub(s); + recoveryParam ^= 1; + } + + return new Signature({ r: r, s: s, recoveryParam: recoveryParam }); + } +}; + +EC.prototype.verify = function verify(msg, signature, key, enc) { + msg = this._truncateToN(new BN(msg, 16)); + key = this.keyFromPublic(key, enc); + signature = new Signature(signature, 'hex'); + + // Perform primitive values validation + var r = signature.r; + var s = signature.s; + if (r.cmpn(1) < 0 || r.cmp(this.n) >= 0) + return false; + if (s.cmpn(1) < 0 || s.cmp(this.n) >= 0) + return false; + + // Validate signature + var sinv = s.invm(this.n); + var u1 = sinv.mul(msg).umod(this.n); + var u2 = sinv.mul(r).umod(this.n); + var p; + + if (!this.curve._maxwellTrick) { + p = this.g.mulAdd(u1, key.getPublic(), u2); + if (p.isInfinity()) + return false; + + return p.getX().umod(this.n).cmp(r) === 0; + } + + // NOTE: Greg Maxwell's trick, inspired by: + // https://git.io/vad3K + + p = this.g.jmulAdd(u1, key.getPublic(), u2); + if (p.isInfinity()) + return false; + + // Compare `p.x` of Jacobian point with `r`, + // this will do `p.x == r * p.z^2` instead of multiplying `p.x` by the + // inverse of `p.z^2` + return p.eqXToP(r); +}; + +EC.prototype.recoverPubKey = function(msg, signature, j, enc) { + assert((3 & j) === j, 'The recovery param is more than two bits'); + signature = new Signature(signature, enc); + + var n = this.n; + var e = new BN(msg); + var r = signature.r; + var s = signature.s; + + // A set LSB signifies that the y-coordinate is odd + var isYOdd = j & 1; + var isSecondKey = j >> 1; + if (r.cmp(this.curve.p.umod(this.curve.n)) >= 0 && isSecondKey) + throw new Error('Unable to find sencond key candinate'); + + // 1.1. Let x = r + jn. + if (isSecondKey) + r = this.curve.pointFromX(r.add(this.curve.n), isYOdd); + else + r = this.curve.pointFromX(r, isYOdd); + + var rInv = signature.r.invm(n); + var s1 = n.sub(e).mul(rInv).umod(n); + var s2 = s.mul(rInv).umod(n); + + // 1.6.1 Compute Q = r^-1 (sR - eG) + // Q = r^-1 (sR + -eG) + return this.g.mulAdd(s1, r, s2); +}; + +EC.prototype.getKeyRecoveryParam = function(e, signature, Q, enc) { + signature = new Signature(signature, enc); + if (signature.recoveryParam !== null) + return signature.recoveryParam; + + for (var i = 0; i < 4; i++) { + var Qprime; + try { + Qprime = this.recoverPubKey(e, signature, i); + } catch (e) { + continue; + } + + if (Qprime.eq(Q)) + return i; + } + throw new Error('Unable to find valid recovery factor'); +}; diff --git a/node_modules/elliptic/lib/elliptic/ec/key.js b/node_modules/elliptic/lib/elliptic/ec/key.js new file mode 100644 index 0000000..55bf299 --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/ec/key.js @@ -0,0 +1,121 @@ +'use strict'; + +var BN = require('bn.js'); +var utils = require('../utils'); +var assert = utils.assert; + +function KeyPair(ec, options) { + this.ec = ec; + this.priv = null; + this.pub = null; + + // KeyPair(ec, { priv: ..., pub: ... }) + if (options.priv) + this._importPrivate(options.priv, options.privEnc); + if (options.pub) + this._importPublic(options.pub, options.pubEnc); +} +module.exports = KeyPair; + +KeyPair.fromPublic = function fromPublic(ec, pub, enc) { + if (pub instanceof KeyPair) + return pub; + + return new KeyPair(ec, { + pub: pub, + pubEnc: enc, + }); +}; + +KeyPair.fromPrivate = function fromPrivate(ec, priv, enc) { + if (priv instanceof KeyPair) + return priv; + + return new KeyPair(ec, { + priv: priv, + privEnc: enc, + }); +}; + +KeyPair.prototype.validate = function validate() { + var pub = this.getPublic(); + + if (pub.isInfinity()) + return { result: false, reason: 'Invalid public key' }; + if (!pub.validate()) + return { result: false, reason: 'Public key is not a point' }; + if (!pub.mul(this.ec.curve.n).isInfinity()) + return { result: false, reason: 'Public key * N != O' }; + + return { result: true, reason: null }; +}; + +KeyPair.prototype.getPublic = function getPublic(compact, enc) { + // compact is optional argument + if (typeof compact === 'string') { + enc = compact; + compact = null; + } + + if (!this.pub) + this.pub = this.ec.g.mul(this.priv); + + if (!enc) + return this.pub; + + return this.pub.encode(enc, compact); +}; + +KeyPair.prototype.getPrivate = function getPrivate(enc) { + if (enc === 'hex') + return this.priv.toString(16, 2); + else + return this.priv; +}; + +KeyPair.prototype._importPrivate = function _importPrivate(key, enc) { + this.priv = new BN(key, enc || 16); + + // Ensure that the priv won't be bigger than n, otherwise we may fail + // in fixed multiplication method + this.priv = this.priv.umod(this.ec.curve.n); +}; + +KeyPair.prototype._importPublic = function _importPublic(key, enc) { + if (key.x || key.y) { + // Montgomery points only have an `x` coordinate. + // Weierstrass/Edwards points on the other hand have both `x` and + // `y` coordinates. + if (this.ec.curve.type === 'mont') { + assert(key.x, 'Need x coordinate'); + } else if (this.ec.curve.type === 'short' || + this.ec.curve.type === 'edwards') { + assert(key.x && key.y, 'Need both x and y coordinate'); + } + this.pub = this.ec.curve.point(key.x, key.y); + return; + } + this.pub = this.ec.curve.decodePoint(key, enc); +}; + +// ECDH +KeyPair.prototype.derive = function derive(pub) { + if(!pub.validate()) { + assert(pub.validate(), 'public point not validated'); + } + return pub.mul(this.priv).getX(); +}; + +// ECDSA +KeyPair.prototype.sign = function sign(msg, enc, options) { + return this.ec.sign(msg, this, enc, options); +}; + +KeyPair.prototype.verify = function verify(msg, signature) { + return this.ec.verify(msg, signature, this); +}; + +KeyPair.prototype.inspect = function inspect() { + return ''; +}; diff --git a/node_modules/elliptic/lib/elliptic/ec/signature.js b/node_modules/elliptic/lib/elliptic/ec/signature.js new file mode 100644 index 0000000..539df6a --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/ec/signature.js @@ -0,0 +1,166 @@ +'use strict'; + +var BN = require('bn.js'); + +var utils = require('../utils'); +var assert = utils.assert; + +function Signature(options, enc) { + if (options instanceof Signature) + return options; + + if (this._importDER(options, enc)) + return; + + assert(options.r && options.s, 'Signature without r or s'); + this.r = new BN(options.r, 16); + this.s = new BN(options.s, 16); + if (options.recoveryParam === undefined) + this.recoveryParam = null; + else + this.recoveryParam = options.recoveryParam; +} +module.exports = Signature; + +function Position() { + this.place = 0; +} + +function getLength(buf, p) { + var initial = buf[p.place++]; + if (!(initial & 0x80)) { + return initial; + } + var octetLen = initial & 0xf; + + // Indefinite length or overflow + if (octetLen === 0 || octetLen > 4) { + return false; + } + + var val = 0; + for (var i = 0, off = p.place; i < octetLen; i++, off++) { + val <<= 8; + val |= buf[off]; + val >>>= 0; + } + + // Leading zeroes + if (val <= 0x7f) { + return false; + } + + p.place = off; + return val; +} + +function rmPadding(buf) { + var i = 0; + var len = buf.length - 1; + while (!buf[i] && !(buf[i + 1] & 0x80) && i < len) { + i++; + } + if (i === 0) { + return buf; + } + return buf.slice(i); +} + +Signature.prototype._importDER = function _importDER(data, enc) { + data = utils.toArray(data, enc); + var p = new Position(); + if (data[p.place++] !== 0x30) { + return false; + } + var len = getLength(data, p); + if (len === false) { + return false; + } + if ((len + p.place) !== data.length) { + return false; + } + if (data[p.place++] !== 0x02) { + return false; + } + var rlen = getLength(data, p); + if (rlen === false) { + return false; + } + var r = data.slice(p.place, rlen + p.place); + p.place += rlen; + if (data[p.place++] !== 0x02) { + return false; + } + var slen = getLength(data, p); + if (slen === false) { + return false; + } + if (data.length !== slen + p.place) { + return false; + } + var s = data.slice(p.place, slen + p.place); + if (r[0] === 0) { + if (r[1] & 0x80) { + r = r.slice(1); + } else { + // Leading zeroes + return false; + } + } + if (s[0] === 0) { + if (s[1] & 0x80) { + s = s.slice(1); + } else { + // Leading zeroes + return false; + } + } + + this.r = new BN(r); + this.s = new BN(s); + this.recoveryParam = null; + + return true; +}; + +function constructLength(arr, len) { + if (len < 0x80) { + arr.push(len); + return; + } + var octets = 1 + (Math.log(len) / Math.LN2 >>> 3); + arr.push(octets | 0x80); + while (--octets) { + arr.push((len >>> (octets << 3)) & 0xff); + } + arr.push(len); +} + +Signature.prototype.toDER = function toDER(enc) { + var r = this.r.toArray(); + var s = this.s.toArray(); + + // Pad values + if (r[0] & 0x80) + r = [ 0 ].concat(r); + // Pad values + if (s[0] & 0x80) + s = [ 0 ].concat(s); + + r = rmPadding(r); + s = rmPadding(s); + + while (!s[0] && !(s[1] & 0x80)) { + s = s.slice(1); + } + var arr = [ 0x02 ]; + constructLength(arr, r.length); + arr = arr.concat(r); + arr.push(0x02); + constructLength(arr, s.length); + var backHalf = arr.concat(s); + var res = [ 0x30 ]; + constructLength(res, backHalf.length); + res = res.concat(backHalf); + return utils.encode(res, enc); +}; diff --git a/node_modules/elliptic/lib/elliptic/eddsa/index.js b/node_modules/elliptic/lib/elliptic/eddsa/index.js new file mode 100644 index 0000000..d777983 --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/eddsa/index.js @@ -0,0 +1,118 @@ +'use strict'; + +var hash = require('hash.js'); +var curves = require('../curves'); +var utils = require('../utils'); +var assert = utils.assert; +var parseBytes = utils.parseBytes; +var KeyPair = require('./key'); +var Signature = require('./signature'); + +function EDDSA(curve) { + assert(curve === 'ed25519', 'only tested with ed25519 so far'); + + if (!(this instanceof EDDSA)) + return new EDDSA(curve); + + curve = curves[curve].curve; + this.curve = curve; + this.g = curve.g; + this.g.precompute(curve.n.bitLength() + 1); + + this.pointClass = curve.point().constructor; + this.encodingLength = Math.ceil(curve.n.bitLength() / 8); + this.hash = hash.sha512; +} + +module.exports = EDDSA; + +/** +* @param {Array|String} message - message bytes +* @param {Array|String|KeyPair} secret - secret bytes or a keypair +* @returns {Signature} - signature +*/ +EDDSA.prototype.sign = function sign(message, secret) { + message = parseBytes(message); + var key = this.keyFromSecret(secret); + var r = this.hashInt(key.messagePrefix(), message); + var R = this.g.mul(r); + var Rencoded = this.encodePoint(R); + var s_ = this.hashInt(Rencoded, key.pubBytes(), message) + .mul(key.priv()); + var S = r.add(s_).umod(this.curve.n); + return this.makeSignature({ R: R, S: S, Rencoded: Rencoded }); +}; + +/** +* @param {Array} message - message bytes +* @param {Array|String|Signature} sig - sig bytes +* @param {Array|String|Point|KeyPair} pub - public key +* @returns {Boolean} - true if public key matches sig of message +*/ +EDDSA.prototype.verify = function verify(message, sig, pub) { + message = parseBytes(message); + sig = this.makeSignature(sig); + var key = this.keyFromPublic(pub); + var h = this.hashInt(sig.Rencoded(), key.pubBytes(), message); + var SG = this.g.mul(sig.S()); + var RplusAh = sig.R().add(key.pub().mul(h)); + return RplusAh.eq(SG); +}; + +EDDSA.prototype.hashInt = function hashInt() { + var hash = this.hash(); + for (var i = 0; i < arguments.length; i++) + hash.update(arguments[i]); + return utils.intFromLE(hash.digest()).umod(this.curve.n); +}; + +EDDSA.prototype.keyFromPublic = function keyFromPublic(pub) { + return KeyPair.fromPublic(this, pub); +}; + +EDDSA.prototype.keyFromSecret = function keyFromSecret(secret) { + return KeyPair.fromSecret(this, secret); +}; + +EDDSA.prototype.makeSignature = function makeSignature(sig) { + if (sig instanceof Signature) + return sig; + return new Signature(this, sig); +}; + +/** +* * https://tools.ietf.org/html/draft-josefsson-eddsa-ed25519-03#section-5.2 +* +* EDDSA defines methods for encoding and decoding points and integers. These are +* helper convenience methods, that pass along to utility functions implied +* parameters. +* +*/ +EDDSA.prototype.encodePoint = function encodePoint(point) { + var enc = point.getY().toArray('le', this.encodingLength); + enc[this.encodingLength - 1] |= point.getX().isOdd() ? 0x80 : 0; + return enc; +}; + +EDDSA.prototype.decodePoint = function decodePoint(bytes) { + bytes = utils.parseBytes(bytes); + + var lastIx = bytes.length - 1; + var normed = bytes.slice(0, lastIx).concat(bytes[lastIx] & ~0x80); + var xIsOdd = (bytes[lastIx] & 0x80) !== 0; + + var y = utils.intFromLE(normed); + return this.curve.pointFromY(y, xIsOdd); +}; + +EDDSA.prototype.encodeInt = function encodeInt(num) { + return num.toArray('le', this.encodingLength); +}; + +EDDSA.prototype.decodeInt = function decodeInt(bytes) { + return utils.intFromLE(bytes); +}; + +EDDSA.prototype.isPoint = function isPoint(val) { + return val instanceof this.pointClass; +}; diff --git a/node_modules/elliptic/lib/elliptic/eddsa/key.js b/node_modules/elliptic/lib/elliptic/eddsa/key.js new file mode 100644 index 0000000..a00028f --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/eddsa/key.js @@ -0,0 +1,95 @@ +'use strict'; + +var utils = require('../utils'); +var assert = utils.assert; +var parseBytes = utils.parseBytes; +var cachedProperty = utils.cachedProperty; + +/** +* @param {EDDSA} eddsa - instance +* @param {Object} params - public/private key parameters +* +* @param {Array} [params.secret] - secret seed bytes +* @param {Point} [params.pub] - public key point (aka `A` in eddsa terms) +* @param {Array} [params.pub] - public key point encoded as bytes +* +*/ +function KeyPair(eddsa, params) { + this.eddsa = eddsa; + this._secret = parseBytes(params.secret); + if (eddsa.isPoint(params.pub)) + this._pub = params.pub; + else + this._pubBytes = parseBytes(params.pub); +} + +KeyPair.fromPublic = function fromPublic(eddsa, pub) { + if (pub instanceof KeyPair) + return pub; + return new KeyPair(eddsa, { pub: pub }); +}; + +KeyPair.fromSecret = function fromSecret(eddsa, secret) { + if (secret instanceof KeyPair) + return secret; + return new KeyPair(eddsa, { secret: secret }); +}; + +KeyPair.prototype.secret = function secret() { + return this._secret; +}; + +cachedProperty(KeyPair, 'pubBytes', function pubBytes() { + return this.eddsa.encodePoint(this.pub()); +}); + +cachedProperty(KeyPair, 'pub', function pub() { + if (this._pubBytes) + return this.eddsa.decodePoint(this._pubBytes); + return this.eddsa.g.mul(this.priv()); +}); + +cachedProperty(KeyPair, 'privBytes', function privBytes() { + var eddsa = this.eddsa; + var hash = this.hash(); + var lastIx = eddsa.encodingLength - 1; + + var a = hash.slice(0, eddsa.encodingLength); + a[0] &= 248; + a[lastIx] &= 127; + a[lastIx] |= 64; + + return a; +}); + +cachedProperty(KeyPair, 'priv', function priv() { + return this.eddsa.decodeInt(this.privBytes()); +}); + +cachedProperty(KeyPair, 'hash', function hash() { + return this.eddsa.hash().update(this.secret()).digest(); +}); + +cachedProperty(KeyPair, 'messagePrefix', function messagePrefix() { + return this.hash().slice(this.eddsa.encodingLength); +}); + +KeyPair.prototype.sign = function sign(message) { + assert(this._secret, 'KeyPair can only verify'); + return this.eddsa.sign(message, this); +}; + +KeyPair.prototype.verify = function verify(message, sig) { + return this.eddsa.verify(message, sig, this); +}; + +KeyPair.prototype.getSecret = function getSecret(enc) { + assert(this._secret, 'KeyPair is public only'); + return utils.encode(this.secret(), enc); +}; + +KeyPair.prototype.getPublic = function getPublic(enc) { + return utils.encode(this.pubBytes(), enc); +}; + +module.exports = KeyPair; diff --git a/node_modules/elliptic/lib/elliptic/eddsa/signature.js b/node_modules/elliptic/lib/elliptic/eddsa/signature.js new file mode 100644 index 0000000..30ebc92 --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/eddsa/signature.js @@ -0,0 +1,65 @@ +'use strict'; + +var BN = require('bn.js'); +var utils = require('../utils'); +var assert = utils.assert; +var cachedProperty = utils.cachedProperty; +var parseBytes = utils.parseBytes; + +/** +* @param {EDDSA} eddsa - eddsa instance +* @param {Array|Object} sig - +* @param {Array|Point} [sig.R] - R point as Point or bytes +* @param {Array|bn} [sig.S] - S scalar as bn or bytes +* @param {Array} [sig.Rencoded] - R point encoded +* @param {Array} [sig.Sencoded] - S scalar encoded +*/ +function Signature(eddsa, sig) { + this.eddsa = eddsa; + + if (typeof sig !== 'object') + sig = parseBytes(sig); + + if (Array.isArray(sig)) { + sig = { + R: sig.slice(0, eddsa.encodingLength), + S: sig.slice(eddsa.encodingLength), + }; + } + + assert(sig.R && sig.S, 'Signature without R or S'); + + if (eddsa.isPoint(sig.R)) + this._R = sig.R; + if (sig.S instanceof BN) + this._S = sig.S; + + this._Rencoded = Array.isArray(sig.R) ? sig.R : sig.Rencoded; + this._Sencoded = Array.isArray(sig.S) ? sig.S : sig.Sencoded; +} + +cachedProperty(Signature, 'S', function S() { + return this.eddsa.decodeInt(this.Sencoded()); +}); + +cachedProperty(Signature, 'R', function R() { + return this.eddsa.decodePoint(this.Rencoded()); +}); + +cachedProperty(Signature, 'Rencoded', function Rencoded() { + return this.eddsa.encodePoint(this.R()); +}); + +cachedProperty(Signature, 'Sencoded', function Sencoded() { + return this.eddsa.encodeInt(this.S()); +}); + +Signature.prototype.toBytes = function toBytes() { + return this.Rencoded().concat(this.Sencoded()); +}; + +Signature.prototype.toHex = function toHex() { + return utils.encode(this.toBytes(), 'hex').toUpperCase(); +}; + +module.exports = Signature; diff --git a/node_modules/elliptic/lib/elliptic/precomputed/secp256k1.js b/node_modules/elliptic/lib/elliptic/precomputed/secp256k1.js new file mode 100644 index 0000000..01a7c4d --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/precomputed/secp256k1.js @@ -0,0 +1,780 @@ +module.exports = { + doubles: { + step: 4, + points: [ + [ + 'e60fce93b59e9ec53011aabc21c23e97b2a31369b87a5ae9c44ee89e2a6dec0a', + 'f7e3507399e595929db99f34f57937101296891e44d23f0be1f32cce69616821', + ], + [ + '8282263212c609d9ea2a6e3e172de238d8c39cabd5ac1ca10646e23fd5f51508', + '11f8a8098557dfe45e8256e830b60ace62d613ac2f7b17bed31b6eaff6e26caf', + ], + [ + '175e159f728b865a72f99cc6c6fc846de0b93833fd2222ed73fce5b551e5b739', + 'd3506e0d9e3c79eba4ef97a51ff71f5eacb5955add24345c6efa6ffee9fed695', + ], + [ + '363d90d447b00c9c99ceac05b6262ee053441c7e55552ffe526bad8f83ff4640', + '4e273adfc732221953b445397f3363145b9a89008199ecb62003c7f3bee9de9', + ], + [ + '8b4b5f165df3c2be8c6244b5b745638843e4a781a15bcd1b69f79a55dffdf80c', + '4aad0a6f68d308b4b3fbd7813ab0da04f9e336546162ee56b3eff0c65fd4fd36', + ], + [ + '723cbaa6e5db996d6bf771c00bd548c7b700dbffa6c0e77bcb6115925232fcda', + '96e867b5595cc498a921137488824d6e2660a0653779494801dc069d9eb39f5f', + ], + [ + 'eebfa4d493bebf98ba5feec812c2d3b50947961237a919839a533eca0e7dd7fa', + '5d9a8ca3970ef0f269ee7edaf178089d9ae4cdc3a711f712ddfd4fdae1de8999', + ], + [ + '100f44da696e71672791d0a09b7bde459f1215a29b3c03bfefd7835b39a48db0', + 'cdd9e13192a00b772ec8f3300c090666b7ff4a18ff5195ac0fbd5cd62bc65a09', + ], + [ + 'e1031be262c7ed1b1dc9227a4a04c017a77f8d4464f3b3852c8acde6e534fd2d', + '9d7061928940405e6bb6a4176597535af292dd419e1ced79a44f18f29456a00d', + ], + [ + 'feea6cae46d55b530ac2839f143bd7ec5cf8b266a41d6af52d5e688d9094696d', + 'e57c6b6c97dce1bab06e4e12bf3ecd5c981c8957cc41442d3155debf18090088', + ], + [ + 'da67a91d91049cdcb367be4be6ffca3cfeed657d808583de33fa978bc1ec6cb1', + '9bacaa35481642bc41f463f7ec9780e5dec7adc508f740a17e9ea8e27a68be1d', + ], + [ + '53904faa0b334cdda6e000935ef22151ec08d0f7bb11069f57545ccc1a37b7c0', + '5bc087d0bc80106d88c9eccac20d3c1c13999981e14434699dcb096b022771c8', + ], + [ + '8e7bcd0bd35983a7719cca7764ca906779b53a043a9b8bcaeff959f43ad86047', + '10b7770b2a3da4b3940310420ca9514579e88e2e47fd68b3ea10047e8460372a', + ], + [ + '385eed34c1cdff21e6d0818689b81bde71a7f4f18397e6690a841e1599c43862', + '283bebc3e8ea23f56701de19e9ebf4576b304eec2086dc8cc0458fe5542e5453', + ], + [ + '6f9d9b803ecf191637c73a4413dfa180fddf84a5947fbc9c606ed86c3fac3a7', + '7c80c68e603059ba69b8e2a30e45c4d47ea4dd2f5c281002d86890603a842160', + ], + [ + '3322d401243c4e2582a2147c104d6ecbf774d163db0f5e5313b7e0e742d0e6bd', + '56e70797e9664ef5bfb019bc4ddaf9b72805f63ea2873af624f3a2e96c28b2a0', + ], + [ + '85672c7d2de0b7da2bd1770d89665868741b3f9af7643397721d74d28134ab83', + '7c481b9b5b43b2eb6374049bfa62c2e5e77f17fcc5298f44c8e3094f790313a6', + ], + [ + '948bf809b1988a46b06c9f1919413b10f9226c60f668832ffd959af60c82a0a', + '53a562856dcb6646dc6b74c5d1c3418c6d4dff08c97cd2bed4cb7f88d8c8e589', + ], + [ + '6260ce7f461801c34f067ce0f02873a8f1b0e44dfc69752accecd819f38fd8e8', + 'bc2da82b6fa5b571a7f09049776a1ef7ecd292238051c198c1a84e95b2b4ae17', + ], + [ + 'e5037de0afc1d8d43d8348414bbf4103043ec8f575bfdc432953cc8d2037fa2d', + '4571534baa94d3b5f9f98d09fb990bddbd5f5b03ec481f10e0e5dc841d755bda', + ], + [ + 'e06372b0f4a207adf5ea905e8f1771b4e7e8dbd1c6a6c5b725866a0ae4fce725', + '7a908974bce18cfe12a27bb2ad5a488cd7484a7787104870b27034f94eee31dd', + ], + [ + '213c7a715cd5d45358d0bbf9dc0ce02204b10bdde2a3f58540ad6908d0559754', + '4b6dad0b5ae462507013ad06245ba190bb4850f5f36a7eeddff2c27534b458f2', + ], + [ + '4e7c272a7af4b34e8dbb9352a5419a87e2838c70adc62cddf0cc3a3b08fbd53c', + '17749c766c9d0b18e16fd09f6def681b530b9614bff7dd33e0b3941817dcaae6', + ], + [ + 'fea74e3dbe778b1b10f238ad61686aa5c76e3db2be43057632427e2840fb27b6', + '6e0568db9b0b13297cf674deccb6af93126b596b973f7b77701d3db7f23cb96f', + ], + [ + '76e64113f677cf0e10a2570d599968d31544e179b760432952c02a4417bdde39', + 'c90ddf8dee4e95cf577066d70681f0d35e2a33d2b56d2032b4b1752d1901ac01', + ], + [ + 'c738c56b03b2abe1e8281baa743f8f9a8f7cc643df26cbee3ab150242bcbb891', + '893fb578951ad2537f718f2eacbfbbbb82314eef7880cfe917e735d9699a84c3', + ], + [ + 'd895626548b65b81e264c7637c972877d1d72e5f3a925014372e9f6588f6c14b', + 'febfaa38f2bc7eae728ec60818c340eb03428d632bb067e179363ed75d7d991f', + ], + [ + 'b8da94032a957518eb0f6433571e8761ceffc73693e84edd49150a564f676e03', + '2804dfa44805a1e4d7c99cc9762808b092cc584d95ff3b511488e4e74efdf6e7', + ], + [ + 'e80fea14441fb33a7d8adab9475d7fab2019effb5156a792f1a11778e3c0df5d', + 'eed1de7f638e00771e89768ca3ca94472d155e80af322ea9fcb4291b6ac9ec78', + ], + [ + 'a301697bdfcd704313ba48e51d567543f2a182031efd6915ddc07bbcc4e16070', + '7370f91cfb67e4f5081809fa25d40f9b1735dbf7c0a11a130c0d1a041e177ea1', + ], + [ + '90ad85b389d6b936463f9d0512678de208cc330b11307fffab7ac63e3fb04ed4', + 'e507a3620a38261affdcbd9427222b839aefabe1582894d991d4d48cb6ef150', + ], + [ + '8f68b9d2f63b5f339239c1ad981f162ee88c5678723ea3351b7b444c9ec4c0da', + '662a9f2dba063986de1d90c2b6be215dbbea2cfe95510bfdf23cbf79501fff82', + ], + [ + 'e4f3fb0176af85d65ff99ff9198c36091f48e86503681e3e6686fd5053231e11', + '1e63633ad0ef4f1c1661a6d0ea02b7286cc7e74ec951d1c9822c38576feb73bc', + ], + [ + '8c00fa9b18ebf331eb961537a45a4266c7034f2f0d4e1d0716fb6eae20eae29e', + 'efa47267fea521a1a9dc343a3736c974c2fadafa81e36c54e7d2a4c66702414b', + ], + [ + 'e7a26ce69dd4829f3e10cec0a9e98ed3143d084f308b92c0997fddfc60cb3e41', + '2a758e300fa7984b471b006a1aafbb18d0a6b2c0420e83e20e8a9421cf2cfd51', + ], + [ + 'b6459e0ee3662ec8d23540c223bcbdc571cbcb967d79424f3cf29eb3de6b80ef', + '67c876d06f3e06de1dadf16e5661db3c4b3ae6d48e35b2ff30bf0b61a71ba45', + ], + [ + 'd68a80c8280bb840793234aa118f06231d6f1fc67e73c5a5deda0f5b496943e8', + 'db8ba9fff4b586d00c4b1f9177b0e28b5b0e7b8f7845295a294c84266b133120', + ], + [ + '324aed7df65c804252dc0270907a30b09612aeb973449cea4095980fc28d3d5d', + '648a365774b61f2ff130c0c35aec1f4f19213b0c7e332843967224af96ab7c84', + ], + [ + '4df9c14919cde61f6d51dfdbe5fee5dceec4143ba8d1ca888e8bd373fd054c96', + '35ec51092d8728050974c23a1d85d4b5d506cdc288490192ebac06cad10d5d', + ], + [ + '9c3919a84a474870faed8a9c1cc66021523489054d7f0308cbfc99c8ac1f98cd', + 'ddb84f0f4a4ddd57584f044bf260e641905326f76c64c8e6be7e5e03d4fc599d', + ], + [ + '6057170b1dd12fdf8de05f281d8e06bb91e1493a8b91d4cc5a21382120a959e5', + '9a1af0b26a6a4807add9a2daf71df262465152bc3ee24c65e899be932385a2a8', + ], + [ + 'a576df8e23a08411421439a4518da31880cef0fba7d4df12b1a6973eecb94266', + '40a6bf20e76640b2c92b97afe58cd82c432e10a7f514d9f3ee8be11ae1b28ec8', + ], + [ + '7778a78c28dec3e30a05fe9629de8c38bb30d1f5cf9a3a208f763889be58ad71', + '34626d9ab5a5b22ff7098e12f2ff580087b38411ff24ac563b513fc1fd9f43ac', + ], + [ + '928955ee637a84463729fd30e7afd2ed5f96274e5ad7e5cb09eda9c06d903ac', + 'c25621003d3f42a827b78a13093a95eeac3d26efa8a8d83fc5180e935bcd091f', + ], + [ + '85d0fef3ec6db109399064f3a0e3b2855645b4a907ad354527aae75163d82751', + '1f03648413a38c0be29d496e582cf5663e8751e96877331582c237a24eb1f962', + ], + [ + 'ff2b0dce97eece97c1c9b6041798b85dfdfb6d8882da20308f5404824526087e', + '493d13fef524ba188af4c4dc54d07936c7b7ed6fb90e2ceb2c951e01f0c29907', + ], + [ + '827fbbe4b1e880ea9ed2b2e6301b212b57f1ee148cd6dd28780e5e2cf856e241', + 'c60f9c923c727b0b71bef2c67d1d12687ff7a63186903166d605b68baec293ec', + ], + [ + 'eaa649f21f51bdbae7be4ae34ce6e5217a58fdce7f47f9aa7f3b58fa2120e2b3', + 'be3279ed5bbbb03ac69a80f89879aa5a01a6b965f13f7e59d47a5305ba5ad93d', + ], + [ + 'e4a42d43c5cf169d9391df6decf42ee541b6d8f0c9a137401e23632dda34d24f', + '4d9f92e716d1c73526fc99ccfb8ad34ce886eedfa8d8e4f13a7f7131deba9414', + ], + [ + '1ec80fef360cbdd954160fadab352b6b92b53576a88fea4947173b9d4300bf19', + 'aeefe93756b5340d2f3a4958a7abbf5e0146e77f6295a07b671cdc1cc107cefd', + ], + [ + '146a778c04670c2f91b00af4680dfa8bce3490717d58ba889ddb5928366642be', + 'b318e0ec3354028add669827f9d4b2870aaa971d2f7e5ed1d0b297483d83efd0', + ], + [ + 'fa50c0f61d22e5f07e3acebb1aa07b128d0012209a28b9776d76a8793180eef9', + '6b84c6922397eba9b72cd2872281a68a5e683293a57a213b38cd8d7d3f4f2811', + ], + [ + 'da1d61d0ca721a11b1a5bf6b7d88e8421a288ab5d5bba5220e53d32b5f067ec2', + '8157f55a7c99306c79c0766161c91e2966a73899d279b48a655fba0f1ad836f1', + ], + [ + 'a8e282ff0c9706907215ff98e8fd416615311de0446f1e062a73b0610d064e13', + '7f97355b8db81c09abfb7f3c5b2515888b679a3e50dd6bd6cef7c73111f4cc0c', + ], + [ + '174a53b9c9a285872d39e56e6913cab15d59b1fa512508c022f382de8319497c', + 'ccc9dc37abfc9c1657b4155f2c47f9e6646b3a1d8cb9854383da13ac079afa73', + ], + [ + '959396981943785c3d3e57edf5018cdbe039e730e4918b3d884fdff09475b7ba', + '2e7e552888c331dd8ba0386a4b9cd6849c653f64c8709385e9b8abf87524f2fd', + ], + [ + 'd2a63a50ae401e56d645a1153b109a8fcca0a43d561fba2dbb51340c9d82b151', + 'e82d86fb6443fcb7565aee58b2948220a70f750af484ca52d4142174dcf89405', + ], + [ + '64587e2335471eb890ee7896d7cfdc866bacbdbd3839317b3436f9b45617e073', + 'd99fcdd5bf6902e2ae96dd6447c299a185b90a39133aeab358299e5e9faf6589', + ], + [ + '8481bde0e4e4d885b3a546d3e549de042f0aa6cea250e7fd358d6c86dd45e458', + '38ee7b8cba5404dd84a25bf39cecb2ca900a79c42b262e556d64b1b59779057e', + ], + [ + '13464a57a78102aa62b6979ae817f4637ffcfed3c4b1ce30bcd6303f6caf666b', + '69be159004614580ef7e433453ccb0ca48f300a81d0942e13f495a907f6ecc27', + ], + [ + 'bc4a9df5b713fe2e9aef430bcc1dc97a0cd9ccede2f28588cada3a0d2d83f366', + 'd3a81ca6e785c06383937adf4b798caa6e8a9fbfa547b16d758d666581f33c1', + ], + [ + '8c28a97bf8298bc0d23d8c749452a32e694b65e30a9472a3954ab30fe5324caa', + '40a30463a3305193378fedf31f7cc0eb7ae784f0451cb9459e71dc73cbef9482', + ], + [ + '8ea9666139527a8c1dd94ce4f071fd23c8b350c5a4bb33748c4ba111faccae0', + '620efabbc8ee2782e24e7c0cfb95c5d735b783be9cf0f8e955af34a30e62b945', + ], + [ + 'dd3625faef5ba06074669716bbd3788d89bdde815959968092f76cc4eb9a9787', + '7a188fa3520e30d461da2501045731ca941461982883395937f68d00c644a573', + ], + [ + 'f710d79d9eb962297e4f6232b40e8f7feb2bc63814614d692c12de752408221e', + 'ea98e67232d3b3295d3b535532115ccac8612c721851617526ae47a9c77bfc82', + ], + ], + }, + naf: { + wnd: 7, + points: [ + [ + 'f9308a019258c31049344f85f89d5229b531c845836f99b08601f113bce036f9', + '388f7b0f632de8140fe337e62a37f3566500a99934c2231b6cb9fd7584b8e672', + ], + [ + '2f8bde4d1a07209355b4a7250a5c5128e88b84bddc619ab7cba8d569b240efe4', + 'd8ac222636e5e3d6d4dba9dda6c9c426f788271bab0d6840dca87d3aa6ac62d6', + ], + [ + '5cbdf0646e5db4eaa398f365f2ea7a0e3d419b7e0330e39ce92bddedcac4f9bc', + '6aebca40ba255960a3178d6d861a54dba813d0b813fde7b5a5082628087264da', + ], + [ + 'acd484e2f0c7f65309ad178a9f559abde09796974c57e714c35f110dfc27ccbe', + 'cc338921b0a7d9fd64380971763b61e9add888a4375f8e0f05cc262ac64f9c37', + ], + [ + '774ae7f858a9411e5ef4246b70c65aac5649980be5c17891bbec17895da008cb', + 'd984a032eb6b5e190243dd56d7b7b365372db1e2dff9d6a8301d74c9c953c61b', + ], + [ + 'f28773c2d975288bc7d1d205c3748651b075fbc6610e58cddeeddf8f19405aa8', + 'ab0902e8d880a89758212eb65cdaf473a1a06da521fa91f29b5cb52db03ed81', + ], + [ + 'd7924d4f7d43ea965a465ae3095ff41131e5946f3c85f79e44adbcf8e27e080e', + '581e2872a86c72a683842ec228cc6defea40af2bd896d3a5c504dc9ff6a26b58', + ], + [ + 'defdea4cdb677750a420fee807eacf21eb9898ae79b9768766e4faa04a2d4a34', + '4211ab0694635168e997b0ead2a93daeced1f4a04a95c0f6cfb199f69e56eb77', + ], + [ + '2b4ea0a797a443d293ef5cff444f4979f06acfebd7e86d277475656138385b6c', + '85e89bc037945d93b343083b5a1c86131a01f60c50269763b570c854e5c09b7a', + ], + [ + '352bbf4a4cdd12564f93fa332ce333301d9ad40271f8107181340aef25be59d5', + '321eb4075348f534d59c18259dda3e1f4a1b3b2e71b1039c67bd3d8bcf81998c', + ], + [ + '2fa2104d6b38d11b0230010559879124e42ab8dfeff5ff29dc9cdadd4ecacc3f', + '2de1068295dd865b64569335bd5dd80181d70ecfc882648423ba76b532b7d67', + ], + [ + '9248279b09b4d68dab21a9b066edda83263c3d84e09572e269ca0cd7f5453714', + '73016f7bf234aade5d1aa71bdea2b1ff3fc0de2a887912ffe54a32ce97cb3402', + ], + [ + 'daed4f2be3a8bf278e70132fb0beb7522f570e144bf615c07e996d443dee8729', + 'a69dce4a7d6c98e8d4a1aca87ef8d7003f83c230f3afa726ab40e52290be1c55', + ], + [ + 'c44d12c7065d812e8acf28d7cbb19f9011ecd9e9fdf281b0e6a3b5e87d22e7db', + '2119a460ce326cdc76c45926c982fdac0e106e861edf61c5a039063f0e0e6482', + ], + [ + '6a245bf6dc698504c89a20cfded60853152b695336c28063b61c65cbd269e6b4', + 'e022cf42c2bd4a708b3f5126f16a24ad8b33ba48d0423b6efd5e6348100d8a82', + ], + [ + '1697ffa6fd9de627c077e3d2fe541084ce13300b0bec1146f95ae57f0d0bd6a5', + 'b9c398f186806f5d27561506e4557433a2cf15009e498ae7adee9d63d01b2396', + ], + [ + '605bdb019981718b986d0f07e834cb0d9deb8360ffb7f61df982345ef27a7479', + '2972d2de4f8d20681a78d93ec96fe23c26bfae84fb14db43b01e1e9056b8c49', + ], + [ + '62d14dab4150bf497402fdc45a215e10dcb01c354959b10cfe31c7e9d87ff33d', + '80fc06bd8cc5b01098088a1950eed0db01aa132967ab472235f5642483b25eaf', + ], + [ + '80c60ad0040f27dade5b4b06c408e56b2c50e9f56b9b8b425e555c2f86308b6f', + '1c38303f1cc5c30f26e66bad7fe72f70a65eed4cbe7024eb1aa01f56430bd57a', + ], + [ + '7a9375ad6167ad54aa74c6348cc54d344cc5dc9487d847049d5eabb0fa03c8fb', + 'd0e3fa9eca8726909559e0d79269046bdc59ea10c70ce2b02d499ec224dc7f7', + ], + [ + 'd528ecd9b696b54c907a9ed045447a79bb408ec39b68df504bb51f459bc3ffc9', + 'eecf41253136e5f99966f21881fd656ebc4345405c520dbc063465b521409933', + ], + [ + '49370a4b5f43412ea25f514e8ecdad05266115e4a7ecb1387231808f8b45963', + '758f3f41afd6ed428b3081b0512fd62a54c3f3afbb5b6764b653052a12949c9a', + ], + [ + '77f230936ee88cbbd73df930d64702ef881d811e0e1498e2f1c13eb1fc345d74', + '958ef42a7886b6400a08266e9ba1b37896c95330d97077cbbe8eb3c7671c60d6', + ], + [ + 'f2dac991cc4ce4b9ea44887e5c7c0bce58c80074ab9d4dbaeb28531b7739f530', + 'e0dedc9b3b2f8dad4da1f32dec2531df9eb5fbeb0598e4fd1a117dba703a3c37', + ], + [ + '463b3d9f662621fb1b4be8fbbe2520125a216cdfc9dae3debcba4850c690d45b', + '5ed430d78c296c3543114306dd8622d7c622e27c970a1de31cb377b01af7307e', + ], + [ + 'f16f804244e46e2a09232d4aff3b59976b98fac14328a2d1a32496b49998f247', + 'cedabd9b82203f7e13d206fcdf4e33d92a6c53c26e5cce26d6579962c4e31df6', + ], + [ + 'caf754272dc84563b0352b7a14311af55d245315ace27c65369e15f7151d41d1', + 'cb474660ef35f5f2a41b643fa5e460575f4fa9b7962232a5c32f908318a04476', + ], + [ + '2600ca4b282cb986f85d0f1709979d8b44a09c07cb86d7c124497bc86f082120', + '4119b88753c15bd6a693b03fcddbb45d5ac6be74ab5f0ef44b0be9475a7e4b40', + ], + [ + '7635ca72d7e8432c338ec53cd12220bc01c48685e24f7dc8c602a7746998e435', + '91b649609489d613d1d5e590f78e6d74ecfc061d57048bad9e76f302c5b9c61', + ], + [ + '754e3239f325570cdbbf4a87deee8a66b7f2b33479d468fbc1a50743bf56cc18', + '673fb86e5bda30fb3cd0ed304ea49a023ee33d0197a695d0c5d98093c536683', + ], + [ + 'e3e6bd1071a1e96aff57859c82d570f0330800661d1c952f9fe2694691d9b9e8', + '59c9e0bba394e76f40c0aa58379a3cb6a5a2283993e90c4167002af4920e37f5', + ], + [ + '186b483d056a033826ae73d88f732985c4ccb1f32ba35f4b4cc47fdcf04aa6eb', + '3b952d32c67cf77e2e17446e204180ab21fb8090895138b4a4a797f86e80888b', + ], + [ + 'df9d70a6b9876ce544c98561f4be4f725442e6d2b737d9c91a8321724ce0963f', + '55eb2dafd84d6ccd5f862b785dc39d4ab157222720ef9da217b8c45cf2ba2417', + ], + [ + '5edd5cc23c51e87a497ca815d5dce0f8ab52554f849ed8995de64c5f34ce7143', + 'efae9c8dbc14130661e8cec030c89ad0c13c66c0d17a2905cdc706ab7399a868', + ], + [ + '290798c2b6476830da12fe02287e9e777aa3fba1c355b17a722d362f84614fba', + 'e38da76dcd440621988d00bcf79af25d5b29c094db2a23146d003afd41943e7a', + ], + [ + 'af3c423a95d9f5b3054754efa150ac39cd29552fe360257362dfdecef4053b45', + 'f98a3fd831eb2b749a93b0e6f35cfb40c8cd5aa667a15581bc2feded498fd9c6', + ], + [ + '766dbb24d134e745cccaa28c99bf274906bb66b26dcf98df8d2fed50d884249a', + '744b1152eacbe5e38dcc887980da38b897584a65fa06cedd2c924f97cbac5996', + ], + [ + '59dbf46f8c94759ba21277c33784f41645f7b44f6c596a58ce92e666191abe3e', + 'c534ad44175fbc300f4ea6ce648309a042ce739a7919798cd85e216c4a307f6e', + ], + [ + 'f13ada95103c4537305e691e74e9a4a8dd647e711a95e73cb62dc6018cfd87b8', + 'e13817b44ee14de663bf4bc808341f326949e21a6a75c2570778419bdaf5733d', + ], + [ + '7754b4fa0e8aced06d4167a2c59cca4cda1869c06ebadfb6488550015a88522c', + '30e93e864e669d82224b967c3020b8fa8d1e4e350b6cbcc537a48b57841163a2', + ], + [ + '948dcadf5990e048aa3874d46abef9d701858f95de8041d2a6828c99e2262519', + 'e491a42537f6e597d5d28a3224b1bc25df9154efbd2ef1d2cbba2cae5347d57e', + ], + [ + '7962414450c76c1689c7b48f8202ec37fb224cf5ac0bfa1570328a8a3d7c77ab', + '100b610ec4ffb4760d5c1fc133ef6f6b12507a051f04ac5760afa5b29db83437', + ], + [ + '3514087834964b54b15b160644d915485a16977225b8847bb0dd085137ec47ca', + 'ef0afbb2056205448e1652c48e8127fc6039e77c15c2378b7e7d15a0de293311', + ], + [ + 'd3cc30ad6b483e4bc79ce2c9dd8bc54993e947eb8df787b442943d3f7b527eaf', + '8b378a22d827278d89c5e9be8f9508ae3c2ad46290358630afb34db04eede0a4', + ], + [ + '1624d84780732860ce1c78fcbfefe08b2b29823db913f6493975ba0ff4847610', + '68651cf9b6da903e0914448c6cd9d4ca896878f5282be4c8cc06e2a404078575', + ], + [ + '733ce80da955a8a26902c95633e62a985192474b5af207da6df7b4fd5fc61cd4', + 'f5435a2bd2badf7d485a4d8b8db9fcce3e1ef8e0201e4578c54673bc1dc5ea1d', + ], + [ + '15d9441254945064cf1a1c33bbd3b49f8966c5092171e699ef258dfab81c045c', + 'd56eb30b69463e7234f5137b73b84177434800bacebfc685fc37bbe9efe4070d', + ], + [ + 'a1d0fcf2ec9de675b612136e5ce70d271c21417c9d2b8aaaac138599d0717940', + 'edd77f50bcb5a3cab2e90737309667f2641462a54070f3d519212d39c197a629', + ], + [ + 'e22fbe15c0af8ccc5780c0735f84dbe9a790badee8245c06c7ca37331cb36980', + 'a855babad5cd60c88b430a69f53a1a7a38289154964799be43d06d77d31da06', + ], + [ + '311091dd9860e8e20ee13473c1155f5f69635e394704eaa74009452246cfa9b3', + '66db656f87d1f04fffd1f04788c06830871ec5a64feee685bd80f0b1286d8374', + ], + [ + '34c1fd04d301be89b31c0442d3e6ac24883928b45a9340781867d4232ec2dbdf', + '9414685e97b1b5954bd46f730174136d57f1ceeb487443dc5321857ba73abee', + ], + [ + 'f219ea5d6b54701c1c14de5b557eb42a8d13f3abbcd08affcc2a5e6b049b8d63', + '4cb95957e83d40b0f73af4544cccf6b1f4b08d3c07b27fb8d8c2962a400766d1', + ], + [ + 'd7b8740f74a8fbaab1f683db8f45de26543a5490bca627087236912469a0b448', + 'fa77968128d9c92ee1010f337ad4717eff15db5ed3c049b3411e0315eaa4593b', + ], + [ + '32d31c222f8f6f0ef86f7c98d3a3335ead5bcd32abdd94289fe4d3091aa824bf', + '5f3032f5892156e39ccd3d7915b9e1da2e6dac9e6f26e961118d14b8462e1661', + ], + [ + '7461f371914ab32671045a155d9831ea8793d77cd59592c4340f86cbc18347b5', + '8ec0ba238b96bec0cbdddcae0aa442542eee1ff50c986ea6b39847b3cc092ff6', + ], + [ + 'ee079adb1df1860074356a25aa38206a6d716b2c3e67453d287698bad7b2b2d6', + '8dc2412aafe3be5c4c5f37e0ecc5f9f6a446989af04c4e25ebaac479ec1c8c1e', + ], + [ + '16ec93e447ec83f0467b18302ee620f7e65de331874c9dc72bfd8616ba9da6b5', + '5e4631150e62fb40d0e8c2a7ca5804a39d58186a50e497139626778e25b0674d', + ], + [ + 'eaa5f980c245f6f038978290afa70b6bd8855897f98b6aa485b96065d537bd99', + 'f65f5d3e292c2e0819a528391c994624d784869d7e6ea67fb18041024edc07dc', + ], + [ + '78c9407544ac132692ee1910a02439958ae04877151342ea96c4b6b35a49f51', + 'f3e0319169eb9b85d5404795539a5e68fa1fbd583c064d2462b675f194a3ddb4', + ], + [ + '494f4be219a1a77016dcd838431aea0001cdc8ae7a6fc688726578d9702857a5', + '42242a969283a5f339ba7f075e36ba2af925ce30d767ed6e55f4b031880d562c', + ], + [ + 'a598a8030da6d86c6bc7f2f5144ea549d28211ea58faa70ebf4c1e665c1fe9b5', + '204b5d6f84822c307e4b4a7140737aec23fc63b65b35f86a10026dbd2d864e6b', + ], + [ + 'c41916365abb2b5d09192f5f2dbeafec208f020f12570a184dbadc3e58595997', + '4f14351d0087efa49d245b328984989d5caf9450f34bfc0ed16e96b58fa9913', + ], + [ + '841d6063a586fa475a724604da03bc5b92a2e0d2e0a36acfe4c73a5514742881', + '73867f59c0659e81904f9a1c7543698e62562d6744c169ce7a36de01a8d6154', + ], + [ + '5e95bb399a6971d376026947f89bde2f282b33810928be4ded112ac4d70e20d5', + '39f23f366809085beebfc71181313775a99c9aed7d8ba38b161384c746012865', + ], + [ + '36e4641a53948fd476c39f8a99fd974e5ec07564b5315d8bf99471bca0ef2f66', + 'd2424b1b1abe4eb8164227b085c9aa9456ea13493fd563e06fd51cf5694c78fc', + ], + [ + '336581ea7bfbbb290c191a2f507a41cf5643842170e914faeab27c2c579f726', + 'ead12168595fe1be99252129b6e56b3391f7ab1410cd1e0ef3dcdcabd2fda224', + ], + [ + '8ab89816dadfd6b6a1f2634fcf00ec8403781025ed6890c4849742706bd43ede', + '6fdcef09f2f6d0a044e654aef624136f503d459c3e89845858a47a9129cdd24e', + ], + [ + '1e33f1a746c9c5778133344d9299fcaa20b0938e8acff2544bb40284b8c5fb94', + '60660257dd11b3aa9c8ed618d24edff2306d320f1d03010e33a7d2057f3b3b6', + ], + [ + '85b7c1dcb3cec1b7ee7f30ded79dd20a0ed1f4cc18cbcfcfa410361fd8f08f31', + '3d98a9cdd026dd43f39048f25a8847f4fcafad1895d7a633c6fed3c35e999511', + ], + [ + '29df9fbd8d9e46509275f4b125d6d45d7fbe9a3b878a7af872a2800661ac5f51', + 'b4c4fe99c775a606e2d8862179139ffda61dc861c019e55cd2876eb2a27d84b', + ], + [ + 'a0b1cae06b0a847a3fea6e671aaf8adfdfe58ca2f768105c8082b2e449fce252', + 'ae434102edde0958ec4b19d917a6a28e6b72da1834aff0e650f049503a296cf2', + ], + [ + '4e8ceafb9b3e9a136dc7ff67e840295b499dfb3b2133e4ba113f2e4c0e121e5', + 'cf2174118c8b6d7a4b48f6d534ce5c79422c086a63460502b827ce62a326683c', + ], + [ + 'd24a44e047e19b6f5afb81c7ca2f69080a5076689a010919f42725c2b789a33b', + '6fb8d5591b466f8fc63db50f1c0f1c69013f996887b8244d2cdec417afea8fa3', + ], + [ + 'ea01606a7a6c9cdd249fdfcfacb99584001edd28abbab77b5104e98e8e3b35d4', + '322af4908c7312b0cfbfe369f7a7b3cdb7d4494bc2823700cfd652188a3ea98d', + ], + [ + 'af8addbf2b661c8a6c6328655eb96651252007d8c5ea31be4ad196de8ce2131f', + '6749e67c029b85f52a034eafd096836b2520818680e26ac8f3dfbcdb71749700', + ], + [ + 'e3ae1974566ca06cc516d47e0fb165a674a3dabcfca15e722f0e3450f45889', + '2aeabe7e4531510116217f07bf4d07300de97e4874f81f533420a72eeb0bd6a4', + ], + [ + '591ee355313d99721cf6993ffed1e3e301993ff3ed258802075ea8ced397e246', + 'b0ea558a113c30bea60fc4775460c7901ff0b053d25ca2bdeee98f1a4be5d196', + ], + [ + '11396d55fda54c49f19aa97318d8da61fa8584e47b084945077cf03255b52984', + '998c74a8cd45ac01289d5833a7beb4744ff536b01b257be4c5767bea93ea57a4', + ], + [ + '3c5d2a1ba39c5a1790000738c9e0c40b8dcdfd5468754b6405540157e017aa7a', + 'b2284279995a34e2f9d4de7396fc18b80f9b8b9fdd270f6661f79ca4c81bd257', + ], + [ + 'cc8704b8a60a0defa3a99a7299f2e9c3fbc395afb04ac078425ef8a1793cc030', + 'bdd46039feed17881d1e0862db347f8cf395b74fc4bcdc4e940b74e3ac1f1b13', + ], + [ + 'c533e4f7ea8555aacd9777ac5cad29b97dd4defccc53ee7ea204119b2889b197', + '6f0a256bc5efdf429a2fb6242f1a43a2d9b925bb4a4b3a26bb8e0f45eb596096', + ], + [ + 'c14f8f2ccb27d6f109f6d08d03cc96a69ba8c34eec07bbcf566d48e33da6593', + 'c359d6923bb398f7fd4473e16fe1c28475b740dd098075e6c0e8649113dc3a38', + ], + [ + 'a6cbc3046bc6a450bac24789fa17115a4c9739ed75f8f21ce441f72e0b90e6ef', + '21ae7f4680e889bb130619e2c0f95a360ceb573c70603139862afd617fa9b9f', + ], + [ + '347d6d9a02c48927ebfb86c1359b1caf130a3c0267d11ce6344b39f99d43cc38', + '60ea7f61a353524d1c987f6ecec92f086d565ab687870cb12689ff1e31c74448', + ], + [ + 'da6545d2181db8d983f7dcb375ef5866d47c67b1bf31c8cf855ef7437b72656a', + '49b96715ab6878a79e78f07ce5680c5d6673051b4935bd897fea824b77dc208a', + ], + [ + 'c40747cc9d012cb1a13b8148309c6de7ec25d6945d657146b9d5994b8feb1111', + '5ca560753be2a12fc6de6caf2cb489565db936156b9514e1bb5e83037e0fa2d4', + ], + [ + '4e42c8ec82c99798ccf3a610be870e78338c7f713348bd34c8203ef4037f3502', + '7571d74ee5e0fb92a7a8b33a07783341a5492144cc54bcc40a94473693606437', + ], + [ + '3775ab7089bc6af823aba2e1af70b236d251cadb0c86743287522a1b3b0dedea', + 'be52d107bcfa09d8bcb9736a828cfa7fac8db17bf7a76a2c42ad961409018cf7', + ], + [ + 'cee31cbf7e34ec379d94fb814d3d775ad954595d1314ba8846959e3e82f74e26', + '8fd64a14c06b589c26b947ae2bcf6bfa0149ef0be14ed4d80f448a01c43b1c6d', + ], + [ + 'b4f9eaea09b6917619f6ea6a4eb5464efddb58fd45b1ebefcdc1a01d08b47986', + '39e5c9925b5a54b07433a4f18c61726f8bb131c012ca542eb24a8ac07200682a', + ], + [ + 'd4263dfc3d2df923a0179a48966d30ce84e2515afc3dccc1b77907792ebcc60e', + '62dfaf07a0f78feb30e30d6295853ce189e127760ad6cf7fae164e122a208d54', + ], + [ + '48457524820fa65a4f8d35eb6930857c0032acc0a4a2de422233eeda897612c4', + '25a748ab367979d98733c38a1fa1c2e7dc6cc07db2d60a9ae7a76aaa49bd0f77', + ], + [ + 'dfeeef1881101f2cb11644f3a2afdfc2045e19919152923f367a1767c11cceda', + 'ecfb7056cf1de042f9420bab396793c0c390bde74b4bbdff16a83ae09a9a7517', + ], + [ + '6d7ef6b17543f8373c573f44e1f389835d89bcbc6062ced36c82df83b8fae859', + 'cd450ec335438986dfefa10c57fea9bcc521a0959b2d80bbf74b190dca712d10', + ], + [ + 'e75605d59102a5a2684500d3b991f2e3f3c88b93225547035af25af66e04541f', + 'f5c54754a8f71ee540b9b48728473e314f729ac5308b06938360990e2bfad125', + ], + [ + 'eb98660f4c4dfaa06a2be453d5020bc99a0c2e60abe388457dd43fefb1ed620c', + '6cb9a8876d9cb8520609af3add26cd20a0a7cd8a9411131ce85f44100099223e', + ], + [ + '13e87b027d8514d35939f2e6892b19922154596941888336dc3563e3b8dba942', + 'fef5a3c68059a6dec5d624114bf1e91aac2b9da568d6abeb2570d55646b8adf1', + ], + [ + 'ee163026e9fd6fe017c38f06a5be6fc125424b371ce2708e7bf4491691e5764a', + '1acb250f255dd61c43d94ccc670d0f58f49ae3fa15b96623e5430da0ad6c62b2', + ], + [ + 'b268f5ef9ad51e4d78de3a750c2dc89b1e626d43505867999932e5db33af3d80', + '5f310d4b3c99b9ebb19f77d41c1dee018cf0d34fd4191614003e945a1216e423', + ], + [ + 'ff07f3118a9df035e9fad85eb6c7bfe42b02f01ca99ceea3bf7ffdba93c4750d', + '438136d603e858a3a5c440c38eccbaddc1d2942114e2eddd4740d098ced1f0d8', + ], + [ + '8d8b9855c7c052a34146fd20ffb658bea4b9f69e0d825ebec16e8c3ce2b526a1', + 'cdb559eedc2d79f926baf44fb84ea4d44bcf50fee51d7ceb30e2e7f463036758', + ], + [ + '52db0b5384dfbf05bfa9d472d7ae26dfe4b851ceca91b1eba54263180da32b63', + 'c3b997d050ee5d423ebaf66a6db9f57b3180c902875679de924b69d84a7b375', + ], + [ + 'e62f9490d3d51da6395efd24e80919cc7d0f29c3f3fa48c6fff543becbd43352', + '6d89ad7ba4876b0b22c2ca280c682862f342c8591f1daf5170e07bfd9ccafa7d', + ], + [ + '7f30ea2476b399b4957509c88f77d0191afa2ff5cb7b14fd6d8e7d65aaab1193', + 'ca5ef7d4b231c94c3b15389a5f6311e9daff7bb67b103e9880ef4bff637acaec', + ], + [ + '5098ff1e1d9f14fb46a210fada6c903fef0fb7b4a1dd1d9ac60a0361800b7a00', + '9731141d81fc8f8084d37c6e7542006b3ee1b40d60dfe5362a5b132fd17ddc0', + ], + [ + '32b78c7de9ee512a72895be6b9cbefa6e2f3c4ccce445c96b9f2c81e2778ad58', + 'ee1849f513df71e32efc3896ee28260c73bb80547ae2275ba497237794c8753c', + ], + [ + 'e2cb74fddc8e9fbcd076eef2a7c72b0ce37d50f08269dfc074b581550547a4f7', + 'd3aa2ed71c9dd2247a62df062736eb0baddea9e36122d2be8641abcb005cc4a4', + ], + [ + '8438447566d4d7bedadc299496ab357426009a35f235cb141be0d99cd10ae3a8', + 'c4e1020916980a4da5d01ac5e6ad330734ef0d7906631c4f2390426b2edd791f', + ], + [ + '4162d488b89402039b584c6fc6c308870587d9c46f660b878ab65c82c711d67e', + '67163e903236289f776f22c25fb8a3afc1732f2b84b4e95dbda47ae5a0852649', + ], + [ + '3fad3fa84caf0f34f0f89bfd2dcf54fc175d767aec3e50684f3ba4a4bf5f683d', + 'cd1bc7cb6cc407bb2f0ca647c718a730cf71872e7d0d2a53fa20efcdfe61826', + ], + [ + '674f2600a3007a00568c1a7ce05d0816c1fb84bf1370798f1c69532faeb1a86b', + '299d21f9413f33b3edf43b257004580b70db57da0b182259e09eecc69e0d38a5', + ], + [ + 'd32f4da54ade74abb81b815ad1fb3b263d82d6c692714bcff87d29bd5ee9f08f', + 'f9429e738b8e53b968e99016c059707782e14f4535359d582fc416910b3eea87', + ], + [ + '30e4e670435385556e593657135845d36fbb6931f72b08cb1ed954f1e3ce3ff6', + '462f9bce619898638499350113bbc9b10a878d35da70740dc695a559eb88db7b', + ], + [ + 'be2062003c51cc3004682904330e4dee7f3dcd10b01e580bf1971b04d4cad297', + '62188bc49d61e5428573d48a74e1c655b1c61090905682a0d5558ed72dccb9bc', + ], + [ + '93144423ace3451ed29e0fb9ac2af211cb6e84a601df5993c419859fff5df04a', + '7c10dfb164c3425f5c71a3f9d7992038f1065224f72bb9d1d902a6d13037b47c', + ], + [ + 'b015f8044f5fcbdcf21ca26d6c34fb8197829205c7b7d2a7cb66418c157b112c', + 'ab8c1e086d04e813744a655b2df8d5f83b3cdc6faa3088c1d3aea1454e3a1d5f', + ], + [ + 'd5e9e1da649d97d89e4868117a465a3a4f8a18de57a140d36b3f2af341a21b52', + '4cb04437f391ed73111a13cc1d4dd0db1693465c2240480d8955e8592f27447a', + ], + [ + 'd3ae41047dd7ca065dbf8ed77b992439983005cd72e16d6f996a5316d36966bb', + 'bd1aeb21ad22ebb22a10f0303417c6d964f8cdd7df0aca614b10dc14d125ac46', + ], + [ + '463e2763d885f958fc66cdd22800f0a487197d0a82e377b49f80af87c897b065', + 'bfefacdb0e5d0fd7df3a311a94de062b26b80c61fbc97508b79992671ef7ca7f', + ], + [ + '7985fdfd127c0567c6f53ec1bb63ec3158e597c40bfe747c83cddfc910641917', + '603c12daf3d9862ef2b25fe1de289aed24ed291e0ec6708703a5bd567f32ed03', + ], + [ + '74a1ad6b5f76e39db2dd249410eac7f99e74c59cb83d2d0ed5ff1543da7703e9', + 'cc6157ef18c9c63cd6193d83631bbea0093e0968942e8c33d5737fd790e0db08', + ], + [ + '30682a50703375f602d416664ba19b7fc9bab42c72747463a71d0896b22f6da3', + '553e04f6b018b4fa6c8f39e7f311d3176290d0e0f19ca73f17714d9977a22ff8', + ], + [ + '9e2158f0d7c0d5f26c3791efefa79597654e7a2b2464f52b1ee6c1347769ef57', + '712fcdd1b9053f09003a3481fa7762e9ffd7c8ef35a38509e2fbf2629008373', + ], + [ + '176e26989a43c9cfeba4029c202538c28172e566e3c4fce7322857f3be327d66', + 'ed8cc9d04b29eb877d270b4878dc43c19aefd31f4eee09ee7b47834c1fa4b1c3', + ], + [ + '75d46efea3771e6e68abb89a13ad747ecf1892393dfc4f1b7004788c50374da8', + '9852390a99507679fd0b86fd2b39a868d7efc22151346e1a3ca4726586a6bed8', + ], + [ + '809a20c67d64900ffb698c4c825f6d5f2310fb0451c869345b7319f645605721', + '9e994980d9917e22b76b061927fa04143d096ccc54963e6a5ebfa5f3f8e286c1', + ], + [ + '1b38903a43f7f114ed4500b4eac7083fdefece1cf29c63528d563446f972c180', + '4036edc931a60ae889353f77fd53de4a2708b26b6f5da72ad3394119daf408f9', + ], + ], + }, +}; diff --git a/node_modules/elliptic/lib/elliptic/utils.js b/node_modules/elliptic/lib/elliptic/utils.js new file mode 100644 index 0000000..b3ef2d2 --- /dev/null +++ b/node_modules/elliptic/lib/elliptic/utils.js @@ -0,0 +1,122 @@ +'use strict'; + +var utils = exports; +var BN = require('bn.js'); +var minAssert = require('minimalistic-assert'); +var minUtils = require('minimalistic-crypto-utils'); + +utils.assert = minAssert; +utils.toArray = minUtils.toArray; +utils.zero2 = minUtils.zero2; +utils.toHex = minUtils.toHex; +utils.encode = minUtils.encode; + +// Represent num in a w-NAF form +function getNAF(num, w, bits) { + var naf = new Array(Math.max(num.bitLength(), bits) + 1); + var i; + for (i = 0; i < naf.length; i += 1) { + naf[i] = 0; + } + + var ws = 1 << (w + 1); + var k = num.clone(); + + for (i = 0; i < naf.length; i++) { + var z; + var mod = k.andln(ws - 1); + if (k.isOdd()) { + if (mod > (ws >> 1) - 1) + z = (ws >> 1) - mod; + else + z = mod; + k.isubn(z); + } else { + z = 0; + } + + naf[i] = z; + k.iushrn(1); + } + + return naf; +} +utils.getNAF = getNAF; + +// Represent k1, k2 in a Joint Sparse Form +function getJSF(k1, k2) { + var jsf = [ + [], + [], + ]; + + k1 = k1.clone(); + k2 = k2.clone(); + var d1 = 0; + var d2 = 0; + var m8; + while (k1.cmpn(-d1) > 0 || k2.cmpn(-d2) > 0) { + // First phase + var m14 = (k1.andln(3) + d1) & 3; + var m24 = (k2.andln(3) + d2) & 3; + if (m14 === 3) + m14 = -1; + if (m24 === 3) + m24 = -1; + var u1; + if ((m14 & 1) === 0) { + u1 = 0; + } else { + m8 = (k1.andln(7) + d1) & 7; + if ((m8 === 3 || m8 === 5) && m24 === 2) + u1 = -m14; + else + u1 = m14; + } + jsf[0].push(u1); + + var u2; + if ((m24 & 1) === 0) { + u2 = 0; + } else { + m8 = (k2.andln(7) + d2) & 7; + if ((m8 === 3 || m8 === 5) && m14 === 2) + u2 = -m24; + else + u2 = m24; + } + jsf[1].push(u2); + + // Second phase + if (2 * d1 === u1 + 1) + d1 = 1 - d1; + if (2 * d2 === u2 + 1) + d2 = 1 - d2; + k1.iushrn(1); + k2.iushrn(1); + } + + return jsf; +} +utils.getJSF = getJSF; + +function cachedProperty(obj, name, computer) { + var key = '_' + name; + obj.prototype[name] = function cachedProperty() { + return this[key] !== undefined ? this[key] : + this[key] = computer.call(this); + }; +} +utils.cachedProperty = cachedProperty; + +function parseBytes(bytes) { + return typeof bytes === 'string' ? utils.toArray(bytes, 'hex') : + bytes; +} +utils.parseBytes = parseBytes; + +function intFromLE(bytes) { + return new BN(bytes, 'hex', 'le'); +} +utils.intFromLE = intFromLE; + diff --git a/node_modules/elliptic/package.json b/node_modules/elliptic/package.json new file mode 100644 index 0000000..a76675e --- /dev/null +++ b/node_modules/elliptic/package.json @@ -0,0 +1,56 @@ +{ + "name": "elliptic", + "version": "6.5.5", + "description": "EC cryptography", + "main": "lib/elliptic.js", + "files": [ + "lib" + ], + "scripts": { + "lint": "eslint lib test", + "lint:fix": "npm run lint -- --fix", + "unit": "istanbul test _mocha --reporter=spec test/index.js", + "test": "npm run lint && npm run unit", + "version": "grunt dist && git add dist/" + }, + "repository": { + "type": "git", + "url": "git@github.com:indutny/elliptic" + }, + "keywords": [ + "EC", + "Elliptic", + "curve", + "Cryptography" + ], + "author": "Fedor Indutny ", + "license": "MIT", + "bugs": { + "url": "https://github.com/indutny/elliptic/issues" + }, + "homepage": "https://github.com/indutny/elliptic", + "devDependencies": { + "brfs": "^2.0.2", + "coveralls": "^3.1.0", + "eslint": "^7.6.0", + "grunt": "^1.2.1", + "grunt-browserify": "^5.3.0", + "grunt-cli": "^1.3.2", + "grunt-contrib-connect": "^3.0.0", + "grunt-contrib-copy": "^1.0.0", + "grunt-contrib-uglify": "^5.0.0", + "grunt-mocha-istanbul": "^5.0.2", + "grunt-saucelabs": "^9.0.1", + "istanbul": "^0.4.5", + "mocha": "^8.0.1" + }, + "dependencies": { + "bn.js": "^4.11.9", + "brorand": "^1.1.0", + "hash.js": "^1.0.0", + "hmac-drbg": "^1.0.1", + "inherits": "^2.0.4", + "minimalistic-assert": "^1.0.1", + "minimalistic-crypto-utils": "^1.0.1" + } +} diff --git a/node_modules/hash-base/LICENSE b/node_modules/hash-base/LICENSE new file mode 100644 index 0000000..6f02ae8 --- /dev/null +++ b/node_modules/hash-base/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2016 Kirill Fomichev + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/node_modules/hash-base/README.md b/node_modules/hash-base/README.md new file mode 100644 index 0000000..83ae2ed --- /dev/null +++ b/node_modules/hash-base/README.md @@ -0,0 +1,48 @@ +# hash-base + +[![NPM Package](https://img.shields.io/npm/v/hash-base.svg?style=flat-square)](https://www.npmjs.org/package/hash-base) +[![Build Status](https://img.shields.io/travis/crypto-browserify/hash-base.svg?branch=master&style=flat-square)](https://travis-ci.org/crypto-browserify/hash-base) +[![Dependency status](https://img.shields.io/david/crypto-browserify/hash-base.svg?style=flat-square)](https://david-dm.org/crypto-browserify/hash-base#info=dependencies) + +[![js-standard-style](https://cdn.rawgit.com/feross/standard/master/badge.svg)](https://github.com/feross/standard) + +Abstract base class to inherit from if you want to create streams implementing the same API as node crypto [Hash][1] (for [Cipher][2] / [Decipher][3] check [crypto-browserify/cipher-base][4]). + +## Example + +```js +const HashBase = require('hash-base') +const inherits = require('inherits') + +// our hash function is XOR sum of all bytes +function MyHash () { + HashBase.call(this, 1) // in bytes + + this._sum = 0x00 +} + +inherits(MyHash, HashBase) + +MyHash.prototype._update = function () { + for (let i = 0; i < this._block.length; ++i) this._sum ^= this._block[i] +} + +MyHash.prototype._digest = function () { + return this._sum +} + +const data = Buffer.from([ 0x00, 0x42, 0x01 ]) +const hash = new MyHash().update(data).digest() +console.log(hash) // => 67 +``` +You also can check [source code](index.js) or [crypto-browserify/md5.js][5] + +## LICENSE + +MIT + +[1]: https://nodejs.org/api/crypto.html#crypto_class_hash +[2]: https://nodejs.org/api/crypto.html#crypto_class_cipher +[3]: https://nodejs.org/api/crypto.html#crypto_class_decipher +[4]: https://github.com/crypto-browserify/cipher-base +[5]: https://github.com/crypto-browserify/md5.js diff --git a/node_modules/hash-base/index.js b/node_modules/hash-base/index.js new file mode 100644 index 0000000..059480b --- /dev/null +++ b/node_modules/hash-base/index.js @@ -0,0 +1,95 @@ +'use strict' +var Buffer = require('safe-buffer').Buffer +var Transform = require('readable-stream').Transform +var inherits = require('inherits') + +function throwIfNotStringOrBuffer (val, prefix) { + if (!Buffer.isBuffer(val) && typeof val !== 'string') { + throw new TypeError(prefix + ' must be a string or a buffer') + } +} + +function HashBase (blockSize) { + Transform.call(this) + + this._block = Buffer.allocUnsafe(blockSize) + this._blockSize = blockSize + this._blockOffset = 0 + this._length = [0, 0, 0, 0] + + this._finalized = false +} + +inherits(HashBase, Transform) + +HashBase.prototype._transform = function (chunk, encoding, callback) { + var error = null + try { + this.update(chunk, encoding) + } catch (err) { + error = err + } + + callback(error) +} + +HashBase.prototype._flush = function (callback) { + var error = null + try { + this.push(this.digest()) + } catch (err) { + error = err + } + + callback(error) +} + +HashBase.prototype.update = function (data, encoding) { + throwIfNotStringOrBuffer(data, 'Data') + if (this._finalized) throw new Error('Digest already called') + if (!Buffer.isBuffer(data)) data = Buffer.from(data, encoding) + + // consume data + var block = this._block + var offset = 0 + while (this._blockOffset + data.length - offset >= this._blockSize) { + for (var i = this._blockOffset; i < this._blockSize;) block[i++] = data[offset++] + this._update() + this._blockOffset = 0 + } + while (offset < data.length) block[this._blockOffset++] = data[offset++] + + // update length + for (var j = 0, carry = data.length * 8; carry > 0; ++j) { + this._length[j] += carry + carry = (this._length[j] / 0x0100000000) | 0 + if (carry > 0) this._length[j] -= 0x0100000000 * carry + } + + return this +} + +HashBase.prototype._update = function () { + throw new Error('_update is not implemented') +} + +HashBase.prototype.digest = function (encoding) { + if (this._finalized) throw new Error('Digest already called') + this._finalized = true + + var digest = this._digest() + if (encoding !== undefined) digest = digest.toString(encoding) + + // reset state + this._block.fill(0) + this._blockOffset = 0 + for (var i = 0; i < 4; ++i) this._length[i] = 0 + + return digest +} + +HashBase.prototype._digest = function () { + throw new Error('_digest is not implemented') +} + +module.exports = HashBase diff --git a/node_modules/hash-base/package.json b/node_modules/hash-base/package.json new file mode 100644 index 0000000..cfea6d4 --- /dev/null +++ b/node_modules/hash-base/package.json @@ -0,0 +1,42 @@ +{ + "name": "hash-base", + "version": "3.1.0", + "description": "abstract base class for hash-streams", + "keywords": [ + "hash", + "stream" + ], + "homepage": "https://github.com/crypto-browserify/hash-base", + "bugs": { + "url": "https://github.com/crypto-browserify/hash-base/issues" + }, + "license": "MIT", + "author": "Kirill Fomichev (https://github.com/fanatid)", + "files": [ + "index.js" + ], + "main": "index.js", + "repository": { + "type": "git", + "url": "https://github.com/crypto-browserify/hash-base.git" + }, + "scripts": { + "coverage": "nyc node test/*.js", + "lint": "standard", + "test": "npm run lint && npm run unit", + "unit": "node test/*.js" + }, + "dependencies": { + "inherits": "^2.0.4", + "readable-stream": "^3.6.0", + "safe-buffer": "^5.2.0" + }, + "devDependencies": { + "nyc": "^15.0.1", + "standard": "^14.3.3", + "tape": "^5.0.0" + }, + "engines": { + "node": ">=4" + } +} diff --git a/node_modules/hash.js/.eslintrc.js b/node_modules/hash.js/.eslintrc.js new file mode 100644 index 0000000..614f8fc --- /dev/null +++ b/node_modules/hash.js/.eslintrc.js @@ -0,0 +1,41 @@ +module.exports = { + 'env': { + 'browser': true, + 'commonjs': true, + 'node': true, + 'es6': true + }, + 'parserOptions': { + 'ecmaVersion': 8 + }, + 'extends': 'eslint:recommended', + 'rules': { + 'indent': [ + 'error', + 2, + { + 'FunctionDeclaration': { + 'parameters': 'first' + }, + 'FunctionExpression': { + 'parameters': 'first' + }, + 'CallExpression': { + 'arguments': 'first' + } + } + ], + 'linebreak-style': [ + 'error', + 'unix' + ], + 'quotes': [ + 'error', + 'single' + ], + 'semi': [ + 'error', + 'always' + ] + } +}; diff --git a/node_modules/hash.js/.travis.yml b/node_modules/hash.js/.travis.yml new file mode 100644 index 0000000..5406edc --- /dev/null +++ b/node_modules/hash.js/.travis.yml @@ -0,0 +1,10 @@ +sudo: false +language: node_js +node_js: + - "6" + - "8" + - "10" + - "stable" +branches: + only: + - master diff --git a/node_modules/hash.js/README.md b/node_modules/hash.js/README.md new file mode 100644 index 0000000..006d3bb --- /dev/null +++ b/node_modules/hash.js/README.md @@ -0,0 +1,48 @@ +# hash.js [![Build Status](https://secure.travis-ci.org/indutny/hash.js.svg)](http://travis-ci.org/indutny/hash.js) + +Just a bike-shed. + +## Install + +```sh +npm install hash.js +``` + +## Usage + +```js +var hash = require('hash.js') +hash.sha256().update('abc').digest('hex') +``` + +## Selective hash usage + +```js +var sha512 = require('hash.js/lib/hash/sha/512'); +sha512().update('abc').digest('hex'); +``` + +#### LICENSE + +This software is licensed under the MIT License. + +Copyright Fedor Indutny, 2014. + +Permission is hereby granted, free of charge, to any person obtaining a +copy of this software and associated documentation files (the +"Software"), to deal in the Software without restriction, including +without limitation the rights to use, copy, modify, merge, publish, +distribute, sublicense, and/or sell copies of the Software, and to permit +persons to whom the Software is furnished to do so, subject to the +following conditions: + +The above copyright notice and this permission notice shall be included +in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN +NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, +DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR +OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE +USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/node_modules/hash.js/lib/hash.d.ts b/node_modules/hash.js/lib/hash.d.ts new file mode 100644 index 0000000..abbb5c5 --- /dev/null +++ b/node_modules/hash.js/lib/hash.d.ts @@ -0,0 +1,106 @@ +declare var hash: Hash; + +declare module "hash.js" { + export = hash; +} + +interface BlockHash { + hmacStrength: number + padLength: number + endian: 'big' | 'little' +} + +interface MessageDigest { + blockSize: number + outSize: number + update(msg: any, enc?: 'hex'): T + digest(): number[] + digest(enc: 'hex'): string +} + +interface Hash { + hmac: HmacConstructor + ripemd: RipemdSet + ripemd160: Ripemd160Constructor + sha: ShaSet + sha1: Sha1Constructor + sha224: Sha224Constructor + sha256: Sha256Constructor + sha384: Sha384Constructor + sha512: Sha512Constructor + utils: Utils +} + +interface Utils { + toArray(msg: any, enc: 'hex'): Array + toHex(msg: any): string +} + +interface RipemdSet { + ripemd160: Ripemd160Constructor +} + +interface ShaSet { + sha1: Sha1Constructor + sha224: Sha224Constructor + sha256: Sha256Constructor + sha384: Sha384Constructor + sha512: Sha512Constructor +} + +interface HmacConstructor { (hash: BlockHash, key: any, enc?: 'hex'): Hmac } +interface Ripemd160Constructor { (): Ripemd160 } +interface Sha1Constructor { (): Sha1; } +interface Sha224Constructor { (): Sha224; } +interface Sha256Constructor { (): Sha256; } +interface Sha384Constructor { (): Sha384; } +interface Sha512Constructor { (): Sha512; } + +interface Hmac extends MessageDigest { + blockSize: 512 + outSize: 160 +} + +interface Ripemd160 extends BlockHash, MessageDigest { + blockSize: 512 + hmacStrength: 192 + outSize: 160 + padLength: 64 + endian: 'little' +} + +interface Sha1 extends BlockHash, MessageDigest { + blockSize: 512 + hmacStrength: 80 + outSize: 160 + padLength: 64 + endian: 'big' +} +interface Sha224 extends BlockHash, MessageDigest { + blockSize: 512 + hmacStrength: 192 + outSize: 224 + padLength: 64 + endian: 'big' +} +interface Sha256 extends BlockHash, MessageDigest { + blockSize: 512 + hmacStrength: 192 + outSize: 256 + padLength: 64 + endian: 'big' +} +interface Sha384 extends BlockHash, MessageDigest { + blockSize: 1024 + hmacStrength: 192 + outSize: 384 + padLength: 128 + endian: 'big' +} +interface Sha512 extends BlockHash, MessageDigest { + blockSize: 1024 + hmacStrength: 192 + outSize: 512 + padLength: 128 + endian: 'big' +} diff --git a/node_modules/hash.js/lib/hash.js b/node_modules/hash.js/lib/hash.js new file mode 100644 index 0000000..f59b673 --- /dev/null +++ b/node_modules/hash.js/lib/hash.js @@ -0,0 +1,15 @@ +var hash = exports; + +hash.utils = require('./hash/utils'); +hash.common = require('./hash/common'); +hash.sha = require('./hash/sha'); +hash.ripemd = require('./hash/ripemd'); +hash.hmac = require('./hash/hmac'); + +// Proxy hash functions to the main object +hash.sha1 = hash.sha.sha1; +hash.sha256 = hash.sha.sha256; +hash.sha224 = hash.sha.sha224; +hash.sha384 = hash.sha.sha384; +hash.sha512 = hash.sha.sha512; +hash.ripemd160 = hash.ripemd.ripemd160; diff --git a/node_modules/hash.js/lib/hash/common.js b/node_modules/hash.js/lib/hash/common.js new file mode 100644 index 0000000..c49f476 --- /dev/null +++ b/node_modules/hash.js/lib/hash/common.js @@ -0,0 +1,92 @@ +'use strict'; + +var utils = require('./utils'); +var assert = require('minimalistic-assert'); + +function BlockHash() { + this.pending = null; + this.pendingTotal = 0; + this.blockSize = this.constructor.blockSize; + this.outSize = this.constructor.outSize; + this.hmacStrength = this.constructor.hmacStrength; + this.padLength = this.constructor.padLength / 8; + this.endian = 'big'; + + this._delta8 = this.blockSize / 8; + this._delta32 = this.blockSize / 32; +} +exports.BlockHash = BlockHash; + +BlockHash.prototype.update = function update(msg, enc) { + // Convert message to array, pad it, and join into 32bit blocks + msg = utils.toArray(msg, enc); + if (!this.pending) + this.pending = msg; + else + this.pending = this.pending.concat(msg); + this.pendingTotal += msg.length; + + // Enough data, try updating + if (this.pending.length >= this._delta8) { + msg = this.pending; + + // Process pending data in blocks + var r = msg.length % this._delta8; + this.pending = msg.slice(msg.length - r, msg.length); + if (this.pending.length === 0) + this.pending = null; + + msg = utils.join32(msg, 0, msg.length - r, this.endian); + for (var i = 0; i < msg.length; i += this._delta32) + this._update(msg, i, i + this._delta32); + } + + return this; +}; + +BlockHash.prototype.digest = function digest(enc) { + this.update(this._pad()); + assert(this.pending === null); + + return this._digest(enc); +}; + +BlockHash.prototype._pad = function pad() { + var len = this.pendingTotal; + var bytes = this._delta8; + var k = bytes - ((len + this.padLength) % bytes); + var res = new Array(k + this.padLength); + res[0] = 0x80; + for (var i = 1; i < k; i++) + res[i] = 0; + + // Append length + len <<= 3; + if (this.endian === 'big') { + for (var t = 8; t < this.padLength; t++) + res[i++] = 0; + + res[i++] = 0; + res[i++] = 0; + res[i++] = 0; + res[i++] = 0; + res[i++] = (len >>> 24) & 0xff; + res[i++] = (len >>> 16) & 0xff; + res[i++] = (len >>> 8) & 0xff; + res[i++] = len & 0xff; + } else { + res[i++] = len & 0xff; + res[i++] = (len >>> 8) & 0xff; + res[i++] = (len >>> 16) & 0xff; + res[i++] = (len >>> 24) & 0xff; + res[i++] = 0; + res[i++] = 0; + res[i++] = 0; + res[i++] = 0; + + for (t = 8; t < this.padLength; t++) + res[i++] = 0; + } + + return res; +}; diff --git a/node_modules/hash.js/lib/hash/hmac.js b/node_modules/hash.js/lib/hash/hmac.js new file mode 100644 index 0000000..faff12d --- /dev/null +++ b/node_modules/hash.js/lib/hash/hmac.js @@ -0,0 +1,47 @@ +'use strict'; + +var utils = require('./utils'); +var assert = require('minimalistic-assert'); + +function Hmac(hash, key, enc) { + if (!(this instanceof Hmac)) + return new Hmac(hash, key, enc); + this.Hash = hash; + this.blockSize = hash.blockSize / 8; + this.outSize = hash.outSize / 8; + this.inner = null; + this.outer = null; + + this._init(utils.toArray(key, enc)); +} +module.exports = Hmac; + +Hmac.prototype._init = function init(key) { + // Shorten key, if needed + if (key.length > this.blockSize) + key = new this.Hash().update(key).digest(); + assert(key.length <= this.blockSize); + + // Add padding to key + for (var i = key.length; i < this.blockSize; i++) + key.push(0); + + for (i = 0; i < key.length; i++) + key[i] ^= 0x36; + this.inner = new this.Hash().update(key); + + // 0x36 ^ 0x5c = 0x6a + for (i = 0; i < key.length; i++) + key[i] ^= 0x6a; + this.outer = new this.Hash().update(key); +}; + +Hmac.prototype.update = function update(msg, enc) { + this.inner.update(msg, enc); + return this; +}; + +Hmac.prototype.digest = function digest(enc) { + this.outer.update(this.inner.digest()); + return this.outer.digest(enc); +}; diff --git a/node_modules/hash.js/lib/hash/ripemd.js b/node_modules/hash.js/lib/hash/ripemd.js new file mode 100644 index 0000000..2dcdff2 --- /dev/null +++ b/node_modules/hash.js/lib/hash/ripemd.js @@ -0,0 +1,146 @@ +'use strict'; + +var utils = require('./utils'); +var common = require('./common'); + +var rotl32 = utils.rotl32; +var sum32 = utils.sum32; +var sum32_3 = utils.sum32_3; +var sum32_4 = utils.sum32_4; +var BlockHash = common.BlockHash; + +function RIPEMD160() { + if (!(this instanceof RIPEMD160)) + return new RIPEMD160(); + + BlockHash.call(this); + + this.h = [ 0x67452301, 0xefcdab89, 0x98badcfe, 0x10325476, 0xc3d2e1f0 ]; + this.endian = 'little'; +} +utils.inherits(RIPEMD160, BlockHash); +exports.ripemd160 = RIPEMD160; + +RIPEMD160.blockSize = 512; +RIPEMD160.outSize = 160; +RIPEMD160.hmacStrength = 192; +RIPEMD160.padLength = 64; + +RIPEMD160.prototype._update = function update(msg, start) { + var A = this.h[0]; + var B = this.h[1]; + var C = this.h[2]; + var D = this.h[3]; + var E = this.h[4]; + var Ah = A; + var Bh = B; + var Ch = C; + var Dh = D; + var Eh = E; + for (var j = 0; j < 80; j++) { + var T = sum32( + rotl32( + sum32_4(A, f(j, B, C, D), msg[r[j] + start], K(j)), + s[j]), + E); + A = E; + E = D; + D = rotl32(C, 10); + C = B; + B = T; + T = sum32( + rotl32( + sum32_4(Ah, f(79 - j, Bh, Ch, Dh), msg[rh[j] + start], Kh(j)), + sh[j]), + Eh); + Ah = Eh; + Eh = Dh; + Dh = rotl32(Ch, 10); + Ch = Bh; + Bh = T; + } + T = sum32_3(this.h[1], C, Dh); + this.h[1] = sum32_3(this.h[2], D, Eh); + this.h[2] = sum32_3(this.h[3], E, Ah); + this.h[3] = sum32_3(this.h[4], A, Bh); + this.h[4] = sum32_3(this.h[0], B, Ch); + this.h[0] = T; +}; + +RIPEMD160.prototype._digest = function digest(enc) { + if (enc === 'hex') + return utils.toHex32(this.h, 'little'); + else + return utils.split32(this.h, 'little'); +}; + +function f(j, x, y, z) { + if (j <= 15) + return x ^ y ^ z; + else if (j <= 31) + return (x & y) | ((~x) & z); + else if (j <= 47) + return (x | (~y)) ^ z; + else if (j <= 63) + return (x & z) | (y & (~z)); + else + return x ^ (y | (~z)); +} + +function K(j) { + if (j <= 15) + return 0x00000000; + else if (j <= 31) + return 0x5a827999; + else if (j <= 47) + return 0x6ed9eba1; + else if (j <= 63) + return 0x8f1bbcdc; + else + return 0xa953fd4e; +} + +function Kh(j) { + if (j <= 15) + return 0x50a28be6; + else if (j <= 31) + return 0x5c4dd124; + else if (j <= 47) + return 0x6d703ef3; + else if (j <= 63) + return 0x7a6d76e9; + else + return 0x00000000; +} + +var r = [ + 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, + 7, 4, 13, 1, 10, 6, 15, 3, 12, 0, 9, 5, 2, 14, 11, 8, + 3, 10, 14, 4, 9, 15, 8, 1, 2, 7, 0, 6, 13, 11, 5, 12, + 1, 9, 11, 10, 0, 8, 12, 4, 13, 3, 7, 15, 14, 5, 6, 2, + 4, 0, 5, 9, 7, 12, 2, 10, 14, 1, 3, 8, 11, 6, 15, 13 +]; + +var rh = [ + 5, 14, 7, 0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12, + 6, 11, 3, 7, 0, 13, 5, 10, 14, 15, 8, 12, 4, 9, 1, 2, + 15, 5, 1, 3, 7, 14, 6, 9, 11, 8, 12, 2, 10, 0, 4, 13, + 8, 6, 4, 1, 3, 11, 15, 0, 5, 12, 2, 13, 9, 7, 10, 14, + 12, 15, 10, 4, 1, 5, 8, 7, 6, 2, 13, 14, 0, 3, 9, 11 +]; + +var s = [ + 11, 14, 15, 12, 5, 8, 7, 9, 11, 13, 14, 15, 6, 7, 9, 8, + 7, 6, 8, 13, 11, 9, 7, 15, 7, 12, 15, 9, 11, 7, 13, 12, + 11, 13, 6, 7, 14, 9, 13, 15, 14, 8, 13, 6, 5, 12, 7, 5, + 11, 12, 14, 15, 14, 15, 9, 8, 9, 14, 5, 6, 8, 6, 5, 12, + 9, 15, 5, 11, 6, 8, 13, 12, 5, 12, 13, 14, 11, 8, 5, 6 +]; + +var sh = [ + 8, 9, 9, 11, 13, 15, 15, 5, 7, 7, 8, 11, 14, 14, 12, 6, + 9, 13, 15, 7, 12, 8, 9, 11, 7, 7, 12, 7, 6, 15, 13, 11, + 9, 7, 15, 11, 8, 6, 6, 14, 12, 13, 5, 14, 13, 13, 7, 5, + 15, 5, 8, 11, 14, 14, 6, 14, 6, 9, 12, 9, 12, 5, 15, 8, + 8, 5, 12, 9, 12, 5, 14, 6, 8, 13, 6, 5, 15, 13, 11, 11 +]; diff --git a/node_modules/hash.js/lib/hash/sha.js b/node_modules/hash.js/lib/hash/sha.js new file mode 100644 index 0000000..f34a38d --- /dev/null +++ b/node_modules/hash.js/lib/hash/sha.js @@ -0,0 +1,7 @@ +'use strict'; + +exports.sha1 = require('./sha/1'); +exports.sha224 = require('./sha/224'); +exports.sha256 = require('./sha/256'); +exports.sha384 = require('./sha/384'); +exports.sha512 = require('./sha/512'); diff --git a/node_modules/hash.js/lib/hash/sha/1.js b/node_modules/hash.js/lib/hash/sha/1.js new file mode 100644 index 0000000..fcdfa29 --- /dev/null +++ b/node_modules/hash.js/lib/hash/sha/1.js @@ -0,0 +1,74 @@ +'use strict'; + +var utils = require('../utils'); +var common = require('../common'); +var shaCommon = require('./common'); + +var rotl32 = utils.rotl32; +var sum32 = utils.sum32; +var sum32_5 = utils.sum32_5; +var ft_1 = shaCommon.ft_1; +var BlockHash = common.BlockHash; + +var sha1_K = [ + 0x5A827999, 0x6ED9EBA1, + 0x8F1BBCDC, 0xCA62C1D6 +]; + +function SHA1() { + if (!(this instanceof SHA1)) + return new SHA1(); + + BlockHash.call(this); + this.h = [ + 0x67452301, 0xefcdab89, 0x98badcfe, + 0x10325476, 0xc3d2e1f0 ]; + this.W = new Array(80); +} + +utils.inherits(SHA1, BlockHash); +module.exports = SHA1; + +SHA1.blockSize = 512; +SHA1.outSize = 160; +SHA1.hmacStrength = 80; +SHA1.padLength = 64; + +SHA1.prototype._update = function _update(msg, start) { + var W = this.W; + + for (var i = 0; i < 16; i++) + W[i] = msg[start + i]; + + for(; i < W.length; i++) + W[i] = rotl32(W[i - 3] ^ W[i - 8] ^ W[i - 14] ^ W[i - 16], 1); + + var a = this.h[0]; + var b = this.h[1]; + var c = this.h[2]; + var d = this.h[3]; + var e = this.h[4]; + + for (i = 0; i < W.length; i++) { + var s = ~~(i / 20); + var t = sum32_5(rotl32(a, 5), ft_1(s, b, c, d), e, W[i], sha1_K[s]); + e = d; + d = c; + c = rotl32(b, 30); + b = a; + a = t; + } + + this.h[0] = sum32(this.h[0], a); + this.h[1] = sum32(this.h[1], b); + this.h[2] = sum32(this.h[2], c); + this.h[3] = sum32(this.h[3], d); + this.h[4] = sum32(this.h[4], e); +}; + +SHA1.prototype._digest = function digest(enc) { + if (enc === 'hex') + return utils.toHex32(this.h, 'big'); + else + return utils.split32(this.h, 'big'); +}; diff --git a/node_modules/hash.js/lib/hash/sha/224.js b/node_modules/hash.js/lib/hash/sha/224.js new file mode 100644 index 0000000..c69882d --- /dev/null +++ b/node_modules/hash.js/lib/hash/sha/224.js @@ -0,0 +1,30 @@ +'use strict'; + +var utils = require('../utils'); +var SHA256 = require('./256'); + +function SHA224() { + if (!(this instanceof SHA224)) + return new SHA224(); + + SHA256.call(this); + this.h = [ + 0xc1059ed8, 0x367cd507, 0x3070dd17, 0xf70e5939, + 0xffc00b31, 0x68581511, 0x64f98fa7, 0xbefa4fa4 ]; +} +utils.inherits(SHA224, SHA256); +module.exports = SHA224; + +SHA224.blockSize = 512; +SHA224.outSize = 224; +SHA224.hmacStrength = 192; +SHA224.padLength = 64; + +SHA224.prototype._digest = function digest(enc) { + // Just truncate output + if (enc === 'hex') + return utils.toHex32(this.h.slice(0, 7), 'big'); + else + return utils.split32(this.h.slice(0, 7), 'big'); +}; + diff --git a/node_modules/hash.js/lib/hash/sha/256.js b/node_modules/hash.js/lib/hash/sha/256.js new file mode 100644 index 0000000..6dbd6a0 --- /dev/null +++ b/node_modules/hash.js/lib/hash/sha/256.js @@ -0,0 +1,105 @@ +'use strict'; + +var utils = require('../utils'); +var common = require('../common'); +var shaCommon = require('./common'); +var assert = require('minimalistic-assert'); + +var sum32 = utils.sum32; +var sum32_4 = utils.sum32_4; +var sum32_5 = utils.sum32_5; +var ch32 = shaCommon.ch32; +var maj32 = shaCommon.maj32; +var s0_256 = shaCommon.s0_256; +var s1_256 = shaCommon.s1_256; +var g0_256 = shaCommon.g0_256; +var g1_256 = shaCommon.g1_256; + +var BlockHash = common.BlockHash; + +var sha256_K = [ + 0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, + 0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5, + 0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3, + 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174, + 0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc, + 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da, + 0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, + 0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967, + 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13, + 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85, + 0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, + 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070, + 0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, + 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3, + 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208, + 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2 +]; + +function SHA256() { + if (!(this instanceof SHA256)) + return new SHA256(); + + BlockHash.call(this); + this.h = [ + 0x6a09e667, 0xbb67ae85, 0x3c6ef372, 0xa54ff53a, + 0x510e527f, 0x9b05688c, 0x1f83d9ab, 0x5be0cd19 + ]; + this.k = sha256_K; + this.W = new Array(64); +} +utils.inherits(SHA256, BlockHash); +module.exports = SHA256; + +SHA256.blockSize = 512; +SHA256.outSize = 256; +SHA256.hmacStrength = 192; +SHA256.padLength = 64; + +SHA256.prototype._update = function _update(msg, start) { + var W = this.W; + + for (var i = 0; i < 16; i++) + W[i] = msg[start + i]; + for (; i < W.length; i++) + W[i] = sum32_4(g1_256(W[i - 2]), W[i - 7], g0_256(W[i - 15]), W[i - 16]); + + var a = this.h[0]; + var b = this.h[1]; + var c = this.h[2]; + var d = this.h[3]; + var e = this.h[4]; + var f = this.h[5]; + var g = this.h[6]; + var h = this.h[7]; + + assert(this.k.length === W.length); + for (i = 0; i < W.length; i++) { + var T1 = sum32_5(h, s1_256(e), ch32(e, f, g), this.k[i], W[i]); + var T2 = sum32(s0_256(a), maj32(a, b, c)); + h = g; + g = f; + f = e; + e = sum32(d, T1); + d = c; + c = b; + b = a; + a = sum32(T1, T2); + } + + this.h[0] = sum32(this.h[0], a); + this.h[1] = sum32(this.h[1], b); + this.h[2] = sum32(this.h[2], c); + this.h[3] = sum32(this.h[3], d); + this.h[4] = sum32(this.h[4], e); + this.h[5] = sum32(this.h[5], f); + this.h[6] = sum32(this.h[6], g); + this.h[7] = sum32(this.h[7], h); +}; + +SHA256.prototype._digest = function digest(enc) { + if (enc === 'hex') + return utils.toHex32(this.h, 'big'); + else + return utils.split32(this.h, 'big'); +}; diff --git a/node_modules/hash.js/lib/hash/sha/384.js b/node_modules/hash.js/lib/hash/sha/384.js new file mode 100644 index 0000000..01df256 --- /dev/null +++ b/node_modules/hash.js/lib/hash/sha/384.js @@ -0,0 +1,35 @@ +'use strict'; + +var utils = require('../utils'); + +var SHA512 = require('./512'); + +function SHA384() { + if (!(this instanceof SHA384)) + return new SHA384(); + + SHA512.call(this); + this.h = [ + 0xcbbb9d5d, 0xc1059ed8, + 0x629a292a, 0x367cd507, + 0x9159015a, 0x3070dd17, + 0x152fecd8, 0xf70e5939, + 0x67332667, 0xffc00b31, + 0x8eb44a87, 0x68581511, + 0xdb0c2e0d, 0x64f98fa7, + 0x47b5481d, 0xbefa4fa4 ]; +} +utils.inherits(SHA384, SHA512); +module.exports = SHA384; + +SHA384.blockSize = 1024; +SHA384.outSize = 384; +SHA384.hmacStrength = 192; +SHA384.padLength = 128; + +SHA384.prototype._digest = function digest(enc) { + if (enc === 'hex') + return utils.toHex32(this.h.slice(0, 12), 'big'); + else + return utils.split32(this.h.slice(0, 12), 'big'); +}; diff --git a/node_modules/hash.js/lib/hash/sha/512.js b/node_modules/hash.js/lib/hash/sha/512.js new file mode 100644 index 0000000..c371a25 --- /dev/null +++ b/node_modules/hash.js/lib/hash/sha/512.js @@ -0,0 +1,330 @@ +'use strict'; + +var utils = require('../utils'); +var common = require('../common'); +var assert = require('minimalistic-assert'); + +var rotr64_hi = utils.rotr64_hi; +var rotr64_lo = utils.rotr64_lo; +var shr64_hi = utils.shr64_hi; +var shr64_lo = utils.shr64_lo; +var sum64 = utils.sum64; +var sum64_hi = utils.sum64_hi; +var sum64_lo = utils.sum64_lo; +var sum64_4_hi = utils.sum64_4_hi; +var sum64_4_lo = utils.sum64_4_lo; +var sum64_5_hi = utils.sum64_5_hi; +var sum64_5_lo = utils.sum64_5_lo; + +var BlockHash = common.BlockHash; + +var sha512_K = [ + 0x428a2f98, 0xd728ae22, 0x71374491, 0x23ef65cd, + 0xb5c0fbcf, 0xec4d3b2f, 0xe9b5dba5, 0x8189dbbc, + 0x3956c25b, 0xf348b538, 0x59f111f1, 0xb605d019, + 0x923f82a4, 0xaf194f9b, 0xab1c5ed5, 0xda6d8118, + 0xd807aa98, 0xa3030242, 0x12835b01, 0x45706fbe, + 0x243185be, 0x4ee4b28c, 0x550c7dc3, 0xd5ffb4e2, + 0x72be5d74, 0xf27b896f, 0x80deb1fe, 0x3b1696b1, + 0x9bdc06a7, 0x25c71235, 0xc19bf174, 0xcf692694, + 0xe49b69c1, 0x9ef14ad2, 0xefbe4786, 0x384f25e3, + 0x0fc19dc6, 0x8b8cd5b5, 0x240ca1cc, 0x77ac9c65, + 0x2de92c6f, 0x592b0275, 0x4a7484aa, 0x6ea6e483, + 0x5cb0a9dc, 0xbd41fbd4, 0x76f988da, 0x831153b5, + 0x983e5152, 0xee66dfab, 0xa831c66d, 0x2db43210, + 0xb00327c8, 0x98fb213f, 0xbf597fc7, 0xbeef0ee4, + 0xc6e00bf3, 0x3da88fc2, 0xd5a79147, 0x930aa725, + 0x06ca6351, 0xe003826f, 0x14292967, 0x0a0e6e70, + 0x27b70a85, 0x46d22ffc, 0x2e1b2138, 0x5c26c926, + 0x4d2c6dfc, 0x5ac42aed, 0x53380d13, 0x9d95b3df, + 0x650a7354, 0x8baf63de, 0x766a0abb, 0x3c77b2a8, + 0x81c2c92e, 0x47edaee6, 0x92722c85, 0x1482353b, + 0xa2bfe8a1, 0x4cf10364, 0xa81a664b, 0xbc423001, + 0xc24b8b70, 0xd0f89791, 0xc76c51a3, 0x0654be30, + 0xd192e819, 0xd6ef5218, 0xd6990624, 0x5565a910, + 0xf40e3585, 0x5771202a, 0x106aa070, 0x32bbd1b8, + 0x19a4c116, 0xb8d2d0c8, 0x1e376c08, 0x5141ab53, + 0x2748774c, 0xdf8eeb99, 0x34b0bcb5, 0xe19b48a8, + 0x391c0cb3, 0xc5c95a63, 0x4ed8aa4a, 0xe3418acb, + 0x5b9cca4f, 0x7763e373, 0x682e6ff3, 0xd6b2b8a3, + 0x748f82ee, 0x5defb2fc, 0x78a5636f, 0x43172f60, + 0x84c87814, 0xa1f0ab72, 0x8cc70208, 0x1a6439ec, + 0x90befffa, 0x23631e28, 0xa4506ceb, 0xde82bde9, + 0xbef9a3f7, 0xb2c67915, 0xc67178f2, 0xe372532b, + 0xca273ece, 0xea26619c, 0xd186b8c7, 0x21c0c207, + 0xeada7dd6, 0xcde0eb1e, 0xf57d4f7f, 0xee6ed178, + 0x06f067aa, 0x72176fba, 0x0a637dc5, 0xa2c898a6, + 0x113f9804, 0xbef90dae, 0x1b710b35, 0x131c471b, + 0x28db77f5, 0x23047d84, 0x32caab7b, 0x40c72493, + 0x3c9ebe0a, 0x15c9bebc, 0x431d67c4, 0x9c100d4c, + 0x4cc5d4be, 0xcb3e42b6, 0x597f299c, 0xfc657e2a, + 0x5fcb6fab, 0x3ad6faec, 0x6c44198c, 0x4a475817 +]; + +function SHA512() { + if (!(this instanceof SHA512)) + return new SHA512(); + + BlockHash.call(this); + this.h = [ + 0x6a09e667, 0xf3bcc908, + 0xbb67ae85, 0x84caa73b, + 0x3c6ef372, 0xfe94f82b, + 0xa54ff53a, 0x5f1d36f1, + 0x510e527f, 0xade682d1, + 0x9b05688c, 0x2b3e6c1f, + 0x1f83d9ab, 0xfb41bd6b, + 0x5be0cd19, 0x137e2179 ]; + this.k = sha512_K; + this.W = new Array(160); +} +utils.inherits(SHA512, BlockHash); +module.exports = SHA512; + +SHA512.blockSize = 1024; +SHA512.outSize = 512; +SHA512.hmacStrength = 192; +SHA512.padLength = 128; + +SHA512.prototype._prepareBlock = function _prepareBlock(msg, start) { + var W = this.W; + + // 32 x 32bit words + for (var i = 0; i < 32; i++) + W[i] = msg[start + i]; + for (; i < W.length; i += 2) { + var c0_hi = g1_512_hi(W[i - 4], W[i - 3]); // i - 2 + var c0_lo = g1_512_lo(W[i - 4], W[i - 3]); + var c1_hi = W[i - 14]; // i - 7 + var c1_lo = W[i - 13]; + var c2_hi = g0_512_hi(W[i - 30], W[i - 29]); // i - 15 + var c2_lo = g0_512_lo(W[i - 30], W[i - 29]); + var c3_hi = W[i - 32]; // i - 16 + var c3_lo = W[i - 31]; + + W[i] = sum64_4_hi( + c0_hi, c0_lo, + c1_hi, c1_lo, + c2_hi, c2_lo, + c3_hi, c3_lo); + W[i + 1] = sum64_4_lo( + c0_hi, c0_lo, + c1_hi, c1_lo, + c2_hi, c2_lo, + c3_hi, c3_lo); + } +}; + +SHA512.prototype._update = function _update(msg, start) { + this._prepareBlock(msg, start); + + var W = this.W; + + var ah = this.h[0]; + var al = this.h[1]; + var bh = this.h[2]; + var bl = this.h[3]; + var ch = this.h[4]; + var cl = this.h[5]; + var dh = this.h[6]; + var dl = this.h[7]; + var eh = this.h[8]; + var el = this.h[9]; + var fh = this.h[10]; + var fl = this.h[11]; + var gh = this.h[12]; + var gl = this.h[13]; + var hh = this.h[14]; + var hl = this.h[15]; + + assert(this.k.length === W.length); + for (var i = 0; i < W.length; i += 2) { + var c0_hi = hh; + var c0_lo = hl; + var c1_hi = s1_512_hi(eh, el); + var c1_lo = s1_512_lo(eh, el); + var c2_hi = ch64_hi(eh, el, fh, fl, gh, gl); + var c2_lo = ch64_lo(eh, el, fh, fl, gh, gl); + var c3_hi = this.k[i]; + var c3_lo = this.k[i + 1]; + var c4_hi = W[i]; + var c4_lo = W[i + 1]; + + var T1_hi = sum64_5_hi( + c0_hi, c0_lo, + c1_hi, c1_lo, + c2_hi, c2_lo, + c3_hi, c3_lo, + c4_hi, c4_lo); + var T1_lo = sum64_5_lo( + c0_hi, c0_lo, + c1_hi, c1_lo, + c2_hi, c2_lo, + c3_hi, c3_lo, + c4_hi, c4_lo); + + c0_hi = s0_512_hi(ah, al); + c0_lo = s0_512_lo(ah, al); + c1_hi = maj64_hi(ah, al, bh, bl, ch, cl); + c1_lo = maj64_lo(ah, al, bh, bl, ch, cl); + + var T2_hi = sum64_hi(c0_hi, c0_lo, c1_hi, c1_lo); + var T2_lo = sum64_lo(c0_hi, c0_lo, c1_hi, c1_lo); + + hh = gh; + hl = gl; + + gh = fh; + gl = fl; + + fh = eh; + fl = el; + + eh = sum64_hi(dh, dl, T1_hi, T1_lo); + el = sum64_lo(dl, dl, T1_hi, T1_lo); + + dh = ch; + dl = cl; + + ch = bh; + cl = bl; + + bh = ah; + bl = al; + + ah = sum64_hi(T1_hi, T1_lo, T2_hi, T2_lo); + al = sum64_lo(T1_hi, T1_lo, T2_hi, T2_lo); + } + + sum64(this.h, 0, ah, al); + sum64(this.h, 2, bh, bl); + sum64(this.h, 4, ch, cl); + sum64(this.h, 6, dh, dl); + sum64(this.h, 8, eh, el); + sum64(this.h, 10, fh, fl); + sum64(this.h, 12, gh, gl); + sum64(this.h, 14, hh, hl); +}; + +SHA512.prototype._digest = function digest(enc) { + if (enc === 'hex') + return utils.toHex32(this.h, 'big'); + else + return utils.split32(this.h, 'big'); +}; + +function ch64_hi(xh, xl, yh, yl, zh) { + var r = (xh & yh) ^ ((~xh) & zh); + if (r < 0) + r += 0x100000000; + return r; +} + +function ch64_lo(xh, xl, yh, yl, zh, zl) { + var r = (xl & yl) ^ ((~xl) & zl); + if (r < 0) + r += 0x100000000; + return r; +} + +function maj64_hi(xh, xl, yh, yl, zh) { + var r = (xh & yh) ^ (xh & zh) ^ (yh & zh); + if (r < 0) + r += 0x100000000; + return r; +} + +function maj64_lo(xh, xl, yh, yl, zh, zl) { + var r = (xl & yl) ^ (xl & zl) ^ (yl & zl); + if (r < 0) + r += 0x100000000; + return r; +} + +function s0_512_hi(xh, xl) { + var c0_hi = rotr64_hi(xh, xl, 28); + var c1_hi = rotr64_hi(xl, xh, 2); // 34 + var c2_hi = rotr64_hi(xl, xh, 7); // 39 + + var r = c0_hi ^ c1_hi ^ c2_hi; + if (r < 0) + r += 0x100000000; + return r; +} + +function s0_512_lo(xh, xl) { + var c0_lo = rotr64_lo(xh, xl, 28); + var c1_lo = rotr64_lo(xl, xh, 2); // 34 + var c2_lo = rotr64_lo(xl, xh, 7); // 39 + + var r = c0_lo ^ c1_lo ^ c2_lo; + if (r < 0) + r += 0x100000000; + return r; +} + +function s1_512_hi(xh, xl) { + var c0_hi = rotr64_hi(xh, xl, 14); + var c1_hi = rotr64_hi(xh, xl, 18); + var c2_hi = rotr64_hi(xl, xh, 9); // 41 + + var r = c0_hi ^ c1_hi ^ c2_hi; + if (r < 0) + r += 0x100000000; + return r; +} + +function s1_512_lo(xh, xl) { + var c0_lo = rotr64_lo(xh, xl, 14); + var c1_lo = rotr64_lo(xh, xl, 18); + var c2_lo = rotr64_lo(xl, xh, 9); // 41 + + var r = c0_lo ^ c1_lo ^ c2_lo; + if (r < 0) + r += 0x100000000; + return r; +} + +function g0_512_hi(xh, xl) { + var c0_hi = rotr64_hi(xh, xl, 1); + var c1_hi = rotr64_hi(xh, xl, 8); + var c2_hi = shr64_hi(xh, xl, 7); + + var r = c0_hi ^ c1_hi ^ c2_hi; + if (r < 0) + r += 0x100000000; + return r; +} + +function g0_512_lo(xh, xl) { + var c0_lo = rotr64_lo(xh, xl, 1); + var c1_lo = rotr64_lo(xh, xl, 8); + var c2_lo = shr64_lo(xh, xl, 7); + + var r = c0_lo ^ c1_lo ^ c2_lo; + if (r < 0) + r += 0x100000000; + return r; +} + +function g1_512_hi(xh, xl) { + var c0_hi = rotr64_hi(xh, xl, 19); + var c1_hi = rotr64_hi(xl, xh, 29); // 61 + var c2_hi = shr64_hi(xh, xl, 6); + + var r = c0_hi ^ c1_hi ^ c2_hi; + if (r < 0) + r += 0x100000000; + return r; +} + +function g1_512_lo(xh, xl) { + var c0_lo = rotr64_lo(xh, xl, 19); + var c1_lo = rotr64_lo(xl, xh, 29); // 61 + var c2_lo = shr64_lo(xh, xl, 6); + + var r = c0_lo ^ c1_lo ^ c2_lo; + if (r < 0) + r += 0x100000000; + return r; +} diff --git a/node_modules/hash.js/lib/hash/sha/common.js b/node_modules/hash.js/lib/hash/sha/common.js new file mode 100644 index 0000000..d41b464 --- /dev/null +++ b/node_modules/hash.js/lib/hash/sha/common.js @@ -0,0 +1,49 @@ +'use strict'; + +var utils = require('../utils'); +var rotr32 = utils.rotr32; + +function ft_1(s, x, y, z) { + if (s === 0) + return ch32(x, y, z); + if (s === 1 || s === 3) + return p32(x, y, z); + if (s === 2) + return maj32(x, y, z); +} +exports.ft_1 = ft_1; + +function ch32(x, y, z) { + return (x & y) ^ ((~x) & z); +} +exports.ch32 = ch32; + +function maj32(x, y, z) { + return (x & y) ^ (x & z) ^ (y & z); +} +exports.maj32 = maj32; + +function p32(x, y, z) { + return x ^ y ^ z; +} +exports.p32 = p32; + +function s0_256(x) { + return rotr32(x, 2) ^ rotr32(x, 13) ^ rotr32(x, 22); +} +exports.s0_256 = s0_256; + +function s1_256(x) { + return rotr32(x, 6) ^ rotr32(x, 11) ^ rotr32(x, 25); +} +exports.s1_256 = s1_256; + +function g0_256(x) { + return rotr32(x, 7) ^ rotr32(x, 18) ^ (x >>> 3); +} +exports.g0_256 = g0_256; + +function g1_256(x) { + return rotr32(x, 17) ^ rotr32(x, 19) ^ (x >>> 10); +} +exports.g1_256 = g1_256; diff --git a/node_modules/hash.js/lib/hash/utils.js b/node_modules/hash.js/lib/hash/utils.js new file mode 100644 index 0000000..7487f8e --- /dev/null +++ b/node_modules/hash.js/lib/hash/utils.js @@ -0,0 +1,278 @@ +'use strict'; + +var assert = require('minimalistic-assert'); +var inherits = require('inherits'); + +exports.inherits = inherits; + +function isSurrogatePair(msg, i) { + if ((msg.charCodeAt(i) & 0xFC00) !== 0xD800) { + return false; + } + if (i < 0 || i + 1 >= msg.length) { + return false; + } + return (msg.charCodeAt(i + 1) & 0xFC00) === 0xDC00; +} + +function toArray(msg, enc) { + if (Array.isArray(msg)) + return msg.slice(); + if (!msg) + return []; + var res = []; + if (typeof msg === 'string') { + if (!enc) { + // Inspired by stringToUtf8ByteArray() in closure-library by Google + // https://github.com/google/closure-library/blob/8598d87242af59aac233270742c8984e2b2bdbe0/closure/goog/crypt/crypt.js#L117-L143 + // Apache License 2.0 + // https://github.com/google/closure-library/blob/master/LICENSE + var p = 0; + for (var i = 0; i < msg.length; i++) { + var c = msg.charCodeAt(i); + if (c < 128) { + res[p++] = c; + } else if (c < 2048) { + res[p++] = (c >> 6) | 192; + res[p++] = (c & 63) | 128; + } else if (isSurrogatePair(msg, i)) { + c = 0x10000 + ((c & 0x03FF) << 10) + (msg.charCodeAt(++i) & 0x03FF); + res[p++] = (c >> 18) | 240; + res[p++] = ((c >> 12) & 63) | 128; + res[p++] = ((c >> 6) & 63) | 128; + res[p++] = (c & 63) | 128; + } else { + res[p++] = (c >> 12) | 224; + res[p++] = ((c >> 6) & 63) | 128; + res[p++] = (c & 63) | 128; + } + } + } else if (enc === 'hex') { + msg = msg.replace(/[^a-z0-9]+/ig, ''); + if (msg.length % 2 !== 0) + msg = '0' + msg; + for (i = 0; i < msg.length; i += 2) + res.push(parseInt(msg[i] + msg[i + 1], 16)); + } + } else { + for (i = 0; i < msg.length; i++) + res[i] = msg[i] | 0; + } + return res; +} +exports.toArray = toArray; + +function toHex(msg) { + var res = ''; + for (var i = 0; i < msg.length; i++) + res += zero2(msg[i].toString(16)); + return res; +} +exports.toHex = toHex; + +function htonl(w) { + var res = (w >>> 24) | + ((w >>> 8) & 0xff00) | + ((w << 8) & 0xff0000) | + ((w & 0xff) << 24); + return res >>> 0; +} +exports.htonl = htonl; + +function toHex32(msg, endian) { + var res = ''; + for (var i = 0; i < msg.length; i++) { + var w = msg[i]; + if (endian === 'little') + w = htonl(w); + res += zero8(w.toString(16)); + } + return res; +} +exports.toHex32 = toHex32; + +function zero2(word) { + if (word.length === 1) + return '0' + word; + else + return word; +} +exports.zero2 = zero2; + +function zero8(word) { + if (word.length === 7) + return '0' + word; + else if (word.length === 6) + return '00' + word; + else if (word.length === 5) + return '000' + word; + else if (word.length === 4) + return '0000' + word; + else if (word.length === 3) + return '00000' + word; + else if (word.length === 2) + return '000000' + word; + else if (word.length === 1) + return '0000000' + word; + else + return word; +} +exports.zero8 = zero8; + +function join32(msg, start, end, endian) { + var len = end - start; + assert(len % 4 === 0); + var res = new Array(len / 4); + for (var i = 0, k = start; i < res.length; i++, k += 4) { + var w; + if (endian === 'big') + w = (msg[k] << 24) | (msg[k + 1] << 16) | (msg[k + 2] << 8) | msg[k + 3]; + else + w = (msg[k + 3] << 24) | (msg[k + 2] << 16) | (msg[k + 1] << 8) | msg[k]; + res[i] = w >>> 0; + } + return res; +} +exports.join32 = join32; + +function split32(msg, endian) { + var res = new Array(msg.length * 4); + for (var i = 0, k = 0; i < msg.length; i++, k += 4) { + var m = msg[i]; + if (endian === 'big') { + res[k] = m >>> 24; + res[k + 1] = (m >>> 16) & 0xff; + res[k + 2] = (m >>> 8) & 0xff; + res[k + 3] = m & 0xff; + } else { + res[k + 3] = m >>> 24; + res[k + 2] = (m >>> 16) & 0xff; + res[k + 1] = (m >>> 8) & 0xff; + res[k] = m & 0xff; + } + } + return res; +} +exports.split32 = split32; + +function rotr32(w, b) { + return (w >>> b) | (w << (32 - b)); +} +exports.rotr32 = rotr32; + +function rotl32(w, b) { + return (w << b) | (w >>> (32 - b)); +} +exports.rotl32 = rotl32; + +function sum32(a, b) { + return (a + b) >>> 0; +} +exports.sum32 = sum32; + +function sum32_3(a, b, c) { + return (a + b + c) >>> 0; +} +exports.sum32_3 = sum32_3; + +function sum32_4(a, b, c, d) { + return (a + b + c + d) >>> 0; +} +exports.sum32_4 = sum32_4; + +function sum32_5(a, b, c, d, e) { + return (a + b + c + d + e) >>> 0; +} +exports.sum32_5 = sum32_5; + +function sum64(buf, pos, ah, al) { + var bh = buf[pos]; + var bl = buf[pos + 1]; + + var lo = (al + bl) >>> 0; + var hi = (lo < al ? 1 : 0) + ah + bh; + buf[pos] = hi >>> 0; + buf[pos + 1] = lo; +} +exports.sum64 = sum64; + +function sum64_hi(ah, al, bh, bl) { + var lo = (al + bl) >>> 0; + var hi = (lo < al ? 1 : 0) + ah + bh; + return hi >>> 0; +} +exports.sum64_hi = sum64_hi; + +function sum64_lo(ah, al, bh, bl) { + var lo = al + bl; + return lo >>> 0; +} +exports.sum64_lo = sum64_lo; + +function sum64_4_hi(ah, al, bh, bl, ch, cl, dh, dl) { + var carry = 0; + var lo = al; + lo = (lo + bl) >>> 0; + carry += lo < al ? 1 : 0; + lo = (lo + cl) >>> 0; + carry += lo < cl ? 1 : 0; + lo = (lo + dl) >>> 0; + carry += lo < dl ? 1 : 0; + + var hi = ah + bh + ch + dh + carry; + return hi >>> 0; +} +exports.sum64_4_hi = sum64_4_hi; + +function sum64_4_lo(ah, al, bh, bl, ch, cl, dh, dl) { + var lo = al + bl + cl + dl; + return lo >>> 0; +} +exports.sum64_4_lo = sum64_4_lo; + +function sum64_5_hi(ah, al, bh, bl, ch, cl, dh, dl, eh, el) { + var carry = 0; + var lo = al; + lo = (lo + bl) >>> 0; + carry += lo < al ? 1 : 0; + lo = (lo + cl) >>> 0; + carry += lo < cl ? 1 : 0; + lo = (lo + dl) >>> 0; + carry += lo < dl ? 1 : 0; + lo = (lo + el) >>> 0; + carry += lo < el ? 1 : 0; + + var hi = ah + bh + ch + dh + eh + carry; + return hi >>> 0; +} +exports.sum64_5_hi = sum64_5_hi; + +function sum64_5_lo(ah, al, bh, bl, ch, cl, dh, dl, eh, el) { + var lo = al + bl + cl + dl + el; + + return lo >>> 0; +} +exports.sum64_5_lo = sum64_5_lo; + +function rotr64_hi(ah, al, num) { + var r = (al << (32 - num)) | (ah >>> num); + return r >>> 0; +} +exports.rotr64_hi = rotr64_hi; + +function rotr64_lo(ah, al, num) { + var r = (ah << (32 - num)) | (al >>> num); + return r >>> 0; +} +exports.rotr64_lo = rotr64_lo; + +function shr64_hi(ah, al, num) { + return ah >>> num; +} +exports.shr64_hi = shr64_hi; + +function shr64_lo(ah, al, num) { + var r = (ah << (32 - num)) | (al >>> num); + return r >>> 0; +} +exports.shr64_lo = shr64_lo; diff --git a/node_modules/hash.js/package.json b/node_modules/hash.js/package.json new file mode 100644 index 0000000..c1a338a --- /dev/null +++ b/node_modules/hash.js/package.json @@ -0,0 +1,35 @@ +{ + "name": "hash.js", + "version": "1.1.7", + "description": "Various hash functions that could be run by both browser and node", + "main": "lib/hash.js", + "typings": "lib/hash.d.ts", + "scripts": { + "test": "mocha --reporter=spec test/*-test.js && npm run lint", + "lint": "eslint lib/*.js lib/**/*.js lib/**/**/*.js test/*.js" + }, + "repository": { + "type": "git", + "url": "git@github.com:indutny/hash.js" + }, + "keywords": [ + "hash", + "sha256", + "sha224", + "hmac" + ], + "author": "Fedor Indutny ", + "license": "MIT", + "bugs": { + "url": "https://github.com/indutny/hash.js/issues" + }, + "homepage": "https://github.com/indutny/hash.js", + "dependencies": { + "inherits": "^2.0.3", + "minimalistic-assert": "^1.0.1" + }, + "devDependencies": { + "eslint": "^4.19.1", + "mocha": "^5.2.0" + } +} diff --git a/node_modules/hash.js/test/hash-test.js b/node_modules/hash.js/test/hash-test.js new file mode 100644 index 0000000..f6ca564 --- /dev/null +++ b/node_modules/hash.js/test/hash-test.js @@ -0,0 +1,140 @@ +'use strict'; +/* global describe it */ + +var assert = require('assert'); +var crypto = require('crypto'); +var hash = require('../'); + +describe('Hash', function() { + function test(fn, cases) { + for (var i = 0; i < cases.length; i++) { + var msg = cases[i][0]; + var res = cases[i][1]; + var enc = cases[i][2]; + + var dgst = fn().update(msg, enc).digest('hex'); + assert.equal(dgst, res); + + // Split message + dgst = fn().update(msg.slice(0, 2), enc) + .update(msg.slice(2), enc) + .digest('hex'); + assert.equal(dgst, res); + } + } + + it('should support sha256', function() { + assert.equal(hash.sha256.blockSize, 512); + assert.equal(hash.sha256.outSize, 256); + + test(hash.sha256, [ + [ 'abc', + 'ba7816bf8f01cfea414140de5dae2223b00361a396177a9cb410ff61f20015ad' ], + [ 'abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq', + '248d6a61d20638b8e5c026930c3e6039a33ce45964ff2167f6ecedd419db06c1' ], + [ 'deadbeef', + '5f78c33274e43fa9de5659265c1d917e25c03722dcb0b8d27db8d5feaa813953', + 'hex' ], + ]); + }); + + it('should support sha224', function() { + assert.equal(hash.sha224.blockSize, 512); + assert.equal(hash.sha224.outSize, 224); + + test(hash.sha224, [ + [ 'abc', + '23097d223405d8228642a477bda255b32aadbce4bda0b3f7e36c9da7' ], + [ 'abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq', + '75388b16512776cc5dba5da1fd890150b0c6455cb4f58b1952522525' ], + [ 'deadbeef', + '55b9eee5f60cc362ddc07676f620372611e22272f60fdbec94f243f8', + 'hex' ], + ]); + }); + + it('should support ripemd160', function() { + assert.equal(hash.ripemd160.blockSize, 512); + assert.equal(hash.ripemd160.outSize, 160); + + test(hash.ripemd160, [ + [ '', '9c1185a5c5e9fc54612808977ee8f548b2258d31'], + [ 'abc', + '8eb208f7e05d987a9b044a8e98c6b087f15a0bfc' ], + [ 'message digest', + '5d0689ef49d2fae572b881b123a85ffa21595f36' ], + [ 'abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq', + '12a053384a9c0c88e405a06c27dcf49ada62eb2b' ], + [ 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789', + 'b0e20b6e3116640286ed3a87a5713079b21f5189' ], + ]); + }); + + it('should support sha1', function() { + assert.equal(hash.sha1.blockSize, 512); + assert.equal(hash.sha1.outSize, 160); + + test(hash.sha1, [ + [ '', + 'da39a3ee5e6b4b0d3255bfef95601890afd80709' ], + [ 'abc', + 'a9993e364706816aba3e25717850c26c9cd0d89d' ], + [ 'abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq', + '84983e441c3bd26ebaae4aa1f95129e5e54670f1' ], + [ 'deadbeef', + 'd78f8bb992a56a597f6c7a1fb918bb78271367eb', + 'hex' ], + ]); + }); + + it('should support sha512', function() { + assert.equal(hash.sha512.blockSize, 1024); + assert.equal(hash.sha512.outSize, 512); + + test(hash.sha512, [ + [ 'abc', + 'ddaf35a193617abacc417349ae20413112e6fa4e89a97ea20a9eeee64b55d39a' + + '2192992a274fc1a836ba3c23a3feebbd454d4423643ce80e2a9ac94fa54ca49f' + ], + [ + 'abcdefghbcdefghicdefghijdefghijkefghijklfghijklmghijklmn' + + 'hijklmnoijklmnopjklmnopqklmnopqrlmnopqrsmnopqrstnopqrstu', + '8e959b75dae313da8cf4f72814fc143f8f7779c6eb9f7fa17299aeadb6889018' + + '501d289e4900f7e4331b99dec4b5433ac7d329eeb6dd26545e96e55b874be909' + ] + ]); + }); + + it('should support sha384', function() { + assert.equal(hash.sha384.blockSize, 1024); + assert.equal(hash.sha384.outSize, 384); + + test(hash.sha384, [ + [ 'abc', + 'cb00753f45a35e8bb5a03d699ac65007272c32ab0eded1631a8b605a43ff5bed' + + '8086072ba1e7cc2358baeca134c825a7' + ], + [ + 'abcdefghbcdefghicdefghijdefghijkefghijklfghijklmghijklmn' + + 'hijklmnoijklmnopjklmnopqklmnopqrlmnopqrsmnopqrstnopqrstu', + '09330c33f71147e83d192fc782cd1b4753111b173b3b05d22fa08086e3b0f712' + + 'fcc7c71a557e2db966c3e9fa91746039' + ] + ]); + }); + + it('handles utf8 in strings just like crypto', function() { + const algorithm = 'sha256'; + test(hash[algorithm], [ + 'hello', // one byte per character + 'привет', // two bytes per character + '您好', // three bytes per character + '👋', // four bytes per character + 'hello привет 您好 👋!!!' // mixed character lengths + ].map(str => [str, crypto + .createHash(algorithm) + .update(str) + .digest('hex')])); + }); + +}); diff --git a/node_modules/hash.js/test/hmac-test.js b/node_modules/hash.js/test/hmac-test.js new file mode 100644 index 0000000..6fe18e3 --- /dev/null +++ b/node_modules/hash.js/test/hmac-test.js @@ -0,0 +1,62 @@ +'use strict'; +/* global describe it */ + +var assert = require('assert'); +var hash = require('../'); + +describe('Hmac', function() { + describe('mixed test vector', function() { + test({ + name: 'nist 1', + key: '00010203 04050607 08090A0B 0C0D0E0F' + + '10111213 14151617 18191A1B 1C1D1E1F 20212223 24252627' + + '28292A2B 2C2D2E2F 30313233 34353637 38393A3B 3C3D3E3F', + msg: 'Sample message for keylen=blocklen', + res: '8bb9a1db9806f20df7f77b82138c7914d174d59e13dc4d0169c9057b133e1d62' + }); + test({ + name: 'nist 2', + key: '00010203 04050607' + + '08090A0B 0C0D0E0F 10111213 14151617 18191A1B 1C1D1E1F', + msg: 'Sample message for keylen= (this.minEntropy / 8), + 'Not enough entropy. Minimum is: ' + this.minEntropy + ' bits'); + this._init(entropy, nonce, pers); +} +module.exports = HmacDRBG; + +HmacDRBG.prototype._init = function init(entropy, nonce, pers) { + var seed = entropy.concat(nonce).concat(pers); + + this.K = new Array(this.outLen / 8); + this.V = new Array(this.outLen / 8); + for (var i = 0; i < this.V.length; i++) { + this.K[i] = 0x00; + this.V[i] = 0x01; + } + + this._update(seed); + this._reseed = 1; + this.reseedInterval = 0x1000000000000; // 2^48 +}; + +HmacDRBG.prototype._hmac = function hmac() { + return new hash.hmac(this.hash, this.K); +}; + +HmacDRBG.prototype._update = function update(seed) { + var kmac = this._hmac() + .update(this.V) + .update([ 0x00 ]); + if (seed) + kmac = kmac.update(seed); + this.K = kmac.digest(); + this.V = this._hmac().update(this.V).digest(); + if (!seed) + return; + + this.K = this._hmac() + .update(this.V) + .update([ 0x01 ]) + .update(seed) + .digest(); + this.V = this._hmac().update(this.V).digest(); +}; + +HmacDRBG.prototype.reseed = function reseed(entropy, entropyEnc, add, addEnc) { + // Optional entropy enc + if (typeof entropyEnc !== 'string') { + addEnc = add; + add = entropyEnc; + entropyEnc = null; + } + + entropy = utils.toArray(entropy, entropyEnc); + add = utils.toArray(add, addEnc); + + assert(entropy.length >= (this.minEntropy / 8), + 'Not enough entropy. Minimum is: ' + this.minEntropy + ' bits'); + + this._update(entropy.concat(add || [])); + this._reseed = 1; +}; + +HmacDRBG.prototype.generate = function generate(len, enc, add, addEnc) { + if (this._reseed > this.reseedInterval) + throw new Error('Reseed is required'); + + // Optional encoding + if (typeof enc !== 'string') { + addEnc = add; + add = enc; + enc = null; + } + + // Optional additional data + if (add) { + add = utils.toArray(add, addEnc || 'hex'); + this._update(add); + } + + var temp = []; + while (temp.length < len) { + this.V = this._hmac().update(this.V).digest(); + temp = temp.concat(this.V); + } + + var res = temp.slice(0, len); + this._update(add); + this._reseed++; + return utils.encode(res, enc); +}; diff --git a/node_modules/hmac-drbg/package.json b/node_modules/hmac-drbg/package.json new file mode 100644 index 0000000..b95d391 --- /dev/null +++ b/node_modules/hmac-drbg/package.json @@ -0,0 +1,32 @@ +{ + "name": "hmac-drbg", + "version": "1.0.1", + "description": "Deterministic random bit generator (hmac)", + "main": "lib/hmac-drbg.js", + "scripts": { + "test": "mocha --reporter=spec test/*-test.js" + }, + "repository": { + "type": "git", + "url": "git+ssh://git@github.com/indutny/hmac-drbg.git" + }, + "keywords": [ + "hmac", + "drbg", + "prng" + ], + "author": "Fedor Indutny ", + "license": "MIT", + "bugs": { + "url": "https://github.com/indutny/hmac-drbg/issues" + }, + "homepage": "https://github.com/indutny/hmac-drbg#readme", + "devDependencies": { + "mocha": "^3.2.0" + }, + "dependencies": { + "hash.js": "^1.0.3", + "minimalistic-assert": "^1.0.0", + "minimalistic-crypto-utils": "^1.0.1" + } +} diff --git a/node_modules/hmac-drbg/test/drbg-test.js b/node_modules/hmac-drbg/test/drbg-test.js new file mode 100644 index 0000000..14fd28a --- /dev/null +++ b/node_modules/hmac-drbg/test/drbg-test.js @@ -0,0 +1,91 @@ +'use strict'; + +const assert = require('assert'); +const HmacDRBG = require('../'); +const hash = require('hash.js'); + +describe('Hmac_DRBG', () => { + it('should support hmac-drbg-sha256', () => { + function doDrbg(opt) { + const drbg = HmacDRBG({ + hash: hash.sha256, + entropy: opt.entropy, + entropyEnc: 'utf8', + nonce: opt.nonce, + nonceEnc: 'utf8', + pers: opt.pers, + persEnc: 'utf8' + }); + return drbg.generate(opt.size, 'hex'); + } + + const test = [ + { + entropy: 'totally random0123456789', + nonce: 'secret nonce', + pers: 'my drbg', + size: 32, + res: '018ec5f8e08c41e5ac974eb129ac297c5388ee1864324fa13d9b15cf98d9a157' + }, + { + entropy: 'totally random0123456789', + nonce: 'secret nonce', + pers: null, + size: 32, + res: 'ed5d61ecf0ef38258e62f03bbb49f19f2cd07ba5145a840d83b134d5963b3633' + } + ]; + for (let i = 0; i < test.length; i++) + assert.equal(doDrbg(test[i]), test[i].res); + }); + + describe('NIST vector', function() { + require('./fixtures/hmac-drbg-nist.json').forEach(function (opt) { + it('should not fail at ' + opt.name, function() { + const drbg = HmacDRBG({ + hash: hash.sha256, + entropy: opt.entropy, + nonce: opt.nonce, + pers: opt.pers + }); + + let last; + for (let i = 0; i < opt.add.length; i++) { + let add = opt.add[i]; + last = drbg.generate(opt.expected.length / 2, 'hex', add); + } + assert.equal(last, opt.expected); + }); + }); + }); + + describe('reseeding', function() { + it('should reseed', function() { + const entropy = 'totally random string with many chars that I typed ' + + 'in agony'; + const nonce = 'nonce'; + const pers = 'pers'; + + const original = HmacDRBG({ + hash: hash.sha256, + entropy, + nonce, + pers + }); + const reseeded = HmacDRBG({ + hash: hash.sha256, + entropy, + nonce, + pers + }); + + assert.strictEqual(original.generate(32, 'hex'), + reseeded.generate(32, 'hex')); + + reseeded.reseed('another absolutely random string'); + + assert.notEqual(original.generate(32, 'hex'), + reseeded.generate(32, 'hex')); + }); + }); +}); diff --git a/node_modules/hmac-drbg/test/fixtures/hmac-drbg-nist.json b/node_modules/hmac-drbg/test/fixtures/hmac-drbg-nist.json new file mode 100644 index 0000000..fce148a --- /dev/null +++ b/node_modules/hmac-drbg/test/fixtures/hmac-drbg-nist.json @@ -0,0 +1,332 @@ +[ + { + "name": "0", + "entropy": "ca851911349384bffe89de1cbdc46e6831e44d34a4fb935ee285dd14b71a7488", + "nonce": "659ba96c601dc69fc902940805ec0ca8", + "pers": null, + "add": [ + null, + null + ], + "expected": "e528e9abf2dece54d47c7e75e5fe302149f817ea9fb4bee6f4199697d04d5b89d54fbb978a15b5c443c9ec21036d2460b6f73ebad0dc2aba6e624abf07745bc107694bb7547bb0995f70de25d6b29e2d3011bb19d27676c07162c8b5ccde0668961df86803482cb37ed6d5c0bb8d50cf1f50d476aa0458bdaba806f48be9dcb8" + }, + { + "name": "1", + "entropy": "79737479ba4e7642a221fcfd1b820b134e9e3540a35bb48ffae29c20f5418ea3", + "nonce": "3593259c092bef4129bc2c6c9e19f343", + "pers": null, + "add": [ + null, + null + ], + "expected": "cf5ad5984f9e43917aa9087380dac46e410ddc8a7731859c84e9d0f31bd43655b924159413e2293b17610f211e09f770f172b8fb693a35b85d3b9e5e63b1dc252ac0e115002e9bedfb4b5b6fd43f33b8e0eafb2d072e1a6fee1f159df9b51e6c8da737e60d5032dd30544ec51558c6f080bdbdab1de8a939e961e06b5f1aca37" + }, + { + "name": "2", + "entropy": "b340907445b97a8b589264de4a17c0bea11bb53ad72f9f33297f05d2879d898d", + "nonce": "65cb27735d83c0708f72684ea58f7ee5", + "pers": null, + "add": [ + null, + null + ], + "expected": "75183aaaf3574bc68003352ad655d0e9ce9dd17552723b47fab0e84ef903694a32987eeddbdc48efd24195dbdac8a46ba2d972f5808f23a869e71343140361f58b243e62722088fe10a98e43372d252b144e00c89c215a76a121734bdc485486f65c0b16b8963524a3a70e6f38f169c12f6cbdd169dd48fe4421a235847a23ff" + }, + { + "name": "3", + "entropy": "8e159f60060a7d6a7e6fe7c9f769c30b98acb1240b25e7ee33f1da834c0858e7", + "nonce": "c39d35052201bdcce4e127a04f04d644", + "pers": null, + "add": [ + null, + null + ], + "expected": "62910a77213967ea93d6457e255af51fc79d49629af2fccd81840cdfbb4910991f50a477cbd29edd8a47c4fec9d141f50dfde7c4d8fcab473eff3cc2ee9e7cc90871f180777a97841597b0dd7e779eff9784b9cc33689fd7d48c0dcd341515ac8fecf5c55a6327aea8d58f97220b7462373e84e3b7417a57e80ce946d6120db5" + }, + { + "name": "4", + "entropy": "74755f196305f7fb6689b2fe6835dc1d81484fc481a6b8087f649a1952f4df6a", + "nonce": "c36387a544a5f2b78007651a7b74b749", + "pers": null, + "add": [ + null, + null + ], + "expected": "b2896f3af4375dab67e8062d82c1a005ef4ed119d13a9f18371b1b873774418684805fd659bfd69964f83a5cfe08667ddad672cafd16befffa9faed49865214f703951b443e6dca22edb636f3308380144b9333de4bcb0735710e4d9266786342fc53babe7bdbe3c01a3addb7f23c63ce2834729fabbd419b47beceb4a460236" + }, + { + "name": "5", + "entropy": "4b222718f56a3260b3c2625a4cf80950b7d6c1250f170bd5c28b118abdf23b2f", + "nonce": "7aed52d0016fcaef0b6492bc40bbe0e9", + "pers": null, + "add": [ + null, + null + ], + "expected": "a6da029b3665cd39fd50a54c553f99fed3626f4902ffe322dc51f0670dfe8742ed48415cf04bbad5ed3b23b18b7892d170a7dcf3ef8052d5717cb0c1a8b3010d9a9ea5de70ae5356249c0e098946030c46d9d3d209864539444374d8fbcae068e1d6548fa59e6562e6b2d1acbda8da0318c23752ebc9be0c1c1c5b3cf66dd967" + }, + { + "name": "6", + "entropy": "b512633f27fb182a076917e39888ba3ff35d23c3742eb8f3c635a044163768e0", + "nonce": "e2c39b84629a3de5c301db5643af1c21", + "pers": null, + "add": [ + null, + null + ], + "expected": "fb931d0d0194a97b48d5d4c231fdad5c61aedf1c3a55ac24983ecbf38487b1c93396c6b86ff3920cfa8c77e0146de835ea5809676e702dee6a78100da9aa43d8ec0bf5720befa71f82193205ac2ea403e8d7e0e6270b366dc4200be26afd9f63b7e79286a35c688c57cbff55ac747d4c28bb80a2b2097b3b62ea439950d75dff" + }, + { + "name": "7", + "entropy": "aae3ffc8605a975befefcea0a7a286642bc3b95fb37bd0eb0585a4cabf8b3d1e", + "nonce": "9504c3c0c4310c1c0746a036c91d9034", + "pers": null, + "add": [ + null, + null + ], + "expected": "2819bd3b0d216dad59ddd6c354c4518153a2b04374b07c49e64a8e4d055575dfbc9a8fcde68bd257ff1ba5c6000564b46d6dd7ecd9c5d684fd757df62d85211575d3562d7814008ab5c8bc00e7b5a649eae2318665b55d762de36eba00c2906c0e0ec8706edb493e51ca5eb4b9f015dc932f262f52a86b11c41e9a6d5b3bd431" + }, + { + "name": "8", + "entropy": "b9475210b79b87180e746df704b3cbc7bf8424750e416a7fbb5ce3ef25a82cc6", + "nonce": "24baf03599c10df6ef44065d715a93f7", + "pers": null, + "add": [ + null, + null + ], + "expected": "ae12d784f796183c50db5a1a283aa35ed9a2b685dacea97c596ff8c294906d1b1305ba1f80254eb062b874a8dfffa3378c809ab2869aa51a4e6a489692284a25038908a347342175c38401193b8afc498077e10522bec5c70882b7f760ea5946870bd9fc72961eedbe8bff4fd58c7cc1589bb4f369ed0d3bf26c5bbc62e0b2b2" + }, + { + "name": "9", + "entropy": "27838eb44ceccb4e36210703ebf38f659bc39dd3277cd76b7a9bcd6bc964b628", + "nonce": "39cfe0210db2e7b0eb52a387476e7ea1", + "pers": null, + "add": [ + null, + null + ], + "expected": "e5e72a53605d2aaa67832f97536445ab774dd9bff7f13a0d11fd27bf6593bfb52309f2d4f09d147192199ea584503181de87002f4ee085c7dc18bf32ce5315647a3708e6f404d6588c92b2dda599c131aa350d18c747b33dc8eda15cf40e95263d1231e1b4b68f8d829f86054d49cfdb1b8d96ab0465110569c8583a424a099a" + }, + { + "name": "10", + "entropy": "d7129e4f47008ad60c9b5d081ff4ca8eb821a6e4deb91608bf4e2647835373a5", + "nonce": "a72882773f78c2fc4878295840a53012", + "pers": null, + "add": [ + null, + null + ], + "expected": "0cbf48585c5de9183b7ff76557f8fc9ebcfdfde07e588a8641156f61b7952725bbee954f87e9b937513b16bba0f2e523d095114658e00f0f3772175acfcb3240a01de631c19c5a834c94cc58d04a6837f0d2782fa53d2f9f65178ee9c837222494c799e64c60406069bd319549b889fa00a0032dd7ba5b1cc9edbf58de82bfcd" + }, + { + "name": "11", + "entropy": "67fe5e300c513371976c80de4b20d4473889c9f1214bce718bc32d1da3ab7532", + "nonce": "e256d88497738a33923aa003a8d7845c", + "pers": null, + "add": [ + null, + null + ], + "expected": "b44660d64ef7bcebc7a1ab71f8407a02285c7592d755ae6766059e894f694373ed9c776c0cfc8594413eefb400ed427e158d687e28da3ecc205e0f7370fb089676bbb0fa591ec8d916c3d5f18a3eb4a417120705f3e2198154cd60648dbfcfc901242e15711cacd501b2c2826abe870ba32da785ed6f1fdc68f203d1ab43a64f" + }, + { + "name": "12", + "entropy": "de8142541255c46d66efc6173b0fe3ffaf5936c897a3ce2e9d5835616aafa2cb", + "nonce": "d01f9002c407127bc3297a561d89b81d", + "pers": null, + "add": [ + null, + null + ], + "expected": "64d1020929d74716446d8a4e17205d0756b5264867811aa24d0d0da8644db25d5cde474143c57d12482f6bf0f31d10af9d1da4eb6d701bdd605a8db74fb4e77f79aaa9e450afda50b18d19fae68f03db1d7b5f1738d2fdce9ad3ee9461b58ee242daf7a1d72c45c9213eca34e14810a9fca5208d5c56d8066bab1586f1513de7" + }, + { + "name": "13", + "entropy": "4a8e0bd90bdb12f7748ad5f147b115d7385bb1b06aee7d8b76136a25d779bcb7", + "nonce": "7f3cce4af8c8ce3c45bdf23c6b181a00", + "pers": null, + "add": [ + null, + null + ], + "expected": "320c7ca4bbeb7af977bc054f604b5086a3f237aa5501658112f3e7a33d2231f5536d2c85c1dad9d9b0bf7f619c81be4854661626839c8c10ae7fdc0c0b571be34b58d66da553676167b00e7d8e49f416aacb2926c6eb2c66ec98bffae20864cf92496db15e3b09e530b7b9648be8d3916b3c20a3a779bec7d66da63396849aaf" + }, + { + "name": "14", + "entropy": "451ed024bc4b95f1025b14ec3616f5e42e80824541dc795a2f07500f92adc665", + "nonce": "2f28e6ee8de5879db1eccd58c994e5f0", + "pers": null, + "add": [ + null, + null + ], + "expected": "3fb637085ab75f4e95655faae95885166a5fbb423bb03dbf0543be063bcd48799c4f05d4e522634d9275fe02e1edd920e26d9accd43709cb0d8f6e50aa54a5f3bdd618be23cf73ef736ed0ef7524b0d14d5bef8c8aec1cf1ed3e1c38a808b35e61a44078127c7cb3a8fd7addfa50fcf3ff3bc6d6bc355d5436fe9b71eb44f7fd" + }, + { + "name": "0 with additional data", + "entropy": "d3cc4d1acf3dde0c4bd2290d262337042dc632948223d3a2eaab87da44295fbd", + "nonce": "0109b0e729f457328aa18569a9224921", + "pers": null, + "add": [ + "3c311848183c9a212a26f27f8c6647e40375e466a0857cc39c4e47575d53f1f6", + "fcb9abd19ccfbccef88c9c39bfb3dd7b1c12266c9808992e305bc3cff566e4e4" + ], + "expected": "9c7b758b212cd0fcecd5daa489821712e3cdea4467b560ef5ddc24ab47749a1f1ffdbbb118f4e62fcfca3371b8fbfc5b0646b83e06bfbbab5fac30ea09ea2bc76f1ea568c9be0444b2cc90517b20ca825f2d0eccd88e7175538b85d90ab390183ca6395535d34473af6b5a5b88f5a59ee7561573337ea819da0dcc3573a22974" + }, + { + "name": "1 with additional data", + "entropy": "f97a3cfd91faa046b9e61b9493d436c4931f604b22f1081521b3419151e8ff06", + "nonce": "11f3a7d43595357d58120bd1e2dd8aed", + "pers": null, + "add": [ + "517289afe444a0fe5ed1a41dbbb5eb17150079bdd31e29cf2ff30034d8268e3b", + "88028d29ef80b4e6f0fe12f91d7449fe75062682e89c571440c0c9b52c42a6e0" + ], + "expected": "c6871cff0824fe55ea7689a52229886730450e5d362da5bf590dcf9acd67fed4cb32107df5d03969a66b1f6494fdf5d63d5b4d0d34ea7399a07d0116126d0d518c7c55ba46e12f62efc8fe28a51c9d428e6d371d7397ab319fc73ded4722e5b4f30004032a6128df5e7497ecf82ca7b0a50e867ef6728a4f509a8c859087039c" + }, + { + "name": "2 with additional data", + "entropy": "0f2f23d64f481cabec7abb01db3aabf125c3173a044b9bf26844300b69dcac8b", + "nonce": "9a5ae13232b43aa19cfe8d7958b4b590", + "pers": null, + "add": [ + "ec4c7a62acab73385f567da10e892ff395a0929f959231a5628188ce0c26e818", + "6b97b8c6b6bb8935e676c410c17caa8042aa3145f856d0a32b641e4ae5298648" + ], + "expected": "7480a361058bd9afa3db82c9d7586e42269102013f6ec5c269b6d05f17987847748684766b44918fd4b65e1648622fc0e0954178b0279dfc9fa99b66c6f53e51c4860131e9e0644287a4afe4ca8e480417e070db68008a97c3397e4b320b5d1a1d7e1d18a95cfedd7d1e74997052bf649d132deb9ec53aae7dafdab55e6dae93" + }, + { + "name": "3 with additional data", + "entropy": "53c56660c78481be9c63284e005fcc14fbc7fb27732c9bf1366d01a426765a31", + "nonce": "dc7a14d0eb5b0b3534e717a0b3c64614", + "pers": null, + "add": [ + "3aa848706ecb877f5bedf4ffc332d57c22e08747a47e75cff6f0fd1316861c95", + "9a401afa739b8f752fddacd291e0b854f5eff4a55b515e20cb319852189d3722" + ], + "expected": "5c0eb420e0bf41ce9323e815310e4e8303cd677a8a8b023f31f0d79f0ca15aeb636099a369fd074d69889865eac1b72ab3cbfebdb8cf460b00072802e2ec648b1349a5303be4ccaadd729f1a9ea17482fd026aaeb93f1602bc1404b9853adde40d6c34b844cf148bc088941ecfc1642c8c0b9778e45f3b07e06e21ee2c9e0300" + }, + { + "name": "4 with additional data", + "entropy": "f63c804404902db334c54bb298fc271a21d7acd9f770278e089775710bf4fdd7", + "nonce": "3e45009ea9cb2a36ba1aa4bf39178200", + "pers": null, + "add": [ + "d165a13dc8cc43f3f0952c3f5d3de4136954d983683d4a3e6d2dc4c89bf23423", + "75106bc86d0336df85097f6af8e80e2da59046a03fa65b06706b8bbc7ffc6785" + ], + "expected": "6363139bba32c22a0f5cd23ca6d437b5669b7d432f786b8af445471bee0b2d24c9d5f2f93717cbe00d1f010cc3b9c515fc9f7336d53d4d26ba5c0d76a90186663c8582eb739c7b6578a3328bf68dc2cec2cd89b3a90201f6993adcc854df0f5c6974d0f5570765a15fe03dbce28942dd2fd16ba2027e68abac83926969349af8" + }, + { + "name": "5 with additional data", + "entropy": "2aaca9147da66c176615726b69e3e851cc3537f5f279fe7344233d8e44cfc99d", + "nonce": "4e171f080af9a6081bee9f183ac9e340", + "pers": null, + "add": [ + "d75a2a6eb66c3833e50f5ec3d2e434cf791448d618026d0c360806d120ded669", + "b643b74c15b37612e6577ed7ca2a4c67a78d560af9eb50a4108fca742e87b8d6" + ], + "expected": "501dcdc977f4ba856f24eaa4968b374bebb3166b280334cb510232c31ebffde10fa47b7840ef3fe3b77725c2272d3a1d4219baf23e0290c622271edcced58838cf428f0517425d2e19e0d8c89377eecfc378245f283236fafa466c914b99672ceafab369e8889a0c866d8bd639db9fb797254262c6fd44cfa9045ad6340a60ef" + }, + { + "name": "6 with additional data", + "entropy": "a2e4cd48a5cf918d6f55942d95fcb4e8465cdc4f77b7c52b6fae5b16a25ca306", + "nonce": "bef036716440db6e6d333d9d760b7ca8", + "pers": null, + "add": [ + "bfa591c7287f3f931168f95e38869441d1f9a11035ad8ea625bb61b9ea17591c", + "c00c735463bca215adc372cb892b05e939bf669583341c06d4e31d0e5b363a37" + ], + "expected": "e7d136af69926a5421d4266ee0420fd729f2a4f7c295d3c966bdfa05268180b508b8a2852d1b3a06fd2ab3e13c54005123ef319f42d0c6d3a575e6e7e1496cb28aacadbcf83740fba8f35fcee04bb2ed8a51db3d3362b01094a62fb57e33c99a432f29fce6676cffbbcc05107e794e75e44a02d5e6d9d748c5fbff00a0178d65" + }, + { + "name": "7 with additional data", + "entropy": "95a67771cba69011a79776e713145d309edae56fad5fd6d41d83eaff89df6e5e", + "nonce": "be5b5164e31ecc51ba6f7c3c5199eb33", + "pers": null, + "add": [ + "065f693b229a7c4fd373cd15b3807552dd9bf98c5485cef361949d4e7d774b53", + "9afb62406f0e812c4f156d58b19a656c904813c1b4a45a0029ae7f50731f8014" + ], + "expected": "f61b61a6e79a41183e8ed6647899d2dc85cdaf5c3abf5c7f3bf37685946dc28f4923dc842f2d4326bd6ce0d50a84cb3ba869d72a36e246910eba6512ba36cd7ed3a5437c9245b00a344308c792b668b458d3c3e16dee2fbec41867da31084d46d8ec168de2148ef64fc5b72069abf5a6ada1ead2b7146bb793ff1c9c3690fa56" + }, + { + "name": "8 with additional data", + "entropy": "a459e1815cbca4514ec8094d5ab2414a557ba6fe10e613c345338d0521e4bf90", + "nonce": "62221392e2552e76cd0d36df6e6068eb", + "pers": null, + "add": [ + "0a3642b02b23b3ef62c701a63401124022f5b896de86dab6e6c7451497aa1dcc", + "c80514865901371c45ba92d9f95d50bb7c9dd1768cb3dfbc45b968da94965c6e" + ], + "expected": "464e6977b8adaef307c9623e41c357013249c9ffd77f405f3925cebb69f151ce8fbb6a277164002aee7858fc224f6499042aa1e6322deee9a5d133c31d640e12a7487c731ba03ad866a24675badb1d79220c40be689f79c2a0be93cb4dada3e0eac4ab140cb91998b6f11953e68f2319b050c40f71c34de9905ae41b2de1c2f6" + }, + { + "name": "9 with additional data", + "entropy": "252c2cad613e002478162861880979ee4e323025eebb6fb2e0aa9f200e28e0a1", + "nonce": "d001bc9a8f2c8c242e4369df0c191989", + "pers": null, + "add": [ + "9bcfc61cb2bc000034bb3db980eb47c76fb5ecdd40553eff113368d639b947fd", + "8b0565c767c2610ee0014582e9fbecb96e173005b60e9581503a6dca5637a26e" + ], + "expected": "e96c15fe8a60692b0a7d67171e0195ff6e1c87aab844221e71700d1bbee75feea695f6a740c9760bbe0e812ecf4061d8f0955bc0195e18c4fd1516ebca50ba6a6db86881737dbab8321707675479b87611db6af2c97ea361a5484555ead454defb1a64335de964fc803d40f3a6f057893d2afc25725754f4f00abc51920743dc" + }, + { + "name": "10 with additional data", + "entropy": "8be0ca6adc8b3870c9d69d6021bc1f1d8eb9e649073d35ee6c5aa0b7e56ad8a5", + "nonce": "9d1265f7d51fdb65377f1e6edd6ae0e4", + "pers": null, + "add": [ + "da86167ac997c406bb7979f423986a84ec6614d6caa7afc10aff0699a9b2cf7f", + "e4baa3c555950b53e2bfdba480cb4c94b59381bac1e33947e0c22e838a9534cf" + ], + "expected": "64384ecc4ea6b458efc227ca697eac5510092265520c0a0d8a0ccf9ed3ca9d58074671188c6a7ad16d0b050cdc072c125d7298d3a31d9f044a9ee40da0089a84fea28cc7f05f1716db952fad29a0e779635cb7a912a959be67be2f0a4170aace2981802e2ff6467e5b46f0ffbff3b42ba5935fd553c82482ac266acf1cd247d7" + }, + { + "name": "11 with additional data", + "entropy": "d43a75b6adf26d60322284cb12ac38327792442aa8f040f60a2f331b33ac4a8f", + "nonce": "0682f8b091f811afacaacaec9b04d279", + "pers": null, + "add": [ + "7fd3b8f512940da7de5d80199d9a7b42670c04a945775a3dba869546cbb9bc65", + "2575db20bc7aafc2a90a5dabab760db851d754777bc9f05616af1858b24ff3da" + ], + "expected": "0da7a8dc73c163014bf0841913d3067806456bbca6d5de92b85534c6545467313648d71ef17c923d090dc92cff8d4d1a9a2bb63e001dc2e8ab1a597999be3d6cf70ff63fee9985801395fbd4f4990430c4259fcae4fa1fcd73dc3187ccc102d04af7c07532885e5a226fc42809c48f22eecf4f6ab996ae4fcb144786957d9f41" + }, + { + "name": "12 with additional data", + "entropy": "64352f236af5d32067a529a8fd05ba00a338c9de306371a0b00c36e610a48d18", + "nonce": "df99ed2c7608c870624b962a5dc68acd", + "pers": null, + "add": [ + "da416335e7aaf60cf3d06fb438735ce796aad09034f8969c8f8c3f81e32fef24", + "a28c07c21a2297311adf172c19e83ca0a87731bdffb80548978d2d1cd82cf8a3" + ], + "expected": "132b9f25868729e3853d3c51f99a3b5fae6d4204bea70890daf62e042b776a526c8fb831b80a6d5d3f153237df1fd39b6fd9137963f5516d9cdd4e3f9195c46e9972c15d3edc6606e3368bde1594977fb88d0ca6e6f5f3d057ccadc7d7dab77dfc42658a1e972aa446b20d418286386a52dfc1c714d2ac548713268b0b709729" + }, + { + "name": "13 with additional data", + "entropy": "282f4d2e05a2cd30e9087f5633089389449f04bac11df718c90bb351cd3653a5", + "nonce": "90a7daf3c0de9ea286081efc4a684dfb", + "pers": null, + "add": [ + "2630b4ccc7271cc379cb580b0aaede3d3aa8c1c7ba002cf791f0752c3d739007", + "c31d69de499f1017be44e3d4fa77ecebc6a9b9934749fcf136f267b29115d2cc" + ], + "expected": "c899094520e0197c37b91dd50778e20a5b950decfb308d39f1db709447ae48f6101d9abe63a783fbb830eec1d359a5f61a2013728966d349213ee96382614aa4135058a967627183810c6622a2158cababe3b8ab99169c89e362108bf5955b4ffc47440f87e4bad0d36bc738e737e072e64d8842e7619f1be0af1141f05afe2d" + }, + { + "name": "14 with additional data", + "entropy": "13c752b9e745ce77bbc7c0dbda982313d3fe66f903e83ebd8dbe4ff0c11380e9", + "nonce": "f1a533095d6174164bd7c82532464ae7", + "pers": null, + "add": [ + "4f53db89b9ba7fc00767bc751fb8f3c103fe0f76acd6d5c7891ab15b2b7cf67c", + "582c2a7d34679088cca6bd28723c99aac07db46c332dc0153d1673256903b446" + ], + "expected": "6311f4c0c4cd1f86bd48349abb9eb930d4f63df5e5f7217d1d1b91a71d8a6938b0ad2b3e897bd7e3d8703db125fab30e03464fad41e5ddf5bf9aeeb5161b244468cfb26a9d956931a5412c97d64188b0da1bd907819c686f39af82e91cfeef0cbffb5d1e229e383bed26d06412988640706815a6e820796876f416653e464961" + } +] diff --git a/node_modules/inherits/LICENSE b/node_modules/inherits/LICENSE new file mode 100644 index 0000000..dea3013 --- /dev/null +++ b/node_modules/inherits/LICENSE @@ -0,0 +1,16 @@ +The ISC License + +Copyright (c) Isaac Z. Schlueter + +Permission to use, copy, modify, and/or distribute this software for any +purpose with or without fee is hereby granted, provided that the above +copyright notice and this permission notice appear in all copies. + +THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH +REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND +FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, +INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM +LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR +OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR +PERFORMANCE OF THIS SOFTWARE. + diff --git a/node_modules/inherits/README.md b/node_modules/inherits/README.md new file mode 100644 index 0000000..b1c5665 --- /dev/null +++ b/node_modules/inherits/README.md @@ -0,0 +1,42 @@ +Browser-friendly inheritance fully compatible with standard node.js +[inherits](http://nodejs.org/api/util.html#util_util_inherits_constructor_superconstructor). + +This package exports standard `inherits` from node.js `util` module in +node environment, but also provides alternative browser-friendly +implementation through [browser +field](https://gist.github.com/shtylman/4339901). Alternative +implementation is a literal copy of standard one located in standalone +module to avoid requiring of `util`. It also has a shim for old +browsers with no `Object.create` support. + +While keeping you sure you are using standard `inherits` +implementation in node.js environment, it allows bundlers such as +[browserify](https://github.com/substack/node-browserify) to not +include full `util` package to your client code if all you need is +just `inherits` function. It worth, because browser shim for `util` +package is large and `inherits` is often the single function you need +from it. + +It's recommended to use this package instead of +`require('util').inherits` for any code that has chances to be used +not only in node.js but in browser too. + +## usage + +```js +var inherits = require('inherits'); +// then use exactly as the standard one +``` + +## note on version ~1.0 + +Version ~1.0 had completely different motivation and is not compatible +neither with 2.0 nor with standard node.js `inherits`. + +If you are using version ~1.0 and planning to switch to ~2.0, be +careful: + +* new version uses `super_` instead of `super` for referencing + superclass +* new version overwrites current prototype while old one preserves any + existing fields on it diff --git a/node_modules/inherits/inherits.js b/node_modules/inherits/inherits.js new file mode 100644 index 0000000..f71f2d9 --- /dev/null +++ b/node_modules/inherits/inherits.js @@ -0,0 +1,9 @@ +try { + var util = require('util'); + /* istanbul ignore next */ + if (typeof util.inherits !== 'function') throw ''; + module.exports = util.inherits; +} catch (e) { + /* istanbul ignore next */ + module.exports = require('./inherits_browser.js'); +} diff --git a/node_modules/inherits/inherits_browser.js b/node_modules/inherits/inherits_browser.js new file mode 100644 index 0000000..86bbb3d --- /dev/null +++ b/node_modules/inherits/inherits_browser.js @@ -0,0 +1,27 @@ +if (typeof Object.create === 'function') { + // implementation from standard node.js 'util' module + module.exports = function inherits(ctor, superCtor) { + if (superCtor) { + ctor.super_ = superCtor + ctor.prototype = Object.create(superCtor.prototype, { + constructor: { + value: ctor, + enumerable: false, + writable: true, + configurable: true + } + }) + } + }; +} else { + // old school shim for old browsers + module.exports = function inherits(ctor, superCtor) { + if (superCtor) { + ctor.super_ = superCtor + var TempCtor = function () {} + TempCtor.prototype = superCtor.prototype + ctor.prototype = new TempCtor() + ctor.prototype.constructor = ctor + } + } +} diff --git a/node_modules/inherits/package.json b/node_modules/inherits/package.json new file mode 100644 index 0000000..37b4366 --- /dev/null +++ b/node_modules/inherits/package.json @@ -0,0 +1,29 @@ +{ + "name": "inherits", + "description": "Browser-friendly inheritance fully compatible with standard node.js inherits()", + "version": "2.0.4", + "keywords": [ + "inheritance", + "class", + "klass", + "oop", + "object-oriented", + "inherits", + "browser", + "browserify" + ], + "main": "./inherits.js", + "browser": "./inherits_browser.js", + "repository": "git://github.com/isaacs/inherits", + "license": "ISC", + "scripts": { + "test": "tap" + }, + "devDependencies": { + "tap": "^14.2.4" + }, + "files": [ + "inherits.js", + "inherits_browser.js" + ] +} diff --git a/node_modules/js-sha256/CHANGELOG.md b/node_modules/js-sha256/CHANGELOG.md new file mode 100644 index 0000000..9b3964a --- /dev/null +++ b/node_modules/js-sha256/CHANGELOG.md @@ -0,0 +1,129 @@ +# Change Log + +## v0.11.0 / 2024-01-24 +### Fixed +- Generates incorrect hash in some cases #43 +- dependencies and security issues. #41 + +## v0.10.1 / 2023-08-31 +### Added +- Disable webpack polyfill. + +## v0.10.0 / 2023-08-30 +### Fixed +- Chrome bug by workaround. #40 +- deprecated `new Buffer`, replace with `Buffer.from`. #34 +- dependencies and security issues. #32, #36 + +### Changed +- TypeScript interface, secretKey can be bytes like message. #23, #25 +- remove `eval` and use `require` directly. #18, #26 + +## v0.9.0 / 2017-12-18 +### Fixed +- incorrect result when first bit is 1 of bytes. + +### Changed +- throw error by Error oject. #13 + +### Added +- TypeScript interfaces. #12 + +## v0.8.0 / 2017-11-19 +### Added +- support for web worker. +- typescript types. #10 + +### Changed +- prevent webpack to require dependencies. + +## v0.7.1 / 2017-10-31 +### Improved +- performance of hBytes increment. + +## v0.7.0 / 2017-10-31 +### Fixed +- incorrect result when file size >= 512M. + +## v0.6.0 / 2017-07-11 +### Added +- HMAC feature. + +### Changed +- throw error if input type is incorrect. + +## v0.5.0 / 2017-01-24 +### Added +- Streaming support #6 + +## v0.4.0 / 2017-01-23 +### Added +- AMD support. + +### Fixed +- ArrayBuffer dosen't work in Webpack. + +## v0.3.2 / 2016-09-12 +### Added +- CommonJS detection. + +## v0.3.1 / 2016-09-08 +### Added +- some files to npm package. + +### Fixed +- coding style. + +## v0.3.0 / 2015-05-23 +### Added +- support for ArrayBuffer input. + +## v0.2.3 / 2015-02-11 +### Added +- support for byte array input. + +## v0.2.2 / 2015-02-10 +### Improved +- performance. + +## v0.2.1 / 2015-02-05 +### Fixed +- special length bug. + +### Added +- test cases. + +## v0.2.0 / 2015-02-03 +### Removed +- ascii parameter. + +### Improved +- performance. + +### Added +- test cases. + +## v0.1.4 / 2015-01-24 +### Improved +- performance. + +## v0.1.3 / 2015-01-09 +### Improved +- performance. + +## v0.1.2 / 2015-01-06 +### Added +- bower package. +- travis. +- coveralls. + +### Fixed +- JSHint warnings. + +## v0.1.1 / 2014-07-27 +### Fixed +- accents bug. + +## v0.1.0 / 2014-01-05 +### Added +- initial release. diff --git a/node_modules/js-sha256/LICENSE.txt b/node_modules/js-sha256/LICENSE.txt new file mode 100644 index 0000000..dd24c23 --- /dev/null +++ b/node_modules/js-sha256/LICENSE.txt @@ -0,0 +1,22 @@ +Copyright (c) 2014-2024 Chen, Yi-Cyuan + +MIT License + +Permission is hereby granted, free of charge, to any person obtaining +a copy of this software and associated documentation files (the +"Software"), to deal in the Software without restriction, including +without limitation the rights to use, copy, modify, merge, publish, +distribute, sublicense, and/or sell copies of the Software, and to +permit persons to whom the Software is furnished to do so, subject to +the following conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE +LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION +OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION +WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/node_modules/js-sha256/README.md b/node_modules/js-sha256/README.md new file mode 100644 index 0000000..95a611b --- /dev/null +++ b/node_modules/js-sha256/README.md @@ -0,0 +1,113 @@ +# js-sha256 +[![Build Status](https://travis-ci.org/emn178/js-sha256.svg?branch=master)](https://travis-ci.org/emn178/js-sha256) +[![Coverage Status](https://coveralls.io/repos/emn178/js-sha256/badge.svg?branch=master)](https://coveralls.io/r/emn178/js-sha256?branch=master) +[![CDNJS](https://img.shields.io/cdnjs/v/js-sha256.svg)](https://cdnjs.com/libraries/js-sha256/) +[![NPM](https://nodei.co/npm/js-sha256.png?stars&downloads)](https://nodei.co/npm/js-sha256/) +A simple SHA-256 / SHA-224 hash function for JavaScript supports UTF-8 encoding. + +## Demo +[SHA256 Online](https://emn178.github.io/online-tools/sha256.html) +[SHA224 Online](https://emn178.github.io/online-tools/sha224.html) +[SHA256 File Online](https://emn178.github.io/online-tools/sha256_checksum.html) +[SHA225 File Online](https://emn178.github.io/online-tools/sha224_checksum.html) + +## Download +[Compress](https://raw.github.com/emn178/js-sha256/master/build/sha256.min.js) +[Uncompress](https://raw.github.com/emn178/js-sha256/master/src/sha256.js) + +## Installation +You can also install js-sha256 by using Bower. + + bower install js-sha256 + +For node.js, you can use this command to install: + + npm install js-sha256 + +## Usage +You could use like this: +```JavaScript +sha256('Message to hash'); +sha224('Message to hash'); + +var hash = sha256.create(); +hash.update('Message to hash'); +hash.hex(); + +var hash2 = sha256.update('Message to hash'); +hash2.update('Message2 to hash'); +hash2.array(); + +// HMAC +sha256.hmac('key', 'Message to hash'); +sha224.hmac('key', 'Message to hash'); + +var hash = sha256.hmac.create('key'); +hash.update('Message to hash'); +hash.hex(); + +var hash2 = sha256.hmac.update('key', 'Message to hash'); +hash2.update('Message2 to hash'); +hash2.array(); +``` + +### Node.js +If you use node.js, you should require the module first: +```JavaScript +var sha256 = require('js-sha256'); +``` +or +```JavaScript +var sha256 = require('js-sha256').sha256; +var sha224 = require('js-sha256').sha224; +``` +or +```JavaScript +const { sha256, sha224 } = require('js-sha256'); +``` +### TypeScript +If you use TypeScript, you can import like this: +```TypeScript +import { sha256, sha224 } from 'js-sha256'; +``` +### RequireJS +It supports AMD: +```JavaScript +require(['your/path/sha256.js'], function(sha256) { +// ... +}); +``` +## Example +```JavaScript +sha256(''); // e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855 +sha256('The quick brown fox jumps over the lazy dog'); // d7a8fbb307d7809469ca9abcb0082e4f8d5651e46d3cdb762d02d0bf37c9e592 +sha256('The quick brown fox jumps over the lazy dog.'); // ef537f25c895bfa782526529a9b63d97aa631564d5d789c2b765448c8635fb6c +sha224(''); // d14a028c2a3a2bc9476102bb288234c415a2b01f828ea62ac5b3e42f +sha224('The quick brown fox jumps over the lazy dog'); // 730e109bd7a8a32b1cb9d9a09aa2325d2430587ddbc0c38bad911525 +sha224('The quick brown fox jumps over the lazy dog.'); // 619cba8e8e05826e9b8c519c0a5c68f4fb653e8a3d8aa04bb2c8cd4c + +// It also supports UTF-8 encoding +sha256('中文'); // 72726d8818f693066ceb69afa364218b692e62ea92b385782363780f47529c21 +sha224('中文'); // dfbab71afdf54388af4d55f8bd3de8c9b15e0eb916bf9125f4a959d4 + +// It also supports byte `Array`, `Uint8Array`, `ArrayBuffer` input +sha256([]); // e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855 +sha256(new Uint8Array([211, 212])); // 182889f925ae4e5cc37118ded6ed87f7bdc7cab5ec5e78faef2e50048999473f + +// Different output +sha256(''); // e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855 +sha256.hex(''); // e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855 +sha256.array(''); // [227, 176, 196, 66, 152, 252, 28, 20, 154, 251, 244, 200, 153, 111, 185, 36, 39, 174, 65, 228, 100, 155, 147, 76, 164, 149, 153, 27, 120, 82, 184, 85] +sha256.digest(''); // [227, 176, 196, 66, 152, 252, 28, 20, 154, 251, 244, 200, 153, 111, 185, 36, 39, 174, 65, 228, 100, 155, 147, 76, 164, 149, 153, 27, 120, 82, 184, 85] +sha256.arrayBuffer(''); // ArrayBuffer +``` + +## Benchmark +[jsPerf Benchmark](https://jsperf.app/cidole/2) + +## License +The project is released under the [MIT license](https://opensource.org/license/mit/). + +## Contact +The project's website is located at https://github.com/emn178/js-sha256 +Author: Chen, Yi-Cyuan (emn178@gmail.com) diff --git a/node_modules/js-sha256/build/sha256.min.js b/node_modules/js-sha256/build/sha256.min.js new file mode 100644 index 0000000..7fe9102 --- /dev/null +++ b/node_modules/js-sha256/build/sha256.min.js @@ -0,0 +1,9 @@ +/** + * [js-sha256]{@link https://github.com/emn178/js-sha256} + * + * @version 0.11.0 + * @author Chen, Yi-Cyuan [emn178@gmail.com] + * @copyright Chen, Yi-Cyuan 2014-2024 + * @license MIT + */ +!function(){"use strict";function t(t,i){i?(d[0]=d[16]=d[1]=d[2]=d[3]=d[4]=d[5]=d[6]=d[7]=d[8]=d[9]=d[10]=d[11]=d[12]=d[13]=d[14]=d[15]=0,this.blocks=d):this.blocks=[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],t?(this.h0=3238371032,this.h1=914150663,this.h2=812702999,this.h3=4144912697,this.h4=4290775857,this.h5=1750603025,this.h6=1694076839,this.h7=3204075428):(this.h0=1779033703,this.h1=3144134277,this.h2=1013904242,this.h3=2773480762,this.h4=1359893119,this.h5=2600822924,this.h6=528734635,this.h7=1541459225),this.block=this.start=this.bytes=this.hBytes=0,this.finalized=this.hashed=!1,this.first=!0,this.is224=t}function i(i,r,s){var e,n=typeof i;if("string"===n){var o,a=[],u=i.length,c=0;for(e=0;e>>6,a[c++]=128|63&o):o<55296||o>=57344?(a[c++]=224|o>>>12,a[c++]=128|o>>>6&63,a[c++]=128|63&o):(o=65536+((1023&o)<<10|1023&i.charCodeAt(++e)),a[c++]=240|o>>>18,a[c++]=128|o>>>12&63,a[c++]=128|o>>>6&63,a[c++]=128|63&o);i=a}else{if("object"!==n)throw new Error(h);if(null===i)throw new Error(h);if(f&&i.constructor===ArrayBuffer)i=new Uint8Array(i);else if(!(Array.isArray(i)||f&&ArrayBuffer.isView(i)))throw new Error(h)}i.length>64&&(i=new t(r,!0).update(i).array());var y=[],p=[];for(e=0;e<64;++e){var l=i[e]||0;y[e]=92^l,p[e]=54^l}t.call(this,r,s),this.update(p),this.oKeyPad=y,this.inner=!0,this.sharedMemory=s}var h="input is invalid type",r="object"==typeof window,s=r?window:{};s.JS_SHA256_NO_WINDOW&&(r=!1);var e=!r&&"object"==typeof self,n=!s.JS_SHA256_NO_NODE_JS&&"object"==typeof process&&process.versions&&process.versions.node;n?s=global:e&&(s=self);var o=!s.JS_SHA256_NO_COMMON_JS&&"object"==typeof module&&module.exports,a="function"==typeof define&&define.amd,f=!s.JS_SHA256_NO_ARRAY_BUFFER&&"undefined"!=typeof ArrayBuffer,u="0123456789abcdef".split(""),c=[-2147483648,8388608,32768,128],y=[24,16,8,0],p=[1116352408,1899447441,3049323471,3921009573,961987163,1508970993,2453635748,2870763221,3624381080,310598401,607225278,1426881987,1925078388,2162078206,2614888103,3248222580,3835390401,4022224774,264347078,604807628,770255983,1249150122,1555081692,1996064986,2554220882,2821834349,2952996808,3210313671,3336571891,3584528711,113926993,338241895,666307205,773529912,1294757372,1396182291,1695183700,1986661051,2177026350,2456956037,2730485921,2820302411,3259730800,3345764771,3516065817,3600352804,4094571909,275423344,430227734,506948616,659060556,883997877,958139571,1322822218,1537002063,1747873779,1955562222,2024104815,2227730452,2361852424,2428436474,2756734187,3204031479,3329325298],l=["hex","array","digest","arrayBuffer"],d=[];!s.JS_SHA256_NO_NODE_JS&&Array.isArray||(Array.isArray=function(t){return"[object Array]"===Object.prototype.toString.call(t)}),!f||!s.JS_SHA256_NO_ARRAY_BUFFER_IS_VIEW&&ArrayBuffer.isView||(ArrayBuffer.isView=function(t){return"object"==typeof t&&t.buffer&&t.buffer.constructor===ArrayBuffer});var A=function(i,h){return function(r){return new t(h,!0).update(r)[i]()}},w=function(i){var h=A("hex",i);n&&(h=b(h,i)),h.create=function(){return new t(i)},h.update=function(t){return h.create().update(t)};for(var r=0;r>>2]|=t[n]<>>2]|=s<>>2]|=(192|s>>>6)<>>2]|=(128|63&s)<=57344?(a[e>>>2]|=(224|s>>>12)<>>2]|=(128|s>>>6&63)<>>2]|=(128|63&s)<>>2]|=(240|s>>>18)<>>2]|=(128|s>>>12&63)<>>2]|=(128|s>>>6&63)<>>2]|=(128|63&s)<=64?(this.block=a[16],this.start=e-64,this.hash(),this.hashed=!0):this.start=e}return this.bytes>4294967295&&(this.hBytes+=this.bytes/4294967296<<0,this.bytes=this.bytes%4294967296),this}},t.prototype.finalize=function(){if(!this.finalized){this.finalized=!0;var t=this.blocks,i=this.lastByteIndex;t[16]=this.block,t[i>>>2]|=c[3&i],this.block=t[16],i>=56&&(this.hashed||this.hash(),t[0]=this.block,t[16]=t[1]=t[2]=t[3]=t[4]=t[5]=t[6]=t[7]=t[8]=t[9]=t[10]=t[11]=t[12]=t[13]=t[14]=t[15]=0),t[14]=this.hBytes<<3|this.bytes>>>29,t[15]=this.bytes<<3,this.hash()}},t.prototype.hash=function(){var t,i,h,r,s,e,n,o,a,f=this.h0,u=this.h1,c=this.h2,y=this.h3,l=this.h4,d=this.h5,A=this.h6,w=this.h7,b=this.blocks;for(t=16;t<64;++t)i=((s=b[t-15])>>>7|s<<25)^(s>>>18|s<<14)^s>>>3,h=((s=b[t-2])>>>17|s<<15)^(s>>>19|s<<13)^s>>>10,b[t]=b[t-16]+i+b[t-7]+h<<0;for(a=u&c,t=0;t<64;t+=4)this.first?(this.is224?(e=300032,w=(s=b[0]-1413257819)-150054599<<0,y=s+24177077<<0):(e=704751109,w=(s=b[0]-210244248)-1521486534<<0,y=s+143694565<<0),this.first=!1):(i=(f>>>2|f<<30)^(f>>>13|f<<19)^(f>>>22|f<<10),r=(e=f&u)^f&c^a,w=y+(s=w+(h=(l>>>6|l<<26)^(l>>>11|l<<21)^(l>>>25|l<<7))+(l&d^~l&A)+p[t]+b[t])<<0,y=s+(i+r)<<0),i=(y>>>2|y<<30)^(y>>>13|y<<19)^(y>>>22|y<<10),r=(n=y&f)^y&u^e,A=c+(s=A+(h=(w>>>6|w<<26)^(w>>>11|w<<21)^(w>>>25|w<<7))+(w&l^~w&d)+p[t+1]+b[t+1])<<0,i=((c=s+(i+r)<<0)>>>2|c<<30)^(c>>>13|c<<19)^(c>>>22|c<<10),r=(o=c&y)^c&f^n,d=u+(s=d+(h=(A>>>6|A<<26)^(A>>>11|A<<21)^(A>>>25|A<<7))+(A&w^~A&l)+p[t+2]+b[t+2])<<0,i=((u=s+(i+r)<<0)>>>2|u<<30)^(u>>>13|u<<19)^(u>>>22|u<<10),r=(a=u&c)^u&y^o,l=f+(s=l+(h=(d>>>6|d<<26)^(d>>>11|d<<21)^(d>>>25|d<<7))+(d&A^~d&w)+p[t+3]+b[t+3])<<0,f=s+(i+r)<<0,this.chromeBugWorkAround=!0;this.h0=this.h0+f<<0,this.h1=this.h1+u<<0,this.h2=this.h2+c<<0,this.h3=this.h3+y<<0,this.h4=this.h4+l<<0,this.h5=this.h5+d<<0,this.h6=this.h6+A<<0,this.h7=this.h7+w<<0},t.prototype.hex=function(){this.finalize();var t=this.h0,i=this.h1,h=this.h2,r=this.h3,s=this.h4,e=this.h5,n=this.h6,o=this.h7,a=u[t>>>28&15]+u[t>>>24&15]+u[t>>>20&15]+u[t>>>16&15]+u[t>>>12&15]+u[t>>>8&15]+u[t>>>4&15]+u[15&t]+u[i>>>28&15]+u[i>>>24&15]+u[i>>>20&15]+u[i>>>16&15]+u[i>>>12&15]+u[i>>>8&15]+u[i>>>4&15]+u[15&i]+u[h>>>28&15]+u[h>>>24&15]+u[h>>>20&15]+u[h>>>16&15]+u[h>>>12&15]+u[h>>>8&15]+u[h>>>4&15]+u[15&h]+u[r>>>28&15]+u[r>>>24&15]+u[r>>>20&15]+u[r>>>16&15]+u[r>>>12&15]+u[r>>>8&15]+u[r>>>4&15]+u[15&r]+u[s>>>28&15]+u[s>>>24&15]+u[s>>>20&15]+u[s>>>16&15]+u[s>>>12&15]+u[s>>>8&15]+u[s>>>4&15]+u[15&s]+u[e>>>28&15]+u[e>>>24&15]+u[e>>>20&15]+u[e>>>16&15]+u[e>>>12&15]+u[e>>>8&15]+u[e>>>4&15]+u[15&e]+u[n>>>28&15]+u[n>>>24&15]+u[n>>>20&15]+u[n>>>16&15]+u[n>>>12&15]+u[n>>>8&15]+u[n>>>4&15]+u[15&n];return this.is224||(a+=u[o>>>28&15]+u[o>>>24&15]+u[o>>>20&15]+u[o>>>16&15]+u[o>>>12&15]+u[o>>>8&15]+u[o>>>4&15]+u[15&o]),a},t.prototype.toString=t.prototype.hex,t.prototype.digest=function(){this.finalize();var t=this.h0,i=this.h1,h=this.h2,r=this.h3,s=this.h4,e=this.h5,n=this.h6,o=this.h7,a=[t>>>24&255,t>>>16&255,t>>>8&255,255&t,i>>>24&255,i>>>16&255,i>>>8&255,255&i,h>>>24&255,h>>>16&255,h>>>8&255,255&h,r>>>24&255,r>>>16&255,r>>>8&255,255&r,s>>>24&255,s>>>16&255,s>>>8&255,255&s,e>>>24&255,e>>>16&255,e>>>8&255,255&e,n>>>24&255,n>>>16&255,n>>>8&255,255&n];return this.is224||a.push(o>>>24&255,o>>>16&255,o>>>8&255,255&o),a},t.prototype.array=t.prototype.digest,t.prototype.arrayBuffer=function(){this.finalize();var t=new ArrayBuffer(this.is224?28:32),i=new DataView(t);return i.setUint32(0,this.h0),i.setUint32(4,this.h1),i.setUint32(8,this.h2),i.setUint32(12,this.h3),i.setUint32(16,this.h4),i.setUint32(20,this.h5),i.setUint32(24,this.h6),this.is224||i.setUint32(28,this.h7),t},(i.prototype=new t).finalize=function(){if(t.prototype.finalize.call(this),this.inner){this.inner=!1;var i=this.array();t.call(this,this.is224,this.sharedMemory),this.update(this.oKeyPad),this.update(i),t.prototype.finalize.call(this)}};var B=w();B.sha256=B,B.sha224=w(!0),B.sha256.hmac=v(),B.sha224.hmac=v(!0),o?module.exports=B:(s.sha256=B.sha256,s.sha224=B.sha224,a&&define(function(){return B}))}(); \ No newline at end of file diff --git a/node_modules/js-sha256/index.d.ts b/node_modules/js-sha256/index.d.ts new file mode 100644 index 0000000..1ac455e --- /dev/null +++ b/node_modules/js-sha256/index.d.ts @@ -0,0 +1,149 @@ +type Message = string | number[] | ArrayBuffer | Uint8Array; + +interface Hasher { + /** + * Update hash + * + * @param message The message you want to hash. + */ + update(message: Message): Hasher; + + /** + * Return hash in hex string. + */ + hex(): string; + + /** + * Return hash in hex string. + */ + toString(): string; + + /** + * Return hash in ArrayBuffer. + */ + arrayBuffer(): ArrayBuffer; + + /** + * Return hash in integer array. + */ + digest(): number[]; + + /** + * Return hash in integer array. + */ + array(): number[]; +} + +interface Hmac { + /** + * Computes a Hash-based message authentication code (HMAC) using a secret key + * + * @param secretKey The Secret Key + * @param message The message you want to hash. + */ + (secretKey: Message, message: Message): string; + + /** + * Create a hash object using a secret key. + * + * @param secretKey The Secret Key + */ + create(secretKey: Message): Hasher; + + /** + * Create a hash object and hash message using a secret key + * + * @param secretKey The Secret Key + * @param message The message you want to hash. + */ + update(secretKey: Message, message: Message): Hasher; + + /** + * Return hash in hex string. + * + * @param secretKey The Secret Key + * @param message The message you want to hash. + */ + hex(secretKey: Message, message: Message): string; + + /** + * Return hash in ArrayBuffer. + * + * @param secretKey The Secret Key + * @param message The message you want to hash. + */ + arrayBuffer(secretKey: Message, message: Message): ArrayBuffer; + + /** + * Return hash in integer array. + * + * @param secretKey The Secret Key + * @param message The message you want to hash. + */ + digest(secretKey: Message, message: Message): number[]; + + /** + * Return hash in integer array. + * + * @param secretKey The Secret Key + * @param message The message you want to hash. + */ + array(secretKey: Message, message: Message): number[]; +} + +interface Hash { + /** + * Hash and return hex string. + * + * @param message The message you want to hash. + */ + (message: Message): string; + + /** + * Create a hash object. + */ + create(): Hasher; + + /** + * Create a hash object and hash message. + * + * @param message The message you want to hash. + */ + update(message: Message): Hasher; + + /** + * Return hash in hex string. + * + * @param message The message you want to hash. + */ + hex(message: Message): string; + + /** + * Return hash in ArrayBuffer. + * + * @param message The message you want to hash. + */ + arrayBuffer(message: Message): ArrayBuffer; + + /** + * Return hash in integer array. + * + * @param message The message you want to hash. + */ + digest(message: Message): number[]; + + /** + * Return hash in integer array. + * + * @param message The message you want to hash. + */ + array(message: Message): number[]; + + /** + * HMAC interface + */ + hmac: Hmac; +} + +export var sha256: Hash; +export var sha224: Hash; diff --git a/node_modules/js-sha256/package.json b/node_modules/js-sha256/package.json new file mode 100644 index 0000000..70d8a54 --- /dev/null +++ b/node_modules/js-sha256/package.json @@ -0,0 +1,48 @@ +{ + "name": "js-sha256", + "version": "0.11.0", + "description": "A simple SHA-256 / SHA-224 hash function for JavaScript supports UTF-8 encoding.", + "main": "src/sha256.js", + "devDependencies": { + "expect.js": "~0.3.1", + "mocha": "~10.2.0", + "nyc": "^15.1.0", + "tiny-worker": "^2.3.0", + "uglify-js": "^3.1.9" + }, + "scripts": { + "test": "nyc mocha tests/node-test.js", + "report": "nyc --reporter=html --reporter=text mocha tests/node-test.js", + "coveralls": "nyc report --reporter=text-lcov | coveralls", + "build": "uglifyjs src/sha256.js -c -m eval --comments -o build/sha256.min.js" + }, + "repository": { + "type": "git", + "url": "https://github.com/emn178/js-sha256.git" + }, + "keywords": [ + "sha", + "sha2", + "sha224", + "sha256", + "hash", + "encryption", + "cryptography", + "HMAC" + ], + "license": "MIT", + "author": "Chen, Yi-Cyuan ", + "homepage": "https://github.com/emn178/js-sha256", + "bugs": { + "url": "https://github.com/emn178/js-sha256/issues" + }, + "nyc": { + "exclude": [ + "tests" + ] + }, + "browser": { + "crypto": false, + "buffer": false + } +} diff --git a/node_modules/js-sha256/src/sha256.js b/node_modules/js-sha256/src/sha256.js new file mode 100644 index 0000000..d028388 --- /dev/null +++ b/node_modules/js-sha256/src/sha256.js @@ -0,0 +1,526 @@ +/** + * [js-sha256]{@link https://github.com/emn178/js-sha256} + * + * @version 0.11.0 + * @author Chen, Yi-Cyuan [emn178@gmail.com] + * @copyright Chen, Yi-Cyuan 2014-2024 + * @license MIT + */ +/*jslint bitwise: true */ +(function () { + 'use strict'; + + var ERROR = 'input is invalid type'; + var WINDOW = typeof window === 'object'; + var root = WINDOW ? window : {}; + if (root.JS_SHA256_NO_WINDOW) { + WINDOW = false; + } + var WEB_WORKER = !WINDOW && typeof self === 'object'; + var NODE_JS = !root.JS_SHA256_NO_NODE_JS && typeof process === 'object' && process.versions && process.versions.node; + if (NODE_JS) { + root = global; + } else if (WEB_WORKER) { + root = self; + } + var COMMON_JS = !root.JS_SHA256_NO_COMMON_JS && typeof module === 'object' && module.exports; + var AMD = typeof define === 'function' && define.amd; + var ARRAY_BUFFER = !root.JS_SHA256_NO_ARRAY_BUFFER && typeof ArrayBuffer !== 'undefined'; + var HEX_CHARS = '0123456789abcdef'.split(''); + var EXTRA = [-2147483648, 8388608, 32768, 128]; + var SHIFT = [24, 16, 8, 0]; + var K = [ + 0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, 0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5, + 0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3, 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174, + 0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc, 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da, + 0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, 0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967, + 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13, 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85, + 0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070, + 0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3, + 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208, 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2 + ]; + var OUTPUT_TYPES = ['hex', 'array', 'digest', 'arrayBuffer']; + + var blocks = []; + + if (root.JS_SHA256_NO_NODE_JS || !Array.isArray) { + Array.isArray = function (obj) { + return Object.prototype.toString.call(obj) === '[object Array]'; + }; + } + + if (ARRAY_BUFFER && (root.JS_SHA256_NO_ARRAY_BUFFER_IS_VIEW || !ArrayBuffer.isView)) { + ArrayBuffer.isView = function (obj) { + return typeof obj === 'object' && obj.buffer && obj.buffer.constructor === ArrayBuffer; + }; + } + + var createOutputMethod = function (outputType, is224) { + return function (message) { + return new Sha256(is224, true).update(message)[outputType](); + }; + }; + + var createMethod = function (is224) { + var method = createOutputMethod('hex', is224); + if (NODE_JS) { + method = nodeWrap(method, is224); + } + method.create = function () { + return new Sha256(is224); + }; + method.update = function (message) { + return method.create().update(message); + }; + for (var i = 0; i < OUTPUT_TYPES.length; ++i) { + var type = OUTPUT_TYPES[i]; + method[type] = createOutputMethod(type, is224); + } + return method; + }; + + var nodeWrap = function (method, is224) { + var crypto = require('crypto') + var Buffer = require('buffer').Buffer; + var algorithm = is224 ? 'sha224' : 'sha256'; + var bufferFrom; + if (Buffer.from && !root.JS_SHA256_NO_BUFFER_FROM) { + bufferFrom = Buffer.from; + } else { + bufferFrom = function (message) { + return new Buffer(message); + }; + } + var nodeMethod = function (message) { + if (typeof message === 'string') { + return crypto.createHash(algorithm).update(message, 'utf8').digest('hex'); + } else { + if (message === null || message === undefined) { + throw new Error(ERROR); + } else if (message.constructor === ArrayBuffer) { + message = new Uint8Array(message); + } + } + if (Array.isArray(message) || ArrayBuffer.isView(message) || + message.constructor === Buffer) { + return crypto.createHash(algorithm).update(bufferFrom(message)).digest('hex'); + } else { + return method(message); + } + }; + return nodeMethod; + }; + + var createHmacOutputMethod = function (outputType, is224) { + return function (key, message) { + return new HmacSha256(key, is224, true).update(message)[outputType](); + }; + }; + + var createHmacMethod = function (is224) { + var method = createHmacOutputMethod('hex', is224); + method.create = function (key) { + return new HmacSha256(key, is224); + }; + method.update = function (key, message) { + return method.create(key).update(message); + }; + for (var i = 0; i < OUTPUT_TYPES.length; ++i) { + var type = OUTPUT_TYPES[i]; + method[type] = createHmacOutputMethod(type, is224); + } + return method; + }; + + function Sha256(is224, sharedMemory) { + if (sharedMemory) { + blocks[0] = blocks[16] = blocks[1] = blocks[2] = blocks[3] = + blocks[4] = blocks[5] = blocks[6] = blocks[7] = + blocks[8] = blocks[9] = blocks[10] = blocks[11] = + blocks[12] = blocks[13] = blocks[14] = blocks[15] = 0; + this.blocks = blocks; + } else { + this.blocks = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]; + } + + if (is224) { + this.h0 = 0xc1059ed8; + this.h1 = 0x367cd507; + this.h2 = 0x3070dd17; + this.h3 = 0xf70e5939; + this.h4 = 0xffc00b31; + this.h5 = 0x68581511; + this.h6 = 0x64f98fa7; + this.h7 = 0xbefa4fa4; + } else { // 256 + this.h0 = 0x6a09e667; + this.h1 = 0xbb67ae85; + this.h2 = 0x3c6ef372; + this.h3 = 0xa54ff53a; + this.h4 = 0x510e527f; + this.h5 = 0x9b05688c; + this.h6 = 0x1f83d9ab; + this.h7 = 0x5be0cd19; + } + + this.block = this.start = this.bytes = this.hBytes = 0; + this.finalized = this.hashed = false; + this.first = true; + this.is224 = is224; + } + + Sha256.prototype.update = function (message) { + if (this.finalized) { + return; + } + var notString, type = typeof message; + if (type !== 'string') { + if (type === 'object') { + if (message === null) { + throw new Error(ERROR); + } else if (ARRAY_BUFFER && message.constructor === ArrayBuffer) { + message = new Uint8Array(message); + } else if (!Array.isArray(message)) { + if (!ARRAY_BUFFER || !ArrayBuffer.isView(message)) { + throw new Error(ERROR); + } + } + } else { + throw new Error(ERROR); + } + notString = true; + } + var code, index = 0, i, length = message.length, blocks = this.blocks; + while (index < length) { + if (this.hashed) { + this.hashed = false; + blocks[0] = this.block; + this.block = blocks[16] = blocks[1] = blocks[2] = blocks[3] = + blocks[4] = blocks[5] = blocks[6] = blocks[7] = + blocks[8] = blocks[9] = blocks[10] = blocks[11] = + blocks[12] = blocks[13] = blocks[14] = blocks[15] = 0; + } + + if (notString) { + for (i = this.start; index < length && i < 64; ++index) { + blocks[i >>> 2] |= message[index] << SHIFT[i++ & 3]; + } + } else { + for (i = this.start; index < length && i < 64; ++index) { + code = message.charCodeAt(index); + if (code < 0x80) { + blocks[i >>> 2] |= code << SHIFT[i++ & 3]; + } else if (code < 0x800) { + blocks[i >>> 2] |= (0xc0 | (code >>> 6)) << SHIFT[i++ & 3]; + blocks[i >>> 2] |= (0x80 | (code & 0x3f)) << SHIFT[i++ & 3]; + } else if (code < 0xd800 || code >= 0xe000) { + blocks[i >>> 2] |= (0xe0 | (code >>> 12)) << SHIFT[i++ & 3]; + blocks[i >>> 2] |= (0x80 | ((code >>> 6) & 0x3f)) << SHIFT[i++ & 3]; + blocks[i >>> 2] |= (0x80 | (code & 0x3f)) << SHIFT[i++ & 3]; + } else { + code = 0x10000 + (((code & 0x3ff) << 10) | (message.charCodeAt(++index) & 0x3ff)); + blocks[i >>> 2] |= (0xf0 | (code >>> 18)) << SHIFT[i++ & 3]; + blocks[i >>> 2] |= (0x80 | ((code >>> 12) & 0x3f)) << SHIFT[i++ & 3]; + blocks[i >>> 2] |= (0x80 | ((code >>> 6) & 0x3f)) << SHIFT[i++ & 3]; + blocks[i >>> 2] |= (0x80 | (code & 0x3f)) << SHIFT[i++ & 3]; + } + } + } + + this.lastByteIndex = i; + this.bytes += i - this.start; + if (i >= 64) { + this.block = blocks[16]; + this.start = i - 64; + this.hash(); + this.hashed = true; + } else { + this.start = i; + } + } + if (this.bytes > 4294967295) { + this.hBytes += this.bytes / 4294967296 << 0; + this.bytes = this.bytes % 4294967296; + } + return this; + }; + + Sha256.prototype.finalize = function () { + if (this.finalized) { + return; + } + this.finalized = true; + var blocks = this.blocks, i = this.lastByteIndex; + blocks[16] = this.block; + blocks[i >>> 2] |= EXTRA[i & 3]; + this.block = blocks[16]; + if (i >= 56) { + if (!this.hashed) { + this.hash(); + } + blocks[0] = this.block; + blocks[16] = blocks[1] = blocks[2] = blocks[3] = + blocks[4] = blocks[5] = blocks[6] = blocks[7] = + blocks[8] = blocks[9] = blocks[10] = blocks[11] = + blocks[12] = blocks[13] = blocks[14] = blocks[15] = 0; + } + blocks[14] = this.hBytes << 3 | this.bytes >>> 29; + blocks[15] = this.bytes << 3; + this.hash(); + }; + + Sha256.prototype.hash = function () { + var a = this.h0, b = this.h1, c = this.h2, d = this.h3, e = this.h4, f = this.h5, g = this.h6, + h = this.h7, blocks = this.blocks, j, s0, s1, maj, t1, t2, ch, ab, da, cd, bc; + + for (j = 16; j < 64; ++j) { + // rightrotate + t1 = blocks[j - 15]; + s0 = ((t1 >>> 7) | (t1 << 25)) ^ ((t1 >>> 18) | (t1 << 14)) ^ (t1 >>> 3); + t1 = blocks[j - 2]; + s1 = ((t1 >>> 17) | (t1 << 15)) ^ ((t1 >>> 19) | (t1 << 13)) ^ (t1 >>> 10); + blocks[j] = blocks[j - 16] + s0 + blocks[j - 7] + s1 << 0; + } + + bc = b & c; + for (j = 0; j < 64; j += 4) { + if (this.first) { + if (this.is224) { + ab = 300032; + t1 = blocks[0] - 1413257819; + h = t1 - 150054599 << 0; + d = t1 + 24177077 << 0; + } else { + ab = 704751109; + t1 = blocks[0] - 210244248; + h = t1 - 1521486534 << 0; + d = t1 + 143694565 << 0; + } + this.first = false; + } else { + s0 = ((a >>> 2) | (a << 30)) ^ ((a >>> 13) | (a << 19)) ^ ((a >>> 22) | (a << 10)); + s1 = ((e >>> 6) | (e << 26)) ^ ((e >>> 11) | (e << 21)) ^ ((e >>> 25) | (e << 7)); + ab = a & b; + maj = ab ^ (a & c) ^ bc; + ch = (e & f) ^ (~e & g); + t1 = h + s1 + ch + K[j] + blocks[j]; + t2 = s0 + maj; + h = d + t1 << 0; + d = t1 + t2 << 0; + } + s0 = ((d >>> 2) | (d << 30)) ^ ((d >>> 13) | (d << 19)) ^ ((d >>> 22) | (d << 10)); + s1 = ((h >>> 6) | (h << 26)) ^ ((h >>> 11) | (h << 21)) ^ ((h >>> 25) | (h << 7)); + da = d & a; + maj = da ^ (d & b) ^ ab; + ch = (h & e) ^ (~h & f); + t1 = g + s1 + ch + K[j + 1] + blocks[j + 1]; + t2 = s0 + maj; + g = c + t1 << 0; + c = t1 + t2 << 0; + s0 = ((c >>> 2) | (c << 30)) ^ ((c >>> 13) | (c << 19)) ^ ((c >>> 22) | (c << 10)); + s1 = ((g >>> 6) | (g << 26)) ^ ((g >>> 11) | (g << 21)) ^ ((g >>> 25) | (g << 7)); + cd = c & d; + maj = cd ^ (c & a) ^ da; + ch = (g & h) ^ (~g & e); + t1 = f + s1 + ch + K[j + 2] + blocks[j + 2]; + t2 = s0 + maj; + f = b + t1 << 0; + b = t1 + t2 << 0; + s0 = ((b >>> 2) | (b << 30)) ^ ((b >>> 13) | (b << 19)) ^ ((b >>> 22) | (b << 10)); + s1 = ((f >>> 6) | (f << 26)) ^ ((f >>> 11) | (f << 21)) ^ ((f >>> 25) | (f << 7)); + bc = b & c; + maj = bc ^ (b & d) ^ cd; + ch = (f & g) ^ (~f & h); + t1 = e + s1 + ch + K[j + 3] + blocks[j + 3]; + t2 = s0 + maj; + e = a + t1 << 0; + a = t1 + t2 << 0; + this.chromeBugWorkAround = true; + } + + this.h0 = this.h0 + a << 0; + this.h1 = this.h1 + b << 0; + this.h2 = this.h2 + c << 0; + this.h3 = this.h3 + d << 0; + this.h4 = this.h4 + e << 0; + this.h5 = this.h5 + f << 0; + this.h6 = this.h6 + g << 0; + this.h7 = this.h7 + h << 0; + }; + + Sha256.prototype.hex = function () { + this.finalize(); + + var h0 = this.h0, h1 = this.h1, h2 = this.h2, h3 = this.h3, h4 = this.h4, h5 = this.h5, + h6 = this.h6, h7 = this.h7; + + var hex = HEX_CHARS[(h0 >>> 28) & 0x0F] + HEX_CHARS[(h0 >>> 24) & 0x0F] + + HEX_CHARS[(h0 >>> 20) & 0x0F] + HEX_CHARS[(h0 >>> 16) & 0x0F] + + HEX_CHARS[(h0 >>> 12) & 0x0F] + HEX_CHARS[(h0 >>> 8) & 0x0F] + + HEX_CHARS[(h0 >>> 4) & 0x0F] + HEX_CHARS[h0 & 0x0F] + + HEX_CHARS[(h1 >>> 28) & 0x0F] + HEX_CHARS[(h1 >>> 24) & 0x0F] + + HEX_CHARS[(h1 >>> 20) & 0x0F] + HEX_CHARS[(h1 >>> 16) & 0x0F] + + HEX_CHARS[(h1 >>> 12) & 0x0F] + HEX_CHARS[(h1 >>> 8) & 0x0F] + + HEX_CHARS[(h1 >>> 4) & 0x0F] + HEX_CHARS[h1 & 0x0F] + + HEX_CHARS[(h2 >>> 28) & 0x0F] + HEX_CHARS[(h2 >>> 24) & 0x0F] + + HEX_CHARS[(h2 >>> 20) & 0x0F] + HEX_CHARS[(h2 >>> 16) & 0x0F] + + HEX_CHARS[(h2 >>> 12) & 0x0F] + HEX_CHARS[(h2 >>> 8) & 0x0F] + + HEX_CHARS[(h2 >>> 4) & 0x0F] + HEX_CHARS[h2 & 0x0F] + + HEX_CHARS[(h3 >>> 28) & 0x0F] + HEX_CHARS[(h3 >>> 24) & 0x0F] + + HEX_CHARS[(h3 >>> 20) & 0x0F] + HEX_CHARS[(h3 >>> 16) & 0x0F] + + HEX_CHARS[(h3 >>> 12) & 0x0F] + HEX_CHARS[(h3 >>> 8) & 0x0F] + + HEX_CHARS[(h3 >>> 4) & 0x0F] + HEX_CHARS[h3 & 0x0F] + + HEX_CHARS[(h4 >>> 28) & 0x0F] + HEX_CHARS[(h4 >>> 24) & 0x0F] + + HEX_CHARS[(h4 >>> 20) & 0x0F] + HEX_CHARS[(h4 >>> 16) & 0x0F] + + HEX_CHARS[(h4 >>> 12) & 0x0F] + HEX_CHARS[(h4 >>> 8) & 0x0F] + + HEX_CHARS[(h4 >>> 4) & 0x0F] + HEX_CHARS[h4 & 0x0F] + + HEX_CHARS[(h5 >>> 28) & 0x0F] + HEX_CHARS[(h5 >>> 24) & 0x0F] + + HEX_CHARS[(h5 >>> 20) & 0x0F] + HEX_CHARS[(h5 >>> 16) & 0x0F] + + HEX_CHARS[(h5 >>> 12) & 0x0F] + HEX_CHARS[(h5 >>> 8) & 0x0F] + + HEX_CHARS[(h5 >>> 4) & 0x0F] + HEX_CHARS[h5 & 0x0F] + + HEX_CHARS[(h6 >>> 28) & 0x0F] + HEX_CHARS[(h6 >>> 24) & 0x0F] + + HEX_CHARS[(h6 >>> 20) & 0x0F] + HEX_CHARS[(h6 >>> 16) & 0x0F] + + HEX_CHARS[(h6 >>> 12) & 0x0F] + HEX_CHARS[(h6 >>> 8) & 0x0F] + + HEX_CHARS[(h6 >>> 4) & 0x0F] + HEX_CHARS[h6 & 0x0F]; + if (!this.is224) { + hex += HEX_CHARS[(h7 >>> 28) & 0x0F] + HEX_CHARS[(h7 >>> 24) & 0x0F] + + HEX_CHARS[(h7 >>> 20) & 0x0F] + HEX_CHARS[(h7 >>> 16) & 0x0F] + + HEX_CHARS[(h7 >>> 12) & 0x0F] + HEX_CHARS[(h7 >>> 8) & 0x0F] + + HEX_CHARS[(h7 >>> 4) & 0x0F] + HEX_CHARS[h7 & 0x0F]; + } + return hex; + }; + + Sha256.prototype.toString = Sha256.prototype.hex; + + Sha256.prototype.digest = function () { + this.finalize(); + + var h0 = this.h0, h1 = this.h1, h2 = this.h2, h3 = this.h3, h4 = this.h4, h5 = this.h5, + h6 = this.h6, h7 = this.h7; + + var arr = [ + (h0 >>> 24) & 0xFF, (h0 >>> 16) & 0xFF, (h0 >>> 8) & 0xFF, h0 & 0xFF, + (h1 >>> 24) & 0xFF, (h1 >>> 16) & 0xFF, (h1 >>> 8) & 0xFF, h1 & 0xFF, + (h2 >>> 24) & 0xFF, (h2 >>> 16) & 0xFF, (h2 >>> 8) & 0xFF, h2 & 0xFF, + (h3 >>> 24) & 0xFF, (h3 >>> 16) & 0xFF, (h3 >>> 8) & 0xFF, h3 & 0xFF, + (h4 >>> 24) & 0xFF, (h4 >>> 16) & 0xFF, (h4 >>> 8) & 0xFF, h4 & 0xFF, + (h5 >>> 24) & 0xFF, (h5 >>> 16) & 0xFF, (h5 >>> 8) & 0xFF, h5 & 0xFF, + (h6 >>> 24) & 0xFF, (h6 >>> 16) & 0xFF, (h6 >>> 8) & 0xFF, h6 & 0xFF + ]; + if (!this.is224) { + arr.push((h7 >>> 24) & 0xFF, (h7 >>> 16) & 0xFF, (h7 >>> 8) & 0xFF, h7 & 0xFF); + } + return arr; + }; + + Sha256.prototype.array = Sha256.prototype.digest; + + Sha256.prototype.arrayBuffer = function () { + this.finalize(); + + var buffer = new ArrayBuffer(this.is224 ? 28 : 32); + var dataView = new DataView(buffer); + dataView.setUint32(0, this.h0); + dataView.setUint32(4, this.h1); + dataView.setUint32(8, this.h2); + dataView.setUint32(12, this.h3); + dataView.setUint32(16, this.h4); + dataView.setUint32(20, this.h5); + dataView.setUint32(24, this.h6); + if (!this.is224) { + dataView.setUint32(28, this.h7); + } + return buffer; + }; + + function HmacSha256(key, is224, sharedMemory) { + var i, type = typeof key; + if (type === 'string') { + var bytes = [], length = key.length, index = 0, code; + for (i = 0; i < length; ++i) { + code = key.charCodeAt(i); + if (code < 0x80) { + bytes[index++] = code; + } else if (code < 0x800) { + bytes[index++] = (0xc0 | (code >>> 6)); + bytes[index++] = (0x80 | (code & 0x3f)); + } else if (code < 0xd800 || code >= 0xe000) { + bytes[index++] = (0xe0 | (code >>> 12)); + bytes[index++] = (0x80 | ((code >>> 6) & 0x3f)); + bytes[index++] = (0x80 | (code & 0x3f)); + } else { + code = 0x10000 + (((code & 0x3ff) << 10) | (key.charCodeAt(++i) & 0x3ff)); + bytes[index++] = (0xf0 | (code >>> 18)); + bytes[index++] = (0x80 | ((code >>> 12) & 0x3f)); + bytes[index++] = (0x80 | ((code >>> 6) & 0x3f)); + bytes[index++] = (0x80 | (code & 0x3f)); + } + } + key = bytes; + } else { + if (type === 'object') { + if (key === null) { + throw new Error(ERROR); + } else if (ARRAY_BUFFER && key.constructor === ArrayBuffer) { + key = new Uint8Array(key); + } else if (!Array.isArray(key)) { + if (!ARRAY_BUFFER || !ArrayBuffer.isView(key)) { + throw new Error(ERROR); + } + } + } else { + throw new Error(ERROR); + } + } + + if (key.length > 64) { + key = (new Sha256(is224, true)).update(key).array(); + } + + var oKeyPad = [], iKeyPad = []; + for (i = 0; i < 64; ++i) { + var b = key[i] || 0; + oKeyPad[i] = 0x5c ^ b; + iKeyPad[i] = 0x36 ^ b; + } + + Sha256.call(this, is224, sharedMemory); + + this.update(iKeyPad); + this.oKeyPad = oKeyPad; + this.inner = true; + this.sharedMemory = sharedMemory; + } + HmacSha256.prototype = new Sha256(); + + HmacSha256.prototype.finalize = function () { + Sha256.prototype.finalize.call(this); + if (this.inner) { + this.inner = false; + var innerHash = this.array(); + Sha256.call(this, this.is224, this.sharedMemory); + this.update(this.oKeyPad); + this.update(innerHash); + Sha256.prototype.finalize.call(this); + } + }; + + var exports = createMethod(); + exports.sha256 = exports; + exports.sha224 = createMethod(true); + exports.sha256.hmac = createHmacMethod(); + exports.sha224.hmac = createHmacMethod(true); + + if (COMMON_JS) { + module.exports = exports; + } else { + root.sha256 = exports.sha256; + root.sha224 = exports.sha224; + if (AMD) { + define(function () { + return exports; + }); + } + } +})(); diff --git a/node_modules/minimalistic-assert/LICENSE b/node_modules/minimalistic-assert/LICENSE new file mode 100644 index 0000000..adca66b --- /dev/null +++ b/node_modules/minimalistic-assert/LICENSE @@ -0,0 +1,13 @@ +Copyright 2015 Calvin Metcalf + +Permission to use, copy, modify, and/or distribute this software for any purpose +with or without fee is hereby granted, provided that the above copyright notice +and this permission notice appear in all copies. + +THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH +REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND +FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, +INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM +LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE +OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR +PERFORMANCE OF THIS SOFTWARE. \ No newline at end of file diff --git a/node_modules/minimalistic-assert/index.js b/node_modules/minimalistic-assert/index.js new file mode 100644 index 0000000..70b4ea5 --- /dev/null +++ b/node_modules/minimalistic-assert/index.js @@ -0,0 +1,11 @@ +module.exports = assert; + +function assert(val, msg) { + if (!val) + throw new Error(msg || 'Assertion failed'); +} + +assert.equal = function assertEqual(l, r, msg) { + if (l != r) + throw new Error(msg || ('Assertion failed: ' + l + ' != ' + r)); +}; diff --git a/node_modules/minimalistic-assert/package.json b/node_modules/minimalistic-assert/package.json new file mode 100644 index 0000000..f8de10d --- /dev/null +++ b/node_modules/minimalistic-assert/package.json @@ -0,0 +1,19 @@ +{ + "name": "minimalistic-assert", + "version": "1.0.1", + "description": "minimalistic-assert ===", + "main": "index.js", + "scripts": { + "test": "echo \"Error: no test specified\" && exit 1" + }, + "repository": { + "type": "git", + "url": "https://github.com/calvinmetcalf/minimalistic-assert.git" + }, + "author": "", + "license": "ISC", + "bugs": { + "url": "https://github.com/calvinmetcalf/minimalistic-assert/issues" + }, + "homepage": "https://github.com/calvinmetcalf/minimalistic-assert" +} diff --git a/node_modules/minimalistic-assert/readme.md b/node_modules/minimalistic-assert/readme.md new file mode 100644 index 0000000..2ca0d25 --- /dev/null +++ b/node_modules/minimalistic-assert/readme.md @@ -0,0 +1,4 @@ +minimalistic-assert +=== + +very minimalistic assert module. diff --git a/node_modules/minimalistic-crypto-utils/.npmignore b/node_modules/minimalistic-crypto-utils/.npmignore new file mode 100644 index 0000000..1ca9571 --- /dev/null +++ b/node_modules/minimalistic-crypto-utils/.npmignore @@ -0,0 +1,2 @@ +node_modules/ +npm-debug.log diff --git a/node_modules/minimalistic-crypto-utils/.travis.yml b/node_modules/minimalistic-crypto-utils/.travis.yml new file mode 100644 index 0000000..ce24b7a --- /dev/null +++ b/node_modules/minimalistic-crypto-utils/.travis.yml @@ -0,0 +1,11 @@ +sudo: false + +language: node_js + +node_js: + - "4" + - "6" + - "stable" + +script: + - npm test diff --git a/node_modules/minimalistic-crypto-utils/README.md b/node_modules/minimalistic-crypto-utils/README.md new file mode 100644 index 0000000..9e58eba --- /dev/null +++ b/node_modules/minimalistic-crypto-utils/README.md @@ -0,0 +1,47 @@ +# minimalistic-crypto-utils +[![Build Status](https://secure.travis-ci.org/indutny/minimalistic-crypto-utils.svg)](http://travis-ci.org/indutny/minimalistic-crypto-utils) +[![NPM version](https://badge.fury.io/js/minimalistic-crypto-utils.svg)](http://badge.fury.io/js/minimalistic-crypto-utils) + +Very minimal utils that are required in order to write reasonable JS-only +crypto module. + +## Usage + +```js +const utils = require('minimalistic-crypto-utils'); + +utils.toArray('abcd', 'hex'); +utils.encode([ 1, 2, 3, 4 ], 'hex'); +utils.toHex([ 1, 2, 3, 4 ]); +``` + +#### LICENSE + +This software is licensed under the MIT License. + +Copyright Fedor Indutny, 2017. + +Permission is hereby granted, free of charge, to any person obtaining a +copy of this software and associated documentation files (the +"Software"), to deal in the Software without restriction, including +without limitation the rights to use, copy, modify, merge, publish, +distribute, sublicense, and/or sell copies of the Software, and to permit +persons to whom the Software is furnished to do so, subject to the +following conditions: + +The above copyright notice and this permission notice shall be included +in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN +NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, +DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR +OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE +USE OR OTHER DEALINGS IN THE SOFTWARE. + +[0]: http://tools.ietf.org/html/rfc6979 +[1]: https://github.com/indutny/bn.js +[2]: https://github.com/indutny/hash.js +[3]: https://github.com/bitchan/eccrypto +[4]: https://github.com/wanderer/secp256k1-node diff --git a/node_modules/minimalistic-crypto-utils/lib/utils.js b/node_modules/minimalistic-crypto-utils/lib/utils.js new file mode 100644 index 0000000..cd48daf --- /dev/null +++ b/node_modules/minimalistic-crypto-utils/lib/utils.js @@ -0,0 +1,58 @@ +'use strict'; + +var utils = exports; + +function toArray(msg, enc) { + if (Array.isArray(msg)) + return msg.slice(); + if (!msg) + return []; + var res = []; + if (typeof msg !== 'string') { + for (var i = 0; i < msg.length; i++) + res[i] = msg[i] | 0; + return res; + } + if (enc === 'hex') { + msg = msg.replace(/[^a-z0-9]+/ig, ''); + if (msg.length % 2 !== 0) + msg = '0' + msg; + for (var i = 0; i < msg.length; i += 2) + res.push(parseInt(msg[i] + msg[i + 1], 16)); + } else { + for (var i = 0; i < msg.length; i++) { + var c = msg.charCodeAt(i); + var hi = c >> 8; + var lo = c & 0xff; + if (hi) + res.push(hi, lo); + else + res.push(lo); + } + } + return res; +} +utils.toArray = toArray; + +function zero2(word) { + if (word.length === 1) + return '0' + word; + else + return word; +} +utils.zero2 = zero2; + +function toHex(msg) { + var res = ''; + for (var i = 0; i < msg.length; i++) + res += zero2(msg[i].toString(16)); + return res; +} +utils.toHex = toHex; + +utils.encode = function encode(arr, enc) { + if (enc === 'hex') + return toHex(arr); + else + return arr; +}; diff --git a/node_modules/minimalistic-crypto-utils/package.json b/node_modules/minimalistic-crypto-utils/package.json new file mode 100644 index 0000000..bc4c79a --- /dev/null +++ b/node_modules/minimalistic-crypto-utils/package.json @@ -0,0 +1,27 @@ +{ + "name": "minimalistic-crypto-utils", + "version": "1.0.1", + "description": "Minimalistic tools for JS crypto modules", + "main": "lib/utils.js", + "scripts": { + "test": "mocha --reporter=spec test/*-test.js" + }, + "repository": { + "type": "git", + "url": "git+ssh://git@github.com/indutny/minimalistic-crypto-utils.git" + }, + "keywords": [ + "minimalistic", + "utils", + "crypto" + ], + "author": "Fedor Indutny ", + "license": "MIT", + "bugs": { + "url": "https://github.com/indutny/minimalistic-crypto-utils/issues" + }, + "homepage": "https://github.com/indutny/minimalistic-crypto-utils#readme", + "devDependencies": { + "mocha": "^3.2.0" + } +} diff --git a/node_modules/minimalistic-crypto-utils/test/utils-test.js b/node_modules/minimalistic-crypto-utils/test/utils-test.js new file mode 100644 index 0000000..3da812d --- /dev/null +++ b/node_modules/minimalistic-crypto-utils/test/utils-test.js @@ -0,0 +1,28 @@ +'use strict'; + +const assert = require('assert'); +const utils = require('../'); + +describe('utils', () => { + it('should convert to array', () => { + assert.deepEqual(utils.toArray('1234', 'hex'), [ 0x12, 0x34 ]); + assert.deepEqual(utils.toArray('1234'), [ 49, 50, 51, 52 ]); + assert.deepEqual(utils.toArray('1234', 'utf8'), [ 49, 50, 51, 52 ]); + assert.deepEqual(utils.toArray('\u1234234'), [ 18, 52, 50, 51, 52 ]); + assert.deepEqual(utils.toArray([ 1, 2, 3, 4 ]), [ 1, 2, 3, 4 ]); + }); + + it('should zero pad byte to hex', () => { + assert.equal(utils.zero2('0'), '00'); + assert.equal(utils.zero2('01'), '01'); + }); + + it('should convert to hex', () => { + assert.equal(utils.toHex([ 0, 1, 2, 3 ]), '00010203'); + }); + + it('should encode', () => { + assert.deepEqual(utils.encode([ 0, 1, 2, 3 ]), [ 0, 1, 2, 3 ]); + assert.deepEqual(utils.encode([ 0, 1, 2, 3 ], 'hex'), '00010203'); + }); +}); diff --git a/node_modules/node-addon-api/LICENSE.md b/node_modules/node-addon-api/LICENSE.md new file mode 100644 index 0000000..e2fad66 --- /dev/null +++ b/node_modules/node-addon-api/LICENSE.md @@ -0,0 +1,13 @@ +The MIT License (MIT) +===================== + +Copyright (c) 2017 Node.js API collaborators +----------------------------------- + +*Node.js API collaborators listed at * + +Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. \ No newline at end of file diff --git a/node_modules/node-addon-api/README.md b/node_modules/node-addon-api/README.md new file mode 100644 index 0000000..6a79c91 --- /dev/null +++ b/node_modules/node-addon-api/README.md @@ -0,0 +1,317 @@ +NOTE: The default branch has been renamed! +master is now named main + +If you have a local clone, you can update it by running: + +```shell +git branch -m master main +git fetch origin +git branch -u origin/main main +``` + +# **node-addon-api module** +This module contains **header-only C++ wrapper classes** which simplify +the use of the C based [Node-API](https://nodejs.org/dist/latest/docs/api/n-api.html) +provided by Node.js when using C++. It provides a C++ object model +and exception handling semantics with low overhead. + +There are three options for implementing addons: Node-API, nan, or direct +use of internal V8, libuv, and Node.js libraries. Unless there is a need for +direct access to functionality that is not exposed by Node-API as outlined +in [C/C++ addons](https://nodejs.org/dist/latest/docs/api/addons.html) +in Node.js core, use Node-API. Refer to +[C/C++ addons with Node-API](https://nodejs.org/dist/latest/docs/api/n-api.html) +for more information on Node-API. + +Node-API is an ABI stable C interface provided by Node.js for building native +addons. It is independent of the underlying JavaScript runtime (e.g. V8 or ChakraCore) +and is maintained as part of Node.js itself. It is intended to insulate +native addons from changes in the underlying JavaScript engine and allow +modules compiled for one version to run on later versions of Node.js without +recompilation. + +The `node-addon-api` module, which is not part of Node.js, preserves the benefits +of the Node-API as it consists only of inline code that depends only on the stable API +provided by Node-API. As such, modules built against one version of Node.js +using node-addon-api should run without having to be rebuilt with newer versions +of Node.js. + +It is important to remember that *other* Node.js interfaces such as +`libuv` (included in a project via `#include `) are not ABI-stable across +Node.js major versions. Thus, an addon must use Node-API and/or `node-addon-api` +exclusively and build against a version of Node.js that includes an +implementation of Node-API (meaning an active LTS version of Node.js) in +order to benefit from ABI stability across Node.js major versions. Node.js +provides an [ABI stability guide][] containing a detailed explanation of ABI +stability in general, and the Node-API ABI stability guarantee in particular. + +As new APIs are added to Node-API, node-addon-api must be updated to provide +wrappers for those new APIs. For this reason, node-addon-api provides +methods that allow callers to obtain the underlying Node-API handles so +direct calls to Node-API and the use of the objects/methods provided by +node-addon-api can be used together. For example, in order to be able +to use an API for which the node-addon-api does not yet provide a wrapper. + +APIs exposed by node-addon-api are generally used to create and +manipulate JavaScript values. Concepts and operations generally map +to ideas specified in the **ECMA262 Language Specification**. + +The [Node-API Resource](https://nodejs.github.io/node-addon-examples/) offers an +excellent orientation and tips for developers just getting started with Node-API +and node-addon-api. + +- **[Setup](#setup)** +- **[API Documentation](#api)** +- **[Examples](#examples)** +- **[Tests](#tests)** +- **[More resource and info about native Addons](#resources)** +- **[Badges](#badges)** +- **[Code of Conduct](CODE_OF_CONDUCT.md)** +- **[Contributors](#contributors)** +- **[License](#license)** + +## **Current version: 5.1.0** + +(See [CHANGELOG.md](CHANGELOG.md) for complete Changelog) + +[![NPM](https://nodei.co/npm/node-addon-api.png?downloads=true&downloadRank=true)](https://nodei.co/npm/node-addon-api/) [![NPM](https://nodei.co/npm-dl/node-addon-api.png?months=6&height=1)](https://nodei.co/npm/node-addon-api/) + + + +node-addon-api is based on [Node-API](https://nodejs.org/api/n-api.html) and supports using different Node-API versions. +This allows addons built with it to run with Node.js versions which support the targeted Node-API version. +**However** the node-addon-api support model is to support only the active LTS Node.js versions. This means that +every year there will be a new major which drops support for the Node.js LTS version which has gone out of service. + +The oldest Node.js version supported by the current version of node-addon-api is Node.js 14.x. + +## Setup + - [Installation and usage](doc/setup.md) + - [node-gyp](doc/node-gyp.md) + - [cmake-js](doc/cmake-js.md) + - [Conversion tool](doc/conversion-tool.md) + - [Checker tool](doc/checker-tool.md) + - [Generator](doc/generator.md) + - [Prebuild tools](doc/prebuild_tools.md) + + + +### **API Documentation** + +The following is the documentation for node-addon-api. + + - [Full Class Hierarchy](doc/hierarchy.md) + - [Addon Structure](doc/addon.md) + - Data Types: + - [Env](doc/env.md) + - [CallbackInfo](doc/callbackinfo.md) + - [Reference](doc/reference.md) + - [Value](doc/value.md) + - [Name](doc/name.md) + - [Symbol](doc/symbol.md) + - [String](doc/string.md) + - [Number](doc/number.md) + - [Date](doc/date.md) + - [BigInt](doc/bigint.md) + - [Boolean](doc/boolean.md) + - [External](doc/external.md) + - [Object](doc/object.md) + - [Array](doc/array.md) + - [ObjectReference](doc/object_reference.md) + - [PropertyDescriptor](doc/property_descriptor.md) + - [Function](doc/function.md) + - [FunctionReference](doc/function_reference.md) + - [ObjectWrap](doc/object_wrap.md) + - [ClassPropertyDescriptor](doc/class_property_descriptor.md) + - [Buffer](doc/buffer.md) + - [ArrayBuffer](doc/array_buffer.md) + - [TypedArray](doc/typed_array.md) + - [TypedArrayOf](doc/typed_array_of.md) + - [DataView](doc/dataview.md) + - [Error Handling](doc/error_handling.md) + - [Error](doc/error.md) + - [TypeError](doc/type_error.md) + - [RangeError](doc/range_error.md) + - [Object Lifetime Management](doc/object_lifetime_management.md) + - [HandleScope](doc/handle_scope.md) + - [EscapableHandleScope](doc/escapable_handle_scope.md) + - [Memory Management](doc/memory_management.md) + - [Async Operations](doc/async_operations.md) + - [AsyncWorker](doc/async_worker.md) + - [AsyncContext](doc/async_context.md) + - [AsyncWorker Variants](doc/async_worker_variants.md) + - [Thread-safe Functions](doc/threadsafe.md) + - [ThreadSafeFunction](doc/threadsafe_function.md) + - [TypedThreadSafeFunction](doc/typed_threadsafe_function.md) + - [Promises](doc/promises.md) + - [Version management](doc/version_management.md) + + + +### **Examples** + +Are you new to **node-addon-api**? Take a look at our **[examples](https://github.com/nodejs/node-addon-examples)** + +- **[Hello World](https://github.com/nodejs/node-addon-examples/tree/HEAD/1_hello_world/node-addon-api)** +- **[Pass arguments to a function](https://github.com/nodejs/node-addon-examples/tree/HEAD/2_function_arguments/node-addon-api)** +- **[Callbacks](https://github.com/nodejs/node-addon-examples/tree/HEAD/3_callbacks/node-addon-api)** +- **[Object factory](https://github.com/nodejs/node-addon-examples/tree/HEAD/4_object_factory/node-addon-api)** +- **[Function factory](https://github.com/nodejs/node-addon-examples/tree/HEAD/5_function_factory/node-addon-api)** +- **[Wrapping C++ Object](https://github.com/nodejs/node-addon-examples/tree/HEAD/6_object_wrap/node-addon-api)** +- **[Factory of wrapped object](https://github.com/nodejs/node-addon-examples/tree/HEAD/7_factory_wrap/node-addon-api)** +- **[Passing wrapped object around](https://github.com/nodejs/node-addon-examples/tree/HEAD/8_passing_wrapped/node-addon-api)** + + + +### **Tests** + +To run the **node-addon-api** tests do: + +``` +npm install +npm test +``` + +To avoid testing the deprecated portions of the API run +``` +npm install +npm test --disable-deprecated +``` + +To run the tests targeting a specific version of Node-API run +``` +npm install +export NAPI_VERSION=X +npm test --NAPI_VERSION=X +``` + +where X is the version of Node-API you want to target. + +To run a specific unit test, filter conditions are available + +**Example:** + compile and run only tests on objectwrap.cc and objectwrap.js + ``` + npm run unit --filter=objectwrap + ``` + +Multiple unit tests cane be selected with wildcards + +**Example:** +compile and run all test files ending with "reference" -> function_reference.cc, object_reference.cc, reference.cc + ``` + npm run unit --filter=*reference + ``` + +Multiple filter conditions can be joined to broaden the test selection + +**Example:** + compile and run all tests under folders threadsafe_function and typed_threadsafe_function and also the objectwrap.cc file + npm run unit --filter='*function objectwrap' + +### **Debug** + +To run the **node-addon-api** tests with `--debug` option: + +``` +npm run-script dev +``` + +If you want a faster build, you might use the following option: + +``` +npm run-script dev:incremental +``` + +Take a look and get inspired by our **[test suite](https://github.com/nodejs/node-addon-api/tree/HEAD/test)** + +### **Benchmarks** + +You can run the available benchmarks using the following command: + +``` +npm run-script benchmark +``` + +See [benchmark/README.md](benchmark/README.md) for more details about running and adding benchmarks. + + + +### **More resource and info about native Addons** +- **[C++ Addons](https://nodejs.org/dist/latest/docs/api/addons.html)** +- **[Node-API](https://nodejs.org/dist/latest/docs/api/n-api.html)** +- **[Node-API - Next Generation Node API for Native Modules](https://youtu.be/-Oniup60Afs)** +- **[How We Migrated Realm JavaScript From NAN to Node-API](https://developer.mongodb.com/article/realm-javascript-nan-to-n-api)** + +As node-addon-api's core mission is to expose the plain C Node-API as C++ +wrappers, tools that facilitate n-api/node-addon-api providing more +convenient patterns for developing a Node.js add-on with n-api/node-addon-api +can be published to NPM as standalone packages. It is also recommended to tag +such packages with `node-addon-api` to provide more visibility to the community. + +Quick links to NPM searches: [keywords:node-addon-api](https://www.npmjs.com/search?q=keywords%3Anode-addon-api). + + + +### **Other bindings** + +- **[napi-rs](https://napi.rs)** - (`Rust`) + + + +### **Badges** + +The use of badges is recommended to indicate the minimum version of Node-API +required for the module. This helps to determine which Node.js major versions are +supported. Addon maintainers can consult the [Node-API support matrix][] to determine +which Node.js versions provide a given Node-API version. The following badges are +available: + +![Node-API v1 Badge](https://github.com/nodejs/abi-stable-node/blob/doc/assets/Node-API%20v1%20Badge.svg) +![Node-API v2 Badge](https://github.com/nodejs/abi-stable-node/blob/doc/assets/Node-API%20v2%20Badge.svg) +![Node-API v3 Badge](https://github.com/nodejs/abi-stable-node/blob/doc/assets/Node-API%20v3%20Badge.svg) +![Node-API v4 Badge](https://github.com/nodejs/abi-stable-node/blob/doc/assets/Node-API%20v4%20Badge.svg) +![Node-API v5 Badge](https://github.com/nodejs/abi-stable-node/blob/doc/assets/Node-API%20v5%20Badge.svg) +![Node-API v6 Badge](https://github.com/nodejs/abi-stable-node/blob/doc/assets/Node-API%20v6%20Badge.svg) +![Node-API v7 Badge](https://github.com/nodejs/abi-stable-node/blob/doc/assets/Node-API%20v7%20Badge.svg) +![Node-API v8 Badge](https://github.com/nodejs/abi-stable-node/blob/doc/assets/Node-API%20v8%20Badge.svg) +![Node-API Experimental Version Badge](https://github.com/nodejs/abi-stable-node/blob/doc/assets/Node-API%20Experimental%20Version%20Badge.svg) + +## **Contributing** + +We love contributions from the community to **node-addon-api**! +See [CONTRIBUTING.md](CONTRIBUTING.md) for more details on our philosophy around extending this module. + + + +## Team members + +### Active +| Name | GitHub Link | +| ------------------- | ----------------------------------------------------- | +| Anna Henningsen | [addaleax](https://github.com/addaleax) | +| Chengzhong Wu | [legendecas](https://github.com/legendecas) | +| Jack Xia | [JckXia](https://github.com/JckXia) | +| Kevin Eady | [KevinEady](https://github.com/KevinEady) | +| Michael Dawson | [mhdawson](https://github.com/mhdawson) | +| Nicola Del Gobbo | [NickNaso](https://github.com/NickNaso) | +| Vladimir Morozov | [vmoroz](https://github.com/vmoroz) | + +### Emeritus +| Name | GitHub Link | +| ------------------- | ----------------------------------------------------- | +| Arunesh Chandra | [aruneshchandra](https://github.com/aruneshchandra) | +| Benjamin Byholm | [kkoopa](https://github.com/kkoopa) | +| Gabriel Schulhof | [gabrielschulhof](https://github.com/gabrielschulhof) | +| Hitesh Kanwathirtha | [digitalinfinity](https://github.com/digitalinfinity) | +| Jason Ginchereau | [jasongin](https://github.com/jasongin) | +| Jim Schlight | [jschlight](https://github.com/jschlight) | +| Sampson Gao | [sampsongao](https://github.com/sampsongao) | +| Taylor Woll | [boingoing](https://github.com/boingoing) | + + + +Licensed under [MIT](./LICENSE.md) + +[ABI stability guide]: https://nodejs.org/en/docs/guides/abi-stability/ +[Node-API support matrix]: https://nodejs.org/dist/latest/docs/api/n-api.html#n_api_n_api_version_matrix diff --git a/node_modules/node-addon-api/common.gypi b/node_modules/node-addon-api/common.gypi new file mode 100644 index 0000000..9be254f --- /dev/null +++ b/node_modules/node-addon-api/common.gypi @@ -0,0 +1,21 @@ +{ + 'variables': { + 'NAPI_VERSION%': " +inline PropertyDescriptor PropertyDescriptor::Accessor( + const char* utf8name, + Getter getter, + napi_property_attributes attributes, + void* /*data*/) { + using CbData = details::CallbackData; + // TODO: Delete when the function is destroyed + auto callbackData = new CbData({getter, nullptr}); + + return PropertyDescriptor({utf8name, + nullptr, + nullptr, + CbData::Wrapper, + nullptr, + nullptr, + attributes, + callbackData}); +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + const std::string& utf8name, + Getter getter, + napi_property_attributes attributes, + void* data) { + return Accessor(utf8name.c_str(), getter, attributes, data); +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + napi_value name, + Getter getter, + napi_property_attributes attributes, + void* /*data*/) { + using CbData = details::CallbackData; + // TODO: Delete when the function is destroyed + auto callbackData = new CbData({getter, nullptr}); + + return PropertyDescriptor({nullptr, + name, + nullptr, + CbData::Wrapper, + nullptr, + nullptr, + attributes, + callbackData}); +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + Name name, Getter getter, napi_property_attributes attributes, void* data) { + napi_value nameValue = name; + return PropertyDescriptor::Accessor(nameValue, getter, attributes, data); +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + const char* utf8name, + Getter getter, + Setter setter, + napi_property_attributes attributes, + void* /*data*/) { + using CbData = details::AccessorCallbackData; + // TODO: Delete when the function is destroyed + auto callbackData = new CbData({getter, setter, nullptr}); + + return PropertyDescriptor({utf8name, + nullptr, + nullptr, + CbData::GetterWrapper, + CbData::SetterWrapper, + nullptr, + attributes, + callbackData}); +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + const std::string& utf8name, + Getter getter, + Setter setter, + napi_property_attributes attributes, + void* data) { + return Accessor(utf8name.c_str(), getter, setter, attributes, data); +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + napi_value name, + Getter getter, + Setter setter, + napi_property_attributes attributes, + void* /*data*/) { + using CbData = details::AccessorCallbackData; + // TODO: Delete when the function is destroyed + auto callbackData = new CbData({getter, setter, nullptr}); + + return PropertyDescriptor({nullptr, + name, + nullptr, + CbData::GetterWrapper, + CbData::SetterWrapper, + nullptr, + attributes, + callbackData}); +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + Name name, + Getter getter, + Setter setter, + napi_property_attributes attributes, + void* data) { + napi_value nameValue = name; + return PropertyDescriptor::Accessor( + nameValue, getter, setter, attributes, data); +} + +template +inline PropertyDescriptor PropertyDescriptor::Function( + const char* utf8name, + Callable cb, + napi_property_attributes attributes, + void* /*data*/) { + using ReturnType = decltype(cb(CallbackInfo(nullptr, nullptr))); + using CbData = details::CallbackData; + // TODO: Delete when the function is destroyed + auto callbackData = new CbData({cb, nullptr}); + + return PropertyDescriptor({utf8name, + nullptr, + CbData::Wrapper, + nullptr, + nullptr, + nullptr, + attributes, + callbackData}); +} + +template +inline PropertyDescriptor PropertyDescriptor::Function( + const std::string& utf8name, + Callable cb, + napi_property_attributes attributes, + void* data) { + return Function(utf8name.c_str(), cb, attributes, data); +} + +template +inline PropertyDescriptor PropertyDescriptor::Function( + napi_value name, + Callable cb, + napi_property_attributes attributes, + void* /*data*/) { + using ReturnType = decltype(cb(CallbackInfo(nullptr, nullptr))); + using CbData = details::CallbackData; + // TODO: Delete when the function is destroyed + auto callbackData = new CbData({cb, nullptr}); + + return PropertyDescriptor({nullptr, + name, + CbData::Wrapper, + nullptr, + nullptr, + nullptr, + attributes, + callbackData}); +} + +template +inline PropertyDescriptor PropertyDescriptor::Function( + Name name, Callable cb, napi_property_attributes attributes, void* data) { + napi_value nameValue = name; + return PropertyDescriptor::Function(nameValue, cb, attributes, data); +} + +#endif // !SRC_NAPI_INL_DEPRECATED_H_ diff --git a/node_modules/node-addon-api/napi-inl.h b/node_modules/node-addon-api/napi-inl.h new file mode 100644 index 0000000..3ddc1ba --- /dev/null +++ b/node_modules/node-addon-api/napi-inl.h @@ -0,0 +1,6303 @@ +#ifndef SRC_NAPI_INL_H_ +#define SRC_NAPI_INL_H_ + +//////////////////////////////////////////////////////////////////////////////// +// Node-API C++ Wrapper Classes +// +// Inline header-only implementations for "Node-API" ABI-stable C APIs for +// Node.js. +//////////////////////////////////////////////////////////////////////////////// + +// Note: Do not include this file directly! Include "napi.h" instead. + +#include +#include +#include +#include +#include + +namespace Napi { + +#ifdef NAPI_CPP_CUSTOM_NAMESPACE +namespace NAPI_CPP_CUSTOM_NAMESPACE { +#endif + +// Helpers to handle functions exposed from C++. +namespace details { + +// Attach a data item to an object and delete it when the object gets +// garbage-collected. +// TODO: Replace this code with `napi_add_finalizer()` whenever it becomes +// available on all supported versions of Node.js. +template +inline napi_status AttachData(napi_env env, + napi_value obj, + FreeType* data, + napi_finalize finalizer = nullptr, + void* hint = nullptr) { + napi_status status; + if (finalizer == nullptr) { + finalizer = [](napi_env /*env*/, void* data, void* /*hint*/) { + delete static_cast(data); + }; + } +#if (NAPI_VERSION < 5) + napi_value symbol, external; + status = napi_create_symbol(env, nullptr, &symbol); + if (status == napi_ok) { + status = napi_create_external(env, data, finalizer, hint, &external); + if (status == napi_ok) { + napi_property_descriptor desc = {nullptr, + symbol, + nullptr, + nullptr, + nullptr, + external, + napi_default, + nullptr}; + status = napi_define_properties(env, obj, 1, &desc); + } + } +#else // NAPI_VERSION >= 5 + status = napi_add_finalizer(env, obj, data, finalizer, hint, nullptr); +#endif + return status; +} + +// For use in JS to C++ callback wrappers to catch any Napi::Error exceptions +// and rethrow them as JavaScript exceptions before returning from the callback. +template +inline napi_value WrapCallback(Callable callback) { +#ifdef NAPI_CPP_EXCEPTIONS + try { + return callback(); + } catch (const Error& e) { + e.ThrowAsJavaScriptException(); + return nullptr; + } +#else // NAPI_CPP_EXCEPTIONS + // When C++ exceptions are disabled, errors are immediately thrown as JS + // exceptions, so there is no need to catch and rethrow them here. + return callback(); +#endif // NAPI_CPP_EXCEPTIONS +} + +// For use in JS to C++ void callback wrappers to catch any Napi::Error +// exceptions and rethrow them as JavaScript exceptions before returning from +// the callback. +template +inline void WrapVoidCallback(Callable callback) { +#ifdef NAPI_CPP_EXCEPTIONS + try { + callback(); + } catch (const Error& e) { + e.ThrowAsJavaScriptException(); + } +#else // NAPI_CPP_EXCEPTIONS + // When C++ exceptions are disabled, errors are immediately thrown as JS + // exceptions, so there is no need to catch and rethrow them here. + callback(); +#endif // NAPI_CPP_EXCEPTIONS +} + +template +struct CallbackData { + static inline napi_value Wrapper(napi_env env, napi_callback_info info) { + return details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + CallbackData* callbackData = + static_cast(callbackInfo.Data()); + callbackInfo.SetData(callbackData->data); + return callbackData->callback(callbackInfo); + }); + } + + Callable callback; + void* data; +}; + +template +struct CallbackData { + static inline napi_value Wrapper(napi_env env, napi_callback_info info) { + return details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + CallbackData* callbackData = + static_cast(callbackInfo.Data()); + callbackInfo.SetData(callbackData->data); + callbackData->callback(callbackInfo); + return nullptr; + }); + } + + Callable callback; + void* data; +}; + +template +napi_value TemplatedVoidCallback(napi_env env, + napi_callback_info info) NAPI_NOEXCEPT { + return details::WrapCallback([&] { + CallbackInfo cbInfo(env, info); + Callback(cbInfo); + return nullptr; + }); +} + +template +napi_value TemplatedCallback(napi_env env, + napi_callback_info info) NAPI_NOEXCEPT { + return details::WrapCallback([&] { + CallbackInfo cbInfo(env, info); + return Callback(cbInfo); + }); +} + +template +napi_value TemplatedInstanceCallback(napi_env env, + napi_callback_info info) NAPI_NOEXCEPT { + return details::WrapCallback([&] { + CallbackInfo cbInfo(env, info); + T* instance = T::Unwrap(cbInfo.This().As()); + return (instance->*UnwrapCallback)(cbInfo); + }); +} + +template +napi_value TemplatedInstanceVoidCallback(napi_env env, napi_callback_info info) + NAPI_NOEXCEPT { + return details::WrapCallback([&] { + CallbackInfo cbInfo(env, info); + T* instance = T::Unwrap(cbInfo.This().As()); + (instance->*UnwrapCallback)(cbInfo); + return nullptr; + }); +} + +template +struct FinalizeData { + static inline void Wrapper(napi_env env, + void* data, + void* finalizeHint) NAPI_NOEXCEPT { + WrapVoidCallback([&] { + FinalizeData* finalizeData = static_cast(finalizeHint); + finalizeData->callback(Env(env), static_cast(data)); + delete finalizeData; + }); + } + + static inline void WrapperWithHint(napi_env env, + void* data, + void* finalizeHint) NAPI_NOEXCEPT { + WrapVoidCallback([&] { + FinalizeData* finalizeData = static_cast(finalizeHint); + finalizeData->callback( + Env(env), static_cast(data), finalizeData->hint); + delete finalizeData; + }); + } + + Finalizer callback; + Hint* hint; +}; + +#if (NAPI_VERSION > 3 && !defined(__wasm32__)) +template , + typename FinalizerDataType = void> +struct ThreadSafeFinalize { + static inline void Wrapper(napi_env env, + void* rawFinalizeData, + void* /* rawContext */) { + if (rawFinalizeData == nullptr) return; + + ThreadSafeFinalize* finalizeData = + static_cast(rawFinalizeData); + finalizeData->callback(Env(env)); + delete finalizeData; + } + + static inline void FinalizeWrapperWithData(napi_env env, + void* rawFinalizeData, + void* /* rawContext */) { + if (rawFinalizeData == nullptr) return; + + ThreadSafeFinalize* finalizeData = + static_cast(rawFinalizeData); + finalizeData->callback(Env(env), finalizeData->data); + delete finalizeData; + } + + static inline void FinalizeWrapperWithContext(napi_env env, + void* rawFinalizeData, + void* rawContext) { + if (rawFinalizeData == nullptr) return; + + ThreadSafeFinalize* finalizeData = + static_cast(rawFinalizeData); + finalizeData->callback(Env(env), static_cast(rawContext)); + delete finalizeData; + } + + static inline void FinalizeFinalizeWrapperWithDataAndContext( + napi_env env, void* rawFinalizeData, void* rawContext) { + if (rawFinalizeData == nullptr) return; + + ThreadSafeFinalize* finalizeData = + static_cast(rawFinalizeData); + finalizeData->callback( + Env(env), finalizeData->data, static_cast(rawContext)); + delete finalizeData; + } + + FinalizerDataType* data; + Finalizer callback; +}; + +template +inline typename std::enable_if(nullptr)>::type +CallJsWrapper(napi_env env, napi_value jsCallback, void* context, void* data) { + call(env, + Function(env, jsCallback), + static_cast(context), + static_cast(data)); +} + +template +inline typename std::enable_if(nullptr)>::type +CallJsWrapper(napi_env env, + napi_value jsCallback, + void* /*context*/, + void* /*data*/) { + if (jsCallback != nullptr) { + Function(env, jsCallback).Call(0, nullptr); + } +} + +#if NAPI_VERSION > 4 + +template +napi_value DefaultCallbackWrapper(napi_env /*env*/, std::nullptr_t /*cb*/) { + return nullptr; +} + +template +napi_value DefaultCallbackWrapper(napi_env /*env*/, Napi::Function cb) { + return cb; +} + +#else +template +napi_value DefaultCallbackWrapper(napi_env env, Napi::Function cb) { + if (cb.IsEmpty()) { + return TSFN::EmptyFunctionFactory(env); + } + return cb; +} +#endif // NAPI_VERSION > 4 +#endif // NAPI_VERSION > 3 && !defined(__wasm32__) + +template +struct AccessorCallbackData { + static inline napi_value GetterWrapper(napi_env env, + napi_callback_info info) { + return details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + AccessorCallbackData* callbackData = + static_cast(callbackInfo.Data()); + callbackInfo.SetData(callbackData->data); + return callbackData->getterCallback(callbackInfo); + }); + } + + static inline napi_value SetterWrapper(napi_env env, + napi_callback_info info) { + return details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + AccessorCallbackData* callbackData = + static_cast(callbackInfo.Data()); + callbackInfo.SetData(callbackData->data); + callbackData->setterCallback(callbackInfo); + return nullptr; + }); + } + + Getter getterCallback; + Setter setterCallback; + void* data; +}; + +} // namespace details + +#ifndef NODE_ADDON_API_DISABLE_DEPRECATED +#include "napi-inl.deprecated.h" +#endif // !NODE_ADDON_API_DISABLE_DEPRECATED + +//////////////////////////////////////////////////////////////////////////////// +// Module registration +//////////////////////////////////////////////////////////////////////////////// + +// Register an add-on based on an initializer function. +#define NODE_API_MODULE(modname, regfunc) \ + static napi_value __napi_##regfunc(napi_env env, napi_value exports) { \ + return Napi::RegisterModule(env, exports, regfunc); \ + } \ + NAPI_MODULE(modname, __napi_##regfunc) + +// Register an add-on based on a subclass of `Addon` with a custom Node.js +// module name. +#define NODE_API_NAMED_ADDON(modname, classname) \ + static napi_value __napi_##classname(napi_env env, napi_value exports) { \ + return Napi::RegisterModule(env, exports, &classname::Init); \ + } \ + NAPI_MODULE(modname, __napi_##classname) + +// Register an add-on based on a subclass of `Addon` with the Node.js module +// name given by node-gyp from the `target_name` in binding.gyp. +#define NODE_API_ADDON(classname) \ + NODE_API_NAMED_ADDON(NODE_GYP_MODULE_NAME, classname) + +// Adapt the NAPI_MODULE registration function: +// - Wrap the arguments in NAPI wrappers. +// - Catch any NAPI errors and rethrow as JS exceptions. +inline napi_value RegisterModule(napi_env env, + napi_value exports, + ModuleRegisterCallback registerCallback) { + return details::WrapCallback([&] { + return napi_value( + registerCallback(Napi::Env(env), Napi::Object(env, exports))); + }); +} + +//////////////////////////////////////////////////////////////////////////////// +// Maybe class +//////////////////////////////////////////////////////////////////////////////// + +template +bool Maybe::IsNothing() const { + return !_has_value; +} + +template +bool Maybe::IsJust() const { + return _has_value; +} + +template +void Maybe::Check() const { + NAPI_CHECK(IsJust(), "Napi::Maybe::Check", "Maybe value is Nothing."); +} + +template +T Maybe::Unwrap() const { + NAPI_CHECK(IsJust(), "Napi::Maybe::Unwrap", "Maybe value is Nothing."); + return _value; +} + +template +T Maybe::UnwrapOr(const T& default_value) const { + return _has_value ? _value : default_value; +} + +template +bool Maybe::UnwrapTo(T* out) const { + if (IsJust()) { + *out = _value; + return true; + }; + return false; +} + +template +bool Maybe::operator==(const Maybe& other) const { + return (IsJust() == other.IsJust()) && + (!IsJust() || Unwrap() == other.Unwrap()); +} + +template +bool Maybe::operator!=(const Maybe& other) const { + return !operator==(other); +} + +template +Maybe::Maybe() : _has_value(false) {} + +template +Maybe::Maybe(const T& t) : _has_value(true), _value(t) {} + +template +inline Maybe Nothing() { + return Maybe(); +} + +template +inline Maybe Just(const T& t) { + return Maybe(t); +} + +//////////////////////////////////////////////////////////////////////////////// +// Env class +//////////////////////////////////////////////////////////////////////////////// + +inline Env::Env(napi_env env) : _env(env) {} + +inline Env::operator napi_env() const { + return _env; +} + +inline Object Env::Global() const { + napi_value value; + napi_status status = napi_get_global(*this, &value); + NAPI_THROW_IF_FAILED(*this, status, Object()); + return Object(*this, value); +} + +inline Value Env::Undefined() const { + napi_value value; + napi_status status = napi_get_undefined(*this, &value); + NAPI_THROW_IF_FAILED(*this, status, Value()); + return Value(*this, value); +} + +inline Value Env::Null() const { + napi_value value; + napi_status status = napi_get_null(*this, &value); + NAPI_THROW_IF_FAILED(*this, status, Value()); + return Value(*this, value); +} + +inline bool Env::IsExceptionPending() const { + bool result; + napi_status status = napi_is_exception_pending(_env, &result); + if (status != napi_ok) + result = false; // Checking for a pending exception shouldn't throw. + return result; +} + +inline Error Env::GetAndClearPendingException() const { + napi_value value; + napi_status status = napi_get_and_clear_last_exception(_env, &value); + if (status != napi_ok) { + // Don't throw another exception when failing to get the exception! + return Error(); + } + return Error(_env, value); +} + +inline MaybeOrValue Env::RunScript(const char* utf8script) const { + String script = String::New(_env, utf8script); + return RunScript(script); +} + +inline MaybeOrValue Env::RunScript(const std::string& utf8script) const { + return RunScript(utf8script.c_str()); +} + +inline MaybeOrValue Env::RunScript(String script) const { + napi_value result; + napi_status status = napi_run_script(_env, script, &result); + NAPI_RETURN_OR_THROW_IF_FAILED( + _env, status, Napi::Value(_env, result), Napi::Value); +} + +#if NAPI_VERSION > 2 +template +void Env::CleanupHook::Wrapper(void* data) NAPI_NOEXCEPT { + auto* cleanupData = + static_cast::CleanupData*>( + data); + cleanupData->hook(); + delete cleanupData; +} + +template +void Env::CleanupHook::WrapperWithArg(void* data) NAPI_NOEXCEPT { + auto* cleanupData = + static_cast::CleanupData*>( + data); + cleanupData->hook(static_cast(cleanupData->arg)); + delete cleanupData; +} +#endif // NAPI_VERSION > 2 + +#if NAPI_VERSION > 5 +template fini> +inline void Env::SetInstanceData(T* data) const { + napi_status status = napi_set_instance_data( + _env, + data, + [](napi_env env, void* data, void*) { fini(env, static_cast(data)); }, + nullptr); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +template fini> +inline void Env::SetInstanceData(DataType* data, HintType* hint) const { + napi_status status = napi_set_instance_data( + _env, + data, + [](napi_env env, void* data, void* hint) { + fini(env, static_cast(data), static_cast(hint)); + }, + hint); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +template +inline T* Env::GetInstanceData() const { + void* data = nullptr; + + napi_status status = napi_get_instance_data(_env, &data); + NAPI_THROW_IF_FAILED(_env, status, nullptr); + + return static_cast(data); +} + +template +void Env::DefaultFini(Env, T* data) { + delete data; +} + +template +void Env::DefaultFiniWithHint(Env, DataType* data, HintType*) { + delete data; +} +#endif // NAPI_VERSION > 5 + +//////////////////////////////////////////////////////////////////////////////// +// Value class +//////////////////////////////////////////////////////////////////////////////// + +inline Value::Value() : _env(nullptr), _value(nullptr) {} + +inline Value::Value(napi_env env, napi_value value) + : _env(env), _value(value) {} + +inline Value::operator napi_value() const { + return _value; +} + +inline bool Value::operator==(const Value& other) const { + return StrictEquals(other); +} + +inline bool Value::operator!=(const Value& other) const { + return !this->operator==(other); +} + +inline bool Value::StrictEquals(const Value& other) const { + bool result; + napi_status status = napi_strict_equals(_env, *this, other, &result); + NAPI_THROW_IF_FAILED(_env, status, false); + return result; +} + +inline Napi::Env Value::Env() const { + return Napi::Env(_env); +} + +inline bool Value::IsEmpty() const { + return _value == nullptr; +} + +inline napi_valuetype Value::Type() const { + if (IsEmpty()) { + return napi_undefined; + } + + napi_valuetype type; + napi_status status = napi_typeof(_env, _value, &type); + NAPI_THROW_IF_FAILED(_env, status, napi_undefined); + return type; +} + +inline bool Value::IsUndefined() const { + return Type() == napi_undefined; +} + +inline bool Value::IsNull() const { + return Type() == napi_null; +} + +inline bool Value::IsBoolean() const { + return Type() == napi_boolean; +} + +inline bool Value::IsNumber() const { + return Type() == napi_number; +} + +#if NAPI_VERSION > 5 +inline bool Value::IsBigInt() const { + return Type() == napi_bigint; +} +#endif // NAPI_VERSION > 5 + +#if (NAPI_VERSION > 4) +inline bool Value::IsDate() const { + if (IsEmpty()) { + return false; + } + + bool result; + napi_status status = napi_is_date(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, false); + return result; +} +#endif + +inline bool Value::IsString() const { + return Type() == napi_string; +} + +inline bool Value::IsSymbol() const { + return Type() == napi_symbol; +} + +inline bool Value::IsArray() const { + if (IsEmpty()) { + return false; + } + + bool result; + napi_status status = napi_is_array(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, false); + return result; +} + +inline bool Value::IsArrayBuffer() const { + if (IsEmpty()) { + return false; + } + + bool result; + napi_status status = napi_is_arraybuffer(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, false); + return result; +} + +inline bool Value::IsTypedArray() const { + if (IsEmpty()) { + return false; + } + + bool result; + napi_status status = napi_is_typedarray(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, false); + return result; +} + +inline bool Value::IsObject() const { + return Type() == napi_object || IsFunction(); +} + +inline bool Value::IsFunction() const { + return Type() == napi_function; +} + +inline bool Value::IsPromise() const { + if (IsEmpty()) { + return false; + } + + bool result; + napi_status status = napi_is_promise(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, false); + return result; +} + +inline bool Value::IsDataView() const { + if (IsEmpty()) { + return false; + } + + bool result; + napi_status status = napi_is_dataview(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, false); + return result; +} + +inline bool Value::IsBuffer() const { + if (IsEmpty()) { + return false; + } + + bool result; + napi_status status = napi_is_buffer(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, false); + return result; +} + +inline bool Value::IsExternal() const { + return Type() == napi_external; +} + +template +inline T Value::As() const { + return T(_env, _value); +} + +inline MaybeOrValue Value::ToBoolean() const { + napi_value result; + napi_status status = napi_coerce_to_bool(_env, _value, &result); + NAPI_RETURN_OR_THROW_IF_FAILED( + _env, status, Napi::Boolean(_env, result), Napi::Boolean); +} + +inline MaybeOrValue Value::ToNumber() const { + napi_value result; + napi_status status = napi_coerce_to_number(_env, _value, &result); + NAPI_RETURN_OR_THROW_IF_FAILED( + _env, status, Napi::Number(_env, result), Napi::Number); +} + +inline MaybeOrValue Value::ToString() const { + napi_value result; + napi_status status = napi_coerce_to_string(_env, _value, &result); + NAPI_RETURN_OR_THROW_IF_FAILED( + _env, status, Napi::String(_env, result), Napi::String); +} + +inline MaybeOrValue Value::ToObject() const { + napi_value result; + napi_status status = napi_coerce_to_object(_env, _value, &result); + NAPI_RETURN_OR_THROW_IF_FAILED( + _env, status, Napi::Object(_env, result), Napi::Object); +} + +//////////////////////////////////////////////////////////////////////////////// +// Boolean class +//////////////////////////////////////////////////////////////////////////////// + +inline Boolean Boolean::New(napi_env env, bool val) { + napi_value value; + napi_status status = napi_get_boolean(env, val, &value); + NAPI_THROW_IF_FAILED(env, status, Boolean()); + return Boolean(env, value); +} + +inline Boolean::Boolean() : Napi::Value() {} + +inline Boolean::Boolean(napi_env env, napi_value value) + : Napi::Value(env, value) {} + +inline Boolean::operator bool() const { + return Value(); +} + +inline bool Boolean::Value() const { + bool result; + napi_status status = napi_get_value_bool(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, false); + return result; +} + +//////////////////////////////////////////////////////////////////////////////// +// Number class +//////////////////////////////////////////////////////////////////////////////// + +inline Number Number::New(napi_env env, double val) { + napi_value value; + napi_status status = napi_create_double(env, val, &value); + NAPI_THROW_IF_FAILED(env, status, Number()); + return Number(env, value); +} + +inline Number::Number() : Value() {} + +inline Number::Number(napi_env env, napi_value value) : Value(env, value) {} + +inline Number::operator int32_t() const { + return Int32Value(); +} + +inline Number::operator uint32_t() const { + return Uint32Value(); +} + +inline Number::operator int64_t() const { + return Int64Value(); +} + +inline Number::operator float() const { + return FloatValue(); +} + +inline Number::operator double() const { + return DoubleValue(); +} + +inline int32_t Number::Int32Value() const { + int32_t result; + napi_status status = napi_get_value_int32(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, 0); + return result; +} + +inline uint32_t Number::Uint32Value() const { + uint32_t result; + napi_status status = napi_get_value_uint32(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, 0); + return result; +} + +inline int64_t Number::Int64Value() const { + int64_t result; + napi_status status = napi_get_value_int64(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, 0); + return result; +} + +inline float Number::FloatValue() const { + return static_cast(DoubleValue()); +} + +inline double Number::DoubleValue() const { + double result; + napi_status status = napi_get_value_double(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, 0); + return result; +} + +#if NAPI_VERSION > 5 +//////////////////////////////////////////////////////////////////////////////// +// BigInt Class +//////////////////////////////////////////////////////////////////////////////// + +inline BigInt BigInt::New(napi_env env, int64_t val) { + napi_value value; + napi_status status = napi_create_bigint_int64(env, val, &value); + NAPI_THROW_IF_FAILED(env, status, BigInt()); + return BigInt(env, value); +} + +inline BigInt BigInt::New(napi_env env, uint64_t val) { + napi_value value; + napi_status status = napi_create_bigint_uint64(env, val, &value); + NAPI_THROW_IF_FAILED(env, status, BigInt()); + return BigInt(env, value); +} + +inline BigInt BigInt::New(napi_env env, + int sign_bit, + size_t word_count, + const uint64_t* words) { + napi_value value; + napi_status status = + napi_create_bigint_words(env, sign_bit, word_count, words, &value); + NAPI_THROW_IF_FAILED(env, status, BigInt()); + return BigInt(env, value); +} + +inline BigInt::BigInt() : Value() {} + +inline BigInt::BigInt(napi_env env, napi_value value) : Value(env, value) {} + +inline int64_t BigInt::Int64Value(bool* lossless) const { + int64_t result; + napi_status status = + napi_get_value_bigint_int64(_env, _value, &result, lossless); + NAPI_THROW_IF_FAILED(_env, status, 0); + return result; +} + +inline uint64_t BigInt::Uint64Value(bool* lossless) const { + uint64_t result; + napi_status status = + napi_get_value_bigint_uint64(_env, _value, &result, lossless); + NAPI_THROW_IF_FAILED(_env, status, 0); + return result; +} + +inline size_t BigInt::WordCount() const { + size_t word_count; + napi_status status = + napi_get_value_bigint_words(_env, _value, nullptr, &word_count, nullptr); + NAPI_THROW_IF_FAILED(_env, status, 0); + return word_count; +} + +inline void BigInt::ToWords(int* sign_bit, + size_t* word_count, + uint64_t* words) { + napi_status status = + napi_get_value_bigint_words(_env, _value, sign_bit, word_count, words); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} +#endif // NAPI_VERSION > 5 + +#if (NAPI_VERSION > 4) +//////////////////////////////////////////////////////////////////////////////// +// Date Class +//////////////////////////////////////////////////////////////////////////////// + +inline Date Date::New(napi_env env, double val) { + napi_value value; + napi_status status = napi_create_date(env, val, &value); + NAPI_THROW_IF_FAILED(env, status, Date()); + return Date(env, value); +} + +inline Date::Date() : Value() {} + +inline Date::Date(napi_env env, napi_value value) : Value(env, value) {} + +inline Date::operator double() const { + return ValueOf(); +} + +inline double Date::ValueOf() const { + double result; + napi_status status = napi_get_date_value(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, 0); + return result; +} +#endif + +//////////////////////////////////////////////////////////////////////////////// +// Name class +//////////////////////////////////////////////////////////////////////////////// + +inline Name::Name() : Value() {} + +inline Name::Name(napi_env env, napi_value value) : Value(env, value) {} + +//////////////////////////////////////////////////////////////////////////////// +// String class +//////////////////////////////////////////////////////////////////////////////// + +inline String String::New(napi_env env, const std::string& val) { + return String::New(env, val.c_str(), val.size()); +} + +inline String String::New(napi_env env, const std::u16string& val) { + return String::New(env, val.c_str(), val.size()); +} + +inline String String::New(napi_env env, const char* val) { + // TODO(@gabrielschulhof) Remove if-statement when core's error handling is + // available in all supported versions. + if (val == nullptr) { + // Throw an error that looks like it came from core. + NAPI_THROW_IF_FAILED(env, napi_invalid_arg, String()); + } + napi_value value; + napi_status status = + napi_create_string_utf8(env, val, std::strlen(val), &value); + NAPI_THROW_IF_FAILED(env, status, String()); + return String(env, value); +} + +inline String String::New(napi_env env, const char16_t* val) { + napi_value value; + // TODO(@gabrielschulhof) Remove if-statement when core's error handling is + // available in all supported versions. + if (val == nullptr) { + // Throw an error that looks like it came from core. + NAPI_THROW_IF_FAILED(env, napi_invalid_arg, String()); + } + napi_status status = + napi_create_string_utf16(env, val, std::u16string(val).size(), &value); + NAPI_THROW_IF_FAILED(env, status, String()); + return String(env, value); +} + +inline String String::New(napi_env env, const char* val, size_t length) { + napi_value value; + napi_status status = napi_create_string_utf8(env, val, length, &value); + NAPI_THROW_IF_FAILED(env, status, String()); + return String(env, value); +} + +inline String String::New(napi_env env, const char16_t* val, size_t length) { + napi_value value; + napi_status status = napi_create_string_utf16(env, val, length, &value); + NAPI_THROW_IF_FAILED(env, status, String()); + return String(env, value); +} + +inline String::String() : Name() {} + +inline String::String(napi_env env, napi_value value) : Name(env, value) {} + +inline String::operator std::string() const { + return Utf8Value(); +} + +inline String::operator std::u16string() const { + return Utf16Value(); +} + +inline std::string String::Utf8Value() const { + size_t length; + napi_status status = + napi_get_value_string_utf8(_env, _value, nullptr, 0, &length); + NAPI_THROW_IF_FAILED(_env, status, ""); + + std::string value; + value.reserve(length + 1); + value.resize(length); + status = napi_get_value_string_utf8( + _env, _value, &value[0], value.capacity(), nullptr); + NAPI_THROW_IF_FAILED(_env, status, ""); + return value; +} + +inline std::u16string String::Utf16Value() const { + size_t length; + napi_status status = + napi_get_value_string_utf16(_env, _value, nullptr, 0, &length); + NAPI_THROW_IF_FAILED(_env, status, NAPI_WIDE_TEXT("")); + + std::u16string value; + value.reserve(length + 1); + value.resize(length); + status = napi_get_value_string_utf16( + _env, _value, &value[0], value.capacity(), nullptr); + NAPI_THROW_IF_FAILED(_env, status, NAPI_WIDE_TEXT("")); + return value; +} + +//////////////////////////////////////////////////////////////////////////////// +// Symbol class +//////////////////////////////////////////////////////////////////////////////// + +inline Symbol Symbol::New(napi_env env, const char* description) { + napi_value descriptionValue = description != nullptr + ? String::New(env, description) + : static_cast(nullptr); + return Symbol::New(env, descriptionValue); +} + +inline Symbol Symbol::New(napi_env env, const std::string& description) { + napi_value descriptionValue = String::New(env, description); + return Symbol::New(env, descriptionValue); +} + +inline Symbol Symbol::New(napi_env env, String description) { + napi_value descriptionValue = description; + return Symbol::New(env, descriptionValue); +} + +inline Symbol Symbol::New(napi_env env, napi_value description) { + napi_value value; + napi_status status = napi_create_symbol(env, description, &value); + NAPI_THROW_IF_FAILED(env, status, Symbol()); + return Symbol(env, value); +} + +inline MaybeOrValue Symbol::WellKnown(napi_env env, + const std::string& name) { +#if defined(NODE_ADDON_API_ENABLE_MAYBE) + Value symbol_obj; + Value symbol_value; + if (Napi::Env(env).Global().Get("Symbol").UnwrapTo(&symbol_obj) && + symbol_obj.As().Get(name).UnwrapTo(&symbol_value)) { + return Just(symbol_value.As()); + } + return Nothing(); +#else + return Napi::Env(env) + .Global() + .Get("Symbol") + .As() + .Get(name) + .As(); +#endif +} + +inline MaybeOrValue Symbol::For(napi_env env, + const std::string& description) { + napi_value descriptionValue = String::New(env, description); + return Symbol::For(env, descriptionValue); +} + +inline MaybeOrValue Symbol::For(napi_env env, const char* description) { + napi_value descriptionValue = String::New(env, description); + return Symbol::For(env, descriptionValue); +} + +inline MaybeOrValue Symbol::For(napi_env env, String description) { + return Symbol::For(env, static_cast(description)); +} + +inline MaybeOrValue Symbol::For(napi_env env, napi_value description) { +#if defined(NODE_ADDON_API_ENABLE_MAYBE) + Value symbol_obj; + Value symbol_for_value; + Value symbol_value; + if (Napi::Env(env).Global().Get("Symbol").UnwrapTo(&symbol_obj) && + symbol_obj.As().Get("for").UnwrapTo(&symbol_for_value) && + symbol_for_value.As() + .Call(symbol_obj, {description}) + .UnwrapTo(&symbol_value)) { + return Just(symbol_value.As()); + } + return Nothing(); +#else + Object symbol_obj = Napi::Env(env).Global().Get("Symbol").As(); + return symbol_obj.Get("for") + .As() + .Call(symbol_obj, {description}) + .As(); +#endif +} + +inline Symbol::Symbol() : Name() {} + +inline Symbol::Symbol(napi_env env, napi_value value) : Name(env, value) {} + +//////////////////////////////////////////////////////////////////////////////// +// Automagic value creation +//////////////////////////////////////////////////////////////////////////////// + +namespace details { +template +struct vf_number { + static Number From(napi_env env, T value) { + return Number::New(env, static_cast(value)); + } +}; + +template <> +struct vf_number { + static Boolean From(napi_env env, bool value) { + return Boolean::New(env, value); + } +}; + +struct vf_utf8_charp { + static String From(napi_env env, const char* value) { + return String::New(env, value); + } +}; + +struct vf_utf16_charp { + static String From(napi_env env, const char16_t* value) { + return String::New(env, value); + } +}; +struct vf_utf8_string { + static String From(napi_env env, const std::string& value) { + return String::New(env, value); + } +}; + +struct vf_utf16_string { + static String From(napi_env env, const std::u16string& value) { + return String::New(env, value); + } +}; + +template +struct vf_fallback { + static Value From(napi_env env, const T& value) { return Value(env, value); } +}; + +template +struct disjunction : std::false_type {}; +template +struct disjunction : B {}; +template +struct disjunction + : std::conditional>::type {}; + +template +struct can_make_string + : disjunction::type, + typename std::is_convertible::type, + typename std::is_convertible::type, + typename std::is_convertible::type> {}; +} // namespace details + +template +Value Value::From(napi_env env, const T& value) { + using Helper = typename std::conditional< + std::is_integral::value || std::is_floating_point::value, + details::vf_number, + typename std::conditional::value, + String, + details::vf_fallback>::type>::type; + return Helper::From(env, value); +} + +template +String String::From(napi_env env, const T& value) { + struct Dummy {}; + using Helper = typename std::conditional< + std::is_convertible::value, + details::vf_utf8_charp, + typename std::conditional< + std::is_convertible::value, + details::vf_utf16_charp, + typename std::conditional< + std::is_convertible::value, + details::vf_utf8_string, + typename std::conditional< + std::is_convertible::value, + details::vf_utf16_string, + Dummy>::type>::type>::type>::type; + return Helper::From(env, value); +} + +//////////////////////////////////////////////////////////////////////////////// +// Object class +//////////////////////////////////////////////////////////////////////////////// + +template +inline Object::PropertyLValue::operator Value() const { + MaybeOrValue val = Object(_env, _object).Get(_key); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + return val.Unwrap(); +#else + return val; +#endif +} + +template +template +inline Object::PropertyLValue& Object::PropertyLValue::operator=( + ValueType value) { +#ifdef NODE_ADDON_API_ENABLE_MAYBE + MaybeOrValue result = +#endif + Object(_env, _object).Set(_key, value); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + result.Unwrap(); +#endif + return *this; +} + +template +inline Object::PropertyLValue::PropertyLValue(Object object, Key key) + : _env(object.Env()), _object(object), _key(key) {} + +inline Object Object::New(napi_env env) { + napi_value value; + napi_status status = napi_create_object(env, &value); + NAPI_THROW_IF_FAILED(env, status, Object()); + return Object(env, value); +} + +inline Object::Object() : Value() {} + +inline Object::Object(napi_env env, napi_value value) : Value(env, value) {} + +inline Object::PropertyLValue Object::operator[]( + const char* utf8name) { + return PropertyLValue(*this, utf8name); +} + +inline Object::PropertyLValue Object::operator[]( + const std::string& utf8name) { + return PropertyLValue(*this, utf8name); +} + +inline Object::PropertyLValue Object::operator[](uint32_t index) { + return PropertyLValue(*this, index); +} + +inline Object::PropertyLValue Object::operator[](Value index) const { + return PropertyLValue(*this, index); +} + +inline MaybeOrValue Object::operator[](const char* utf8name) const { + return Get(utf8name); +} + +inline MaybeOrValue Object::operator[]( + const std::string& utf8name) const { + return Get(utf8name); +} + +inline MaybeOrValue Object::operator[](uint32_t index) const { + return Get(index); +} + +inline MaybeOrValue Object::Has(napi_value key) const { + bool result; + napi_status status = napi_has_property(_env, _value, key, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, result, bool); +} + +inline MaybeOrValue Object::Has(Value key) const { + bool result; + napi_status status = napi_has_property(_env, _value, key, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, result, bool); +} + +inline MaybeOrValue Object::Has(const char* utf8name) const { + bool result; + napi_status status = napi_has_named_property(_env, _value, utf8name, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, result, bool); +} + +inline MaybeOrValue Object::Has(const std::string& utf8name) const { + return Has(utf8name.c_str()); +} + +inline MaybeOrValue Object::HasOwnProperty(napi_value key) const { + bool result; + napi_status status = napi_has_own_property(_env, _value, key, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, result, bool); +} + +inline MaybeOrValue Object::HasOwnProperty(Value key) const { + bool result; + napi_status status = napi_has_own_property(_env, _value, key, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, result, bool); +} + +inline MaybeOrValue Object::HasOwnProperty(const char* utf8name) const { + napi_value key; + napi_status status = + napi_create_string_utf8(_env, utf8name, std::strlen(utf8name), &key); + NAPI_MAYBE_THROW_IF_FAILED(_env, status, bool); + return HasOwnProperty(key); +} + +inline MaybeOrValue Object::HasOwnProperty( + const std::string& utf8name) const { + return HasOwnProperty(utf8name.c_str()); +} + +inline MaybeOrValue Object::Get(napi_value key) const { + napi_value result; + napi_status status = napi_get_property(_env, _value, key, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, Value(_env, result), Value); +} + +inline MaybeOrValue Object::Get(Value key) const { + napi_value result; + napi_status status = napi_get_property(_env, _value, key, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, Value(_env, result), Value); +} + +inline MaybeOrValue Object::Get(const char* utf8name) const { + napi_value result; + napi_status status = napi_get_named_property(_env, _value, utf8name, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, Value(_env, result), Value); +} + +inline MaybeOrValue Object::Get(const std::string& utf8name) const { + return Get(utf8name.c_str()); +} + +template +inline MaybeOrValue Object::Set(napi_value key, + const ValueType& value) const { + napi_status status = + napi_set_property(_env, _value, key, Value::From(_env, value)); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, status == napi_ok, bool); +} + +template +inline MaybeOrValue Object::Set(Value key, const ValueType& value) const { + napi_status status = + napi_set_property(_env, _value, key, Value::From(_env, value)); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, status == napi_ok, bool); +} + +template +inline MaybeOrValue Object::Set(const char* utf8name, + const ValueType& value) const { + napi_status status = + napi_set_named_property(_env, _value, utf8name, Value::From(_env, value)); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, status == napi_ok, bool); +} + +template +inline MaybeOrValue Object::Set(const std::string& utf8name, + const ValueType& value) const { + return Set(utf8name.c_str(), value); +} + +inline MaybeOrValue Object::Delete(napi_value key) const { + bool result; + napi_status status = napi_delete_property(_env, _value, key, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, result, bool); +} + +inline MaybeOrValue Object::Delete(Value key) const { + bool result; + napi_status status = napi_delete_property(_env, _value, key, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, result, bool); +} + +inline MaybeOrValue Object::Delete(const char* utf8name) const { + return Delete(String::New(_env, utf8name)); +} + +inline MaybeOrValue Object::Delete(const std::string& utf8name) const { + return Delete(String::New(_env, utf8name)); +} + +inline MaybeOrValue Object::Has(uint32_t index) const { + bool result; + napi_status status = napi_has_element(_env, _value, index, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, result, bool); +} + +inline MaybeOrValue Object::Get(uint32_t index) const { + napi_value value; + napi_status status = napi_get_element(_env, _value, index, &value); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, Value(_env, value), Value); +} + +template +inline MaybeOrValue Object::Set(uint32_t index, + const ValueType& value) const { + napi_status status = + napi_set_element(_env, _value, index, Value::From(_env, value)); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, status == napi_ok, bool); +} + +inline MaybeOrValue Object::Delete(uint32_t index) const { + bool result; + napi_status status = napi_delete_element(_env, _value, index, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, result, bool); +} + +inline MaybeOrValue Object::GetPropertyNames() const { + napi_value result; + napi_status status = napi_get_property_names(_env, _value, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, Array(_env, result), Array); +} + +inline MaybeOrValue Object::DefineProperty( + const PropertyDescriptor& property) const { + napi_status status = napi_define_properties( + _env, + _value, + 1, + reinterpret_cast(&property)); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, status == napi_ok, bool); +} + +inline MaybeOrValue Object::DefineProperties( + const std::initializer_list& properties) const { + napi_status status = napi_define_properties( + _env, + _value, + properties.size(), + reinterpret_cast(properties.begin())); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, status == napi_ok, bool); +} + +inline MaybeOrValue Object::DefineProperties( + const std::vector& properties) const { + napi_status status = napi_define_properties( + _env, + _value, + properties.size(), + reinterpret_cast(properties.data())); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, status == napi_ok, bool); +} + +inline MaybeOrValue Object::InstanceOf( + const Function& constructor) const { + bool result; + napi_status status = napi_instanceof(_env, _value, constructor, &result); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, result, bool); +} + +template +inline void Object::AddFinalizer(Finalizer finalizeCallback, T* data) const { + details::FinalizeData* finalizeData = + new details::FinalizeData( + {std::move(finalizeCallback), nullptr}); + napi_status status = + details::AttachData(_env, + *this, + data, + details::FinalizeData::Wrapper, + finalizeData); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED_VOID(_env, status); + } +} + +template +inline void Object::AddFinalizer(Finalizer finalizeCallback, + T* data, + Hint* finalizeHint) const { + details::FinalizeData* finalizeData = + new details::FinalizeData( + {std::move(finalizeCallback), finalizeHint}); + napi_status status = details::AttachData( + _env, + *this, + data, + details::FinalizeData::WrapperWithHint, + finalizeData); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED_VOID(_env, status); + } +} + +#ifdef NAPI_CPP_EXCEPTIONS +inline Object::const_iterator::const_iterator(const Object* object, + const Type type) { + _object = object; + _keys = object->GetPropertyNames(); + _index = type == Type::BEGIN ? 0 : _keys.Length(); +} + +inline Object::const_iterator Napi::Object::begin() const { + const_iterator it(this, Object::const_iterator::Type::BEGIN); + return it; +} + +inline Object::const_iterator Napi::Object::end() const { + const_iterator it(this, Object::const_iterator::Type::END); + return it; +} + +inline Object::const_iterator& Object::const_iterator::operator++() { + ++_index; + return *this; +} + +inline bool Object::const_iterator::operator==( + const const_iterator& other) const { + return _index == other._index; +} + +inline bool Object::const_iterator::operator!=( + const const_iterator& other) const { + return _index != other._index; +} + +inline const std::pair> +Object::const_iterator::operator*() const { + const Value key = _keys[_index]; + const PropertyLValue value = (*_object)[key]; + return {key, value}; +} + +inline Object::iterator::iterator(Object* object, const Type type) { + _object = object; + _keys = object->GetPropertyNames(); + _index = type == Type::BEGIN ? 0 : _keys.Length(); +} + +inline Object::iterator Napi::Object::begin() { + iterator it(this, Object::iterator::Type::BEGIN); + return it; +} + +inline Object::iterator Napi::Object::end() { + iterator it(this, Object::iterator::Type::END); + return it; +} + +inline Object::iterator& Object::iterator::operator++() { + ++_index; + return *this; +} + +inline bool Object::iterator::operator==(const iterator& other) const { + return _index == other._index; +} + +inline bool Object::iterator::operator!=(const iterator& other) const { + return _index != other._index; +} + +inline std::pair> +Object::iterator::operator*() { + Value key = _keys[_index]; + PropertyLValue value = (*_object)[key]; + return {key, value}; +} +#endif // NAPI_CPP_EXCEPTIONS + +#if NAPI_VERSION >= 8 +inline MaybeOrValue Object::Freeze() const { + napi_status status = napi_object_freeze(_env, _value); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, status == napi_ok, bool); +} + +inline MaybeOrValue Object::Seal() const { + napi_status status = napi_object_seal(_env, _value); + NAPI_RETURN_OR_THROW_IF_FAILED(_env, status, status == napi_ok, bool); +} +#endif // NAPI_VERSION >= 8 + +//////////////////////////////////////////////////////////////////////////////// +// External class +//////////////////////////////////////////////////////////////////////////////// + +template +inline External External::New(napi_env env, T* data) { + napi_value value; + napi_status status = + napi_create_external(env, data, nullptr, nullptr, &value); + NAPI_THROW_IF_FAILED(env, status, External()); + return External(env, value); +} + +template +template +inline External External::New(napi_env env, + T* data, + Finalizer finalizeCallback) { + napi_value value; + details::FinalizeData* finalizeData = + new details::FinalizeData( + {std::move(finalizeCallback), nullptr}); + napi_status status = + napi_create_external(env, + data, + details::FinalizeData::Wrapper, + finalizeData, + &value); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED(env, status, External()); + } + return External(env, value); +} + +template +template +inline External External::New(napi_env env, + T* data, + Finalizer finalizeCallback, + Hint* finalizeHint) { + napi_value value; + details::FinalizeData* finalizeData = + new details::FinalizeData( + {std::move(finalizeCallback), finalizeHint}); + napi_status status = napi_create_external( + env, + data, + details::FinalizeData::WrapperWithHint, + finalizeData, + &value); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED(env, status, External()); + } + return External(env, value); +} + +template +inline External::External() : Value() {} + +template +inline External::External(napi_env env, napi_value value) + : Value(env, value) {} + +template +inline T* External::Data() const { + void* data; + napi_status status = napi_get_value_external(_env, _value, &data); + NAPI_THROW_IF_FAILED(_env, status, nullptr); + return reinterpret_cast(data); +} + +//////////////////////////////////////////////////////////////////////////////// +// Array class +//////////////////////////////////////////////////////////////////////////////// + +inline Array Array::New(napi_env env) { + napi_value value; + napi_status status = napi_create_array(env, &value); + NAPI_THROW_IF_FAILED(env, status, Array()); + return Array(env, value); +} + +inline Array Array::New(napi_env env, size_t length) { + napi_value value; + napi_status status = napi_create_array_with_length(env, length, &value); + NAPI_THROW_IF_FAILED(env, status, Array()); + return Array(env, value); +} + +inline Array::Array() : Object() {} + +inline Array::Array(napi_env env, napi_value value) : Object(env, value) {} + +inline uint32_t Array::Length() const { + uint32_t result; + napi_status status = napi_get_array_length(_env, _value, &result); + NAPI_THROW_IF_FAILED(_env, status, 0); + return result; +} + +//////////////////////////////////////////////////////////////////////////////// +// ArrayBuffer class +//////////////////////////////////////////////////////////////////////////////// + +inline ArrayBuffer ArrayBuffer::New(napi_env env, size_t byteLength) { + napi_value value; + void* data; + napi_status status = napi_create_arraybuffer(env, byteLength, &data, &value); + NAPI_THROW_IF_FAILED(env, status, ArrayBuffer()); + + return ArrayBuffer(env, value); +} + +inline ArrayBuffer ArrayBuffer::New(napi_env env, + void* externalData, + size_t byteLength) { + napi_value value; + napi_status status = napi_create_external_arraybuffer( + env, externalData, byteLength, nullptr, nullptr, &value); + NAPI_THROW_IF_FAILED(env, status, ArrayBuffer()); + + return ArrayBuffer(env, value); +} + +template +inline ArrayBuffer ArrayBuffer::New(napi_env env, + void* externalData, + size_t byteLength, + Finalizer finalizeCallback) { + napi_value value; + details::FinalizeData* finalizeData = + new details::FinalizeData( + {std::move(finalizeCallback), nullptr}); + napi_status status = napi_create_external_arraybuffer( + env, + externalData, + byteLength, + details::FinalizeData::Wrapper, + finalizeData, + &value); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED(env, status, ArrayBuffer()); + } + + return ArrayBuffer(env, value); +} + +template +inline ArrayBuffer ArrayBuffer::New(napi_env env, + void* externalData, + size_t byteLength, + Finalizer finalizeCallback, + Hint* finalizeHint) { + napi_value value; + details::FinalizeData* finalizeData = + new details::FinalizeData( + {std::move(finalizeCallback), finalizeHint}); + napi_status status = napi_create_external_arraybuffer( + env, + externalData, + byteLength, + details::FinalizeData::WrapperWithHint, + finalizeData, + &value); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED(env, status, ArrayBuffer()); + } + + return ArrayBuffer(env, value); +} + +inline ArrayBuffer::ArrayBuffer() : Object() {} + +inline ArrayBuffer::ArrayBuffer(napi_env env, napi_value value) + : Object(env, value) {} + +inline void* ArrayBuffer::Data() { + void* data; + napi_status status = napi_get_arraybuffer_info(_env, _value, &data, nullptr); + NAPI_THROW_IF_FAILED(_env, status, nullptr); + return data; +} + +inline size_t ArrayBuffer::ByteLength() { + size_t length; + napi_status status = + napi_get_arraybuffer_info(_env, _value, nullptr, &length); + NAPI_THROW_IF_FAILED(_env, status, 0); + return length; +} + +#if NAPI_VERSION >= 7 +inline bool ArrayBuffer::IsDetached() const { + bool detached; + napi_status status = napi_is_detached_arraybuffer(_env, _value, &detached); + NAPI_THROW_IF_FAILED(_env, status, false); + return detached; +} + +inline void ArrayBuffer::Detach() { + napi_status status = napi_detach_arraybuffer(_env, _value); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} +#endif // NAPI_VERSION >= 7 + +//////////////////////////////////////////////////////////////////////////////// +// DataView class +//////////////////////////////////////////////////////////////////////////////// +inline DataView DataView::New(napi_env env, Napi::ArrayBuffer arrayBuffer) { + return New(env, arrayBuffer, 0, arrayBuffer.ByteLength()); +} + +inline DataView DataView::New(napi_env env, + Napi::ArrayBuffer arrayBuffer, + size_t byteOffset) { + if (byteOffset > arrayBuffer.ByteLength()) { + NAPI_THROW(RangeError::New( + env, "Start offset is outside the bounds of the buffer"), + DataView()); + } + return New( + env, arrayBuffer, byteOffset, arrayBuffer.ByteLength() - byteOffset); +} + +inline DataView DataView::New(napi_env env, + Napi::ArrayBuffer arrayBuffer, + size_t byteOffset, + size_t byteLength) { + if (byteOffset + byteLength > arrayBuffer.ByteLength()) { + NAPI_THROW(RangeError::New(env, "Invalid DataView length"), DataView()); + } + napi_value value; + napi_status status = + napi_create_dataview(env, byteLength, arrayBuffer, byteOffset, &value); + NAPI_THROW_IF_FAILED(env, status, DataView()); + return DataView(env, value); +} + +inline DataView::DataView() : Object() {} + +inline DataView::DataView(napi_env env, napi_value value) : Object(env, value) { + napi_status status = napi_get_dataview_info(_env, + _value /* dataView */, + &_length /* byteLength */, + &_data /* data */, + nullptr /* arrayBuffer */, + nullptr /* byteOffset */); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +inline Napi::ArrayBuffer DataView::ArrayBuffer() const { + napi_value arrayBuffer; + napi_status status = napi_get_dataview_info(_env, + _value /* dataView */, + nullptr /* byteLength */, + nullptr /* data */, + &arrayBuffer /* arrayBuffer */, + nullptr /* byteOffset */); + NAPI_THROW_IF_FAILED(_env, status, Napi::ArrayBuffer()); + return Napi::ArrayBuffer(_env, arrayBuffer); +} + +inline size_t DataView::ByteOffset() const { + size_t byteOffset; + napi_status status = napi_get_dataview_info(_env, + _value /* dataView */, + nullptr /* byteLength */, + nullptr /* data */, + nullptr /* arrayBuffer */, + &byteOffset /* byteOffset */); + NAPI_THROW_IF_FAILED(_env, status, 0); + return byteOffset; +} + +inline size_t DataView::ByteLength() const { + return _length; +} + +inline void* DataView::Data() const { + return _data; +} + +inline float DataView::GetFloat32(size_t byteOffset) const { + return ReadData(byteOffset); +} + +inline double DataView::GetFloat64(size_t byteOffset) const { + return ReadData(byteOffset); +} + +inline int8_t DataView::GetInt8(size_t byteOffset) const { + return ReadData(byteOffset); +} + +inline int16_t DataView::GetInt16(size_t byteOffset) const { + return ReadData(byteOffset); +} + +inline int32_t DataView::GetInt32(size_t byteOffset) const { + return ReadData(byteOffset); +} + +inline uint8_t DataView::GetUint8(size_t byteOffset) const { + return ReadData(byteOffset); +} + +inline uint16_t DataView::GetUint16(size_t byteOffset) const { + return ReadData(byteOffset); +} + +inline uint32_t DataView::GetUint32(size_t byteOffset) const { + return ReadData(byteOffset); +} + +inline void DataView::SetFloat32(size_t byteOffset, float value) const { + WriteData(byteOffset, value); +} + +inline void DataView::SetFloat64(size_t byteOffset, double value) const { + WriteData(byteOffset, value); +} + +inline void DataView::SetInt8(size_t byteOffset, int8_t value) const { + WriteData(byteOffset, value); +} + +inline void DataView::SetInt16(size_t byteOffset, int16_t value) const { + WriteData(byteOffset, value); +} + +inline void DataView::SetInt32(size_t byteOffset, int32_t value) const { + WriteData(byteOffset, value); +} + +inline void DataView::SetUint8(size_t byteOffset, uint8_t value) const { + WriteData(byteOffset, value); +} + +inline void DataView::SetUint16(size_t byteOffset, uint16_t value) const { + WriteData(byteOffset, value); +} + +inline void DataView::SetUint32(size_t byteOffset, uint32_t value) const { + WriteData(byteOffset, value); +} + +template +inline T DataView::ReadData(size_t byteOffset) const { + if (byteOffset + sizeof(T) > _length || + byteOffset + sizeof(T) < byteOffset) { // overflow + NAPI_THROW( + RangeError::New(_env, "Offset is outside the bounds of the DataView"), + 0); + } + + return *reinterpret_cast(static_cast(_data) + byteOffset); +} + +template +inline void DataView::WriteData(size_t byteOffset, T value) const { + if (byteOffset + sizeof(T) > _length || + byteOffset + sizeof(T) < byteOffset) { // overflow + NAPI_THROW_VOID( + RangeError::New(_env, "Offset is outside the bounds of the DataView")); + } + + *reinterpret_cast(static_cast(_data) + byteOffset) = value; +} + +//////////////////////////////////////////////////////////////////////////////// +// TypedArray class +//////////////////////////////////////////////////////////////////////////////// + +inline TypedArray::TypedArray() + : Object(), _type(napi_typedarray_type::napi_int8_array), _length(0) {} + +inline TypedArray::TypedArray(napi_env env, napi_value value) + : Object(env, value), + _type(napi_typedarray_type::napi_int8_array), + _length(0) { + if (value != nullptr) { + napi_status status = + napi_get_typedarray_info(_env, + _value, + &const_cast(this)->_type, + &const_cast(this)->_length, + nullptr, + nullptr, + nullptr); + NAPI_THROW_IF_FAILED_VOID(_env, status); + } +} + +inline TypedArray::TypedArray(napi_env env, + napi_value value, + napi_typedarray_type type, + size_t length) + : Object(env, value), _type(type), _length(length) {} + +inline napi_typedarray_type TypedArray::TypedArrayType() const { + return _type; +} + +inline uint8_t TypedArray::ElementSize() const { + switch (_type) { + case napi_int8_array: + case napi_uint8_array: + case napi_uint8_clamped_array: + return 1; + case napi_int16_array: + case napi_uint16_array: + return 2; + case napi_int32_array: + case napi_uint32_array: + case napi_float32_array: + return 4; + case napi_float64_array: +#if (NAPI_VERSION > 5) + case napi_bigint64_array: + case napi_biguint64_array: +#endif // (NAPI_VERSION > 5) + return 8; + default: + return 0; + } +} + +inline size_t TypedArray::ElementLength() const { + return _length; +} + +inline size_t TypedArray::ByteOffset() const { + size_t byteOffset; + napi_status status = napi_get_typedarray_info( + _env, _value, nullptr, nullptr, nullptr, nullptr, &byteOffset); + NAPI_THROW_IF_FAILED(_env, status, 0); + return byteOffset; +} + +inline size_t TypedArray::ByteLength() const { + return ElementSize() * ElementLength(); +} + +inline Napi::ArrayBuffer TypedArray::ArrayBuffer() const { + napi_value arrayBuffer; + napi_status status = napi_get_typedarray_info( + _env, _value, nullptr, nullptr, nullptr, &arrayBuffer, nullptr); + NAPI_THROW_IF_FAILED(_env, status, Napi::ArrayBuffer()); + return Napi::ArrayBuffer(_env, arrayBuffer); +} + +//////////////////////////////////////////////////////////////////////////////// +// TypedArrayOf class +//////////////////////////////////////////////////////////////////////////////// + +template +inline TypedArrayOf TypedArrayOf::New(napi_env env, + size_t elementLength, + napi_typedarray_type type) { + Napi::ArrayBuffer arrayBuffer = + Napi::ArrayBuffer::New(env, elementLength * sizeof(T)); + return New(env, elementLength, arrayBuffer, 0, type); +} + +template +inline TypedArrayOf TypedArrayOf::New(napi_env env, + size_t elementLength, + Napi::ArrayBuffer arrayBuffer, + size_t bufferOffset, + napi_typedarray_type type) { + napi_value value; + napi_status status = napi_create_typedarray( + env, type, elementLength, arrayBuffer, bufferOffset, &value); + NAPI_THROW_IF_FAILED(env, status, TypedArrayOf()); + + return TypedArrayOf( + env, + value, + type, + elementLength, + reinterpret_cast(reinterpret_cast(arrayBuffer.Data()) + + bufferOffset)); +} + +template +inline TypedArrayOf::TypedArrayOf() : TypedArray(), _data(nullptr) {} + +template +inline TypedArrayOf::TypedArrayOf(napi_env env, napi_value value) + : TypedArray(env, value), _data(nullptr) { + napi_status status = napi_ok; + if (value != nullptr) { + void* data = nullptr; + status = napi_get_typedarray_info( + _env, _value, &_type, &_length, &data, nullptr, nullptr); + _data = static_cast(data); + } else { + _type = TypedArrayTypeForPrimitiveType(); + _length = 0; + } + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +template +inline TypedArrayOf::TypedArrayOf(napi_env env, + napi_value value, + napi_typedarray_type type, + size_t length, + T* data) + : TypedArray(env, value, type, length), _data(data) { + if (!(type == TypedArrayTypeForPrimitiveType() || + (type == napi_uint8_clamped_array && + std::is_same::value))) { + NAPI_THROW_VOID(TypeError::New( + env, + "Array type must match the template parameter. " + "(Uint8 arrays may optionally have the \"clamped\" array type.)")); + } +} + +template +inline T& TypedArrayOf::operator[](size_t index) { + return _data[index]; +} + +template +inline const T& TypedArrayOf::operator[](size_t index) const { + return _data[index]; +} + +template +inline T* TypedArrayOf::Data() { + return _data; +} + +template +inline const T* TypedArrayOf::Data() const { + return _data; +} + +//////////////////////////////////////////////////////////////////////////////// +// Function class +//////////////////////////////////////////////////////////////////////////////// + +template +inline napi_status CreateFunction(napi_env env, + const char* utf8name, + napi_callback cb, + CbData* data, + napi_value* result) { + napi_status status = + napi_create_function(env, utf8name, NAPI_AUTO_LENGTH, cb, data, result); + if (status == napi_ok) { + status = Napi::details::AttachData(env, *result, data); + } + + return status; +} + +template +inline Function Function::New(napi_env env, const char* utf8name, void* data) { + napi_value result = nullptr; + napi_status status = napi_create_function(env, + utf8name, + NAPI_AUTO_LENGTH, + details::TemplatedVoidCallback, + data, + &result); + NAPI_THROW_IF_FAILED(env, status, Function()); + return Function(env, result); +} + +template +inline Function Function::New(napi_env env, const char* utf8name, void* data) { + napi_value result = nullptr; + napi_status status = napi_create_function(env, + utf8name, + NAPI_AUTO_LENGTH, + details::TemplatedCallback, + data, + &result); + NAPI_THROW_IF_FAILED(env, status, Function()); + return Function(env, result); +} + +template +inline Function Function::New(napi_env env, + const std::string& utf8name, + void* data) { + return Function::New(env, utf8name.c_str(), data); +} + +template +inline Function Function::New(napi_env env, + const std::string& utf8name, + void* data) { + return Function::New(env, utf8name.c_str(), data); +} + +template +inline Function Function::New(napi_env env, + Callable cb, + const char* utf8name, + void* data) { + using ReturnType = decltype(cb(CallbackInfo(nullptr, nullptr))); + using CbData = details::CallbackData; + auto callbackData = new CbData{std::move(cb), data}; + + napi_value value; + napi_status status = + CreateFunction(env, utf8name, CbData::Wrapper, callbackData, &value); + if (status != napi_ok) { + delete callbackData; + NAPI_THROW_IF_FAILED(env, status, Function()); + } + + return Function(env, value); +} + +template +inline Function Function::New(napi_env env, + Callable cb, + const std::string& utf8name, + void* data) { + return New(env, cb, utf8name.c_str(), data); +} + +inline Function::Function() : Object() {} + +inline Function::Function(napi_env env, napi_value value) + : Object(env, value) {} + +inline MaybeOrValue Function::operator()( + const std::initializer_list& args) const { + return Call(Env().Undefined(), args); +} + +inline MaybeOrValue Function::Call( + const std::initializer_list& args) const { + return Call(Env().Undefined(), args); +} + +inline MaybeOrValue Function::Call( + const std::vector& args) const { + return Call(Env().Undefined(), args); +} + +inline MaybeOrValue Function::Call( + const std::vector& args) const { + return Call(Env().Undefined(), args); +} + +inline MaybeOrValue Function::Call(size_t argc, + const napi_value* args) const { + return Call(Env().Undefined(), argc, args); +} + +inline MaybeOrValue Function::Call( + napi_value recv, const std::initializer_list& args) const { + return Call(recv, args.size(), args.begin()); +} + +inline MaybeOrValue Function::Call( + napi_value recv, const std::vector& args) const { + return Call(recv, args.size(), args.data()); +} + +inline MaybeOrValue Function::Call( + napi_value recv, const std::vector& args) const { + const size_t argc = args.size(); + const size_t stackArgsCount = 6; + napi_value stackArgs[stackArgsCount]; + std::vector heapArgs; + napi_value* argv; + if (argc <= stackArgsCount) { + argv = stackArgs; + } else { + heapArgs.resize(argc); + argv = heapArgs.data(); + } + + for (size_t index = 0; index < argc; index++) { + argv[index] = static_cast(args[index]); + } + + return Call(recv, argc, argv); +} + +inline MaybeOrValue Function::Call(napi_value recv, + size_t argc, + const napi_value* args) const { + napi_value result; + napi_status status = + napi_call_function(_env, recv, _value, argc, args, &result); + NAPI_RETURN_OR_THROW_IF_FAILED( + _env, status, Napi::Value(_env, result), Napi::Value); +} + +inline MaybeOrValue Function::MakeCallback( + napi_value recv, + const std::initializer_list& args, + napi_async_context context) const { + return MakeCallback(recv, args.size(), args.begin(), context); +} + +inline MaybeOrValue Function::MakeCallback( + napi_value recv, + const std::vector& args, + napi_async_context context) const { + return MakeCallback(recv, args.size(), args.data(), context); +} + +inline MaybeOrValue Function::MakeCallback( + napi_value recv, + size_t argc, + const napi_value* args, + napi_async_context context) const { + napi_value result; + napi_status status = + napi_make_callback(_env, context, recv, _value, argc, args, &result); + NAPI_RETURN_OR_THROW_IF_FAILED( + _env, status, Napi::Value(_env, result), Napi::Value); +} + +inline MaybeOrValue Function::New( + const std::initializer_list& args) const { + return New(args.size(), args.begin()); +} + +inline MaybeOrValue Function::New( + const std::vector& args) const { + return New(args.size(), args.data()); +} + +inline MaybeOrValue Function::New(size_t argc, + const napi_value* args) const { + napi_value result; + napi_status status = napi_new_instance(_env, _value, argc, args, &result); + NAPI_RETURN_OR_THROW_IF_FAILED( + _env, status, Napi::Object(_env, result), Napi::Object); +} + +//////////////////////////////////////////////////////////////////////////////// +// Promise class +//////////////////////////////////////////////////////////////////////////////// + +inline Promise::Deferred Promise::Deferred::New(napi_env env) { + return Promise::Deferred(env); +} + +inline Promise::Deferred::Deferred(napi_env env) : _env(env) { + napi_status status = napi_create_promise(_env, &_deferred, &_promise); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +inline Promise Promise::Deferred::Promise() const { + return Napi::Promise(_env, _promise); +} + +inline Napi::Env Promise::Deferred::Env() const { + return Napi::Env(_env); +} + +inline void Promise::Deferred::Resolve(napi_value value) const { + napi_status status = napi_resolve_deferred(_env, _deferred, value); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +inline void Promise::Deferred::Reject(napi_value value) const { + napi_status status = napi_reject_deferred(_env, _deferred, value); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +inline Promise::Promise(napi_env env, napi_value value) : Object(env, value) {} + +//////////////////////////////////////////////////////////////////////////////// +// Buffer class +//////////////////////////////////////////////////////////////////////////////// + +template +inline Buffer Buffer::New(napi_env env, size_t length) { + napi_value value; + void* data; + napi_status status = + napi_create_buffer(env, length * sizeof(T), &data, &value); + NAPI_THROW_IF_FAILED(env, status, Buffer()); + return Buffer(env, value, length, static_cast(data)); +} + +template +inline Buffer Buffer::New(napi_env env, T* data, size_t length) { + napi_value value; + napi_status status = napi_create_external_buffer( + env, length * sizeof(T), data, nullptr, nullptr, &value); + NAPI_THROW_IF_FAILED(env, status, Buffer()); + return Buffer(env, value, length, data); +} + +template +template +inline Buffer Buffer::New(napi_env env, + T* data, + size_t length, + Finalizer finalizeCallback) { + napi_value value; + details::FinalizeData* finalizeData = + new details::FinalizeData( + {std::move(finalizeCallback), nullptr}); + napi_status status = + napi_create_external_buffer(env, + length * sizeof(T), + data, + details::FinalizeData::Wrapper, + finalizeData, + &value); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED(env, status, Buffer()); + } + return Buffer(env, value, length, data); +} + +template +template +inline Buffer Buffer::New(napi_env env, + T* data, + size_t length, + Finalizer finalizeCallback, + Hint* finalizeHint) { + napi_value value; + details::FinalizeData* finalizeData = + new details::FinalizeData( + {std::move(finalizeCallback), finalizeHint}); + napi_status status = napi_create_external_buffer( + env, + length * sizeof(T), + data, + details::FinalizeData::WrapperWithHint, + finalizeData, + &value); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED(env, status, Buffer()); + } + return Buffer(env, value, length, data); +} + +template +inline Buffer Buffer::Copy(napi_env env, const T* data, size_t length) { + napi_value value; + napi_status status = + napi_create_buffer_copy(env, length * sizeof(T), data, nullptr, &value); + NAPI_THROW_IF_FAILED(env, status, Buffer()); + return Buffer(env, value); +} + +template +inline Buffer::Buffer() : Uint8Array(), _length(0), _data(nullptr) {} + +template +inline Buffer::Buffer(napi_env env, napi_value value) + : Uint8Array(env, value), _length(0), _data(nullptr) {} + +template +inline Buffer::Buffer(napi_env env, napi_value value, size_t length, T* data) + : Uint8Array(env, value), _length(length), _data(data) {} + +template +inline size_t Buffer::Length() const { + EnsureInfo(); + return _length; +} + +template +inline T* Buffer::Data() const { + EnsureInfo(); + return _data; +} + +template +inline void Buffer::EnsureInfo() const { + // The Buffer instance may have been constructed from a napi_value whose + // length/data are not yet known. Fetch and cache these values just once, + // since they can never change during the lifetime of the Buffer. + if (_data == nullptr) { + size_t byteLength; + void* voidData; + napi_status status = + napi_get_buffer_info(_env, _value, &voidData, &byteLength); + NAPI_THROW_IF_FAILED_VOID(_env, status); + _length = byteLength / sizeof(T); + _data = static_cast(voidData); + } +} + +//////////////////////////////////////////////////////////////////////////////// +// Error class +//////////////////////////////////////////////////////////////////////////////// + +inline Error Error::New(napi_env env) { + napi_status status; + napi_value error = nullptr; + bool is_exception_pending; + napi_extended_error_info last_error_info_copy; + + { + // We must retrieve the last error info before doing anything else because + // doing anything else will replace the last error info. + const napi_extended_error_info* last_error_info; + status = napi_get_last_error_info(env, &last_error_info); + NAPI_FATAL_IF_FAILED(status, "Error::New", "napi_get_last_error_info"); + + // All fields of the `napi_extended_error_info` structure gets reset in + // subsequent Node-API function calls on the same `env`. This includes a + // call to `napi_is_exception_pending()`. So here it is necessary to make a + // copy of the information as the `error_code` field is used later on. + memcpy(&last_error_info_copy, + last_error_info, + sizeof(napi_extended_error_info)); + } + + status = napi_is_exception_pending(env, &is_exception_pending); + NAPI_FATAL_IF_FAILED(status, "Error::New", "napi_is_exception_pending"); + + // A pending exception takes precedence over any internal error status. + if (is_exception_pending) { + status = napi_get_and_clear_last_exception(env, &error); + NAPI_FATAL_IF_FAILED( + status, "Error::New", "napi_get_and_clear_last_exception"); + } else { + const char* error_message = last_error_info_copy.error_message != nullptr + ? last_error_info_copy.error_message + : "Error in native callback"; + + napi_value message; + status = napi_create_string_utf8( + env, error_message, std::strlen(error_message), &message); + NAPI_FATAL_IF_FAILED(status, "Error::New", "napi_create_string_utf8"); + + switch (last_error_info_copy.error_code) { + case napi_object_expected: + case napi_string_expected: + case napi_boolean_expected: + case napi_number_expected: + status = napi_create_type_error(env, nullptr, message, &error); + break; + default: + status = napi_create_error(env, nullptr, message, &error); + break; + } + NAPI_FATAL_IF_FAILED(status, "Error::New", "napi_create_error"); + } + + return Error(env, error); +} + +inline Error Error::New(napi_env env, const char* message) { + return Error::New( + env, message, std::strlen(message), napi_create_error); +} + +inline Error Error::New(napi_env env, const std::string& message) { + return Error::New( + env, message.c_str(), message.size(), napi_create_error); +} + +inline NAPI_NO_RETURN void Error::Fatal(const char* location, + const char* message) { + napi_fatal_error(location, NAPI_AUTO_LENGTH, message, NAPI_AUTO_LENGTH); +} + +inline Error::Error() : ObjectReference() {} + +inline Error::Error(napi_env env, napi_value value) + : ObjectReference(env, nullptr) { + if (value != nullptr) { + // Attempting to create a reference on the error object. + // If it's not a Object/Function/Symbol, this call will return an error + // status. + napi_status status = napi_create_reference(env, value, 1, &_ref); + + if (status != napi_ok) { + napi_value wrappedErrorObj; + + // Create an error object + status = napi_create_object(env, &wrappedErrorObj); + NAPI_FATAL_IF_FAILED(status, "Error::Error", "napi_create_object"); + + // property flag that we attach to show the error object is wrapped + napi_property_descriptor wrapObjFlag = { + ERROR_WRAP_VALUE(), // Unique GUID identifier since Symbol isn't a + // viable option + nullptr, + nullptr, + nullptr, + nullptr, + Value::From(env, value), + napi_enumerable, + nullptr}; + + status = napi_define_properties(env, wrappedErrorObj, 1, &wrapObjFlag); + NAPI_FATAL_IF_FAILED(status, "Error::Error", "napi_define_properties"); + + // Create a reference on the newly wrapped object + status = napi_create_reference(env, wrappedErrorObj, 1, &_ref); + } + + // Avoid infinite recursion in the failure case. + NAPI_FATAL_IF_FAILED(status, "Error::Error", "napi_create_reference"); + } +} + +inline Object Error::Value() const { + if (_ref == nullptr) { + return Object(_env, nullptr); + } + + napi_value refValue; + napi_status status = napi_get_reference_value(_env, _ref, &refValue); + NAPI_THROW_IF_FAILED(_env, status, Object()); + + napi_valuetype type; + status = napi_typeof(_env, refValue, &type); + NAPI_THROW_IF_FAILED(_env, status, Object()); + + // If refValue isn't a symbol, then we proceed to whether the refValue has the + // wrapped error flag + if (type != napi_symbol) { + // We are checking if the object is wrapped + bool isWrappedObject = false; + + status = napi_has_property(_env, + refValue, + String::From(_env, ERROR_WRAP_VALUE()), + &isWrappedObject); + + // Don't care about status + if (isWrappedObject) { + napi_value unwrappedValue; + status = napi_get_property(_env, + refValue, + String::From(_env, ERROR_WRAP_VALUE()), + &unwrappedValue); + NAPI_THROW_IF_FAILED(_env, status, Object()); + + return Object(_env, unwrappedValue); + } + } + + return Object(_env, refValue); +} + +inline Error::Error(Error&& other) : ObjectReference(std::move(other)) {} + +inline Error& Error::operator=(Error&& other) { + static_cast*>(this)->operator=(std::move(other)); + return *this; +} + +inline Error::Error(const Error& other) : ObjectReference(other) {} + +inline Error& Error::operator=(const Error& other) { + Reset(); + + _env = other.Env(); + HandleScope scope(_env); + + napi_value value = other.Value(); + if (value != nullptr) { + napi_status status = napi_create_reference(_env, value, 1, &_ref); + NAPI_THROW_IF_FAILED(_env, status, *this); + } + + return *this; +} + +inline const std::string& Error::Message() const NAPI_NOEXCEPT { + if (_message.size() == 0 && _env != nullptr) { +#ifdef NAPI_CPP_EXCEPTIONS + try { + _message = Get("message").As(); + } catch (...) { + // Catch all errors here, to include e.g. a std::bad_alloc from + // the std::string::operator=, because this method may not throw. + } +#else // NAPI_CPP_EXCEPTIONS +#if defined(NODE_ADDON_API_ENABLE_MAYBE) + Napi::Value message_val; + if (Get("message").UnwrapTo(&message_val)) { + _message = message_val.As(); + } +#else + _message = Get("message").As(); +#endif +#endif // NAPI_CPP_EXCEPTIONS + } + return _message; +} + +// we created an object on the &_ref +inline void Error::ThrowAsJavaScriptException() const { + HandleScope scope(_env); + if (!IsEmpty()) { +#ifdef NODE_API_SWALLOW_UNTHROWABLE_EXCEPTIONS + bool pendingException = false; + + // check if there is already a pending exception. If so don't try to throw a + // new one as that is not allowed/possible + napi_status status = napi_is_exception_pending(_env, &pendingException); + + if ((status != napi_ok) || + ((status == napi_ok) && (pendingException == false))) { + // We intentionally don't use `NAPI_THROW_*` macros here to ensure + // that there is no possible recursion as `ThrowAsJavaScriptException` + // is part of `NAPI_THROW_*` macro definition for noexcept. + + status = napi_throw(_env, Value()); + + if (status == napi_pending_exception) { + // The environment must be terminating as we checked earlier and there + // was no pending exception. In this case continuing will result + // in a fatal error and there is nothing the author has done incorrectly + // in their code that is worth flagging through a fatal error + return; + } + } else { + status = napi_pending_exception; + } +#else + // We intentionally don't use `NAPI_THROW_*` macros here to ensure + // that there is no possible recursion as `ThrowAsJavaScriptException` + // is part of `NAPI_THROW_*` macro definition for noexcept. + + napi_status status = napi_throw(_env, Value()); +#endif + +#ifdef NAPI_CPP_EXCEPTIONS + if (status != napi_ok) { + throw Error::New(_env); + } +#else // NAPI_CPP_EXCEPTIONS + NAPI_FATAL_IF_FAILED( + status, "Error::ThrowAsJavaScriptException", "napi_throw"); +#endif // NAPI_CPP_EXCEPTIONS + } +} + +#ifdef NAPI_CPP_EXCEPTIONS + +inline const char* Error::what() const NAPI_NOEXCEPT { + return Message().c_str(); +} + +#endif // NAPI_CPP_EXCEPTIONS + +inline const char* Error::ERROR_WRAP_VALUE() NAPI_NOEXCEPT { + return "4bda9e7e-4913-4dbc-95de-891cbf66598e-errorVal"; +} + +template +inline TError Error::New(napi_env env, + const char* message, + size_t length, + create_error_fn create_error) { + napi_value str; + napi_status status = napi_create_string_utf8(env, message, length, &str); + NAPI_THROW_IF_FAILED(env, status, TError()); + + napi_value error; + status = create_error(env, nullptr, str, &error); + NAPI_THROW_IF_FAILED(env, status, TError()); + + return TError(env, error); +} + +inline TypeError TypeError::New(napi_env env, const char* message) { + return Error::New( + env, message, std::strlen(message), napi_create_type_error); +} + +inline TypeError TypeError::New(napi_env env, const std::string& message) { + return Error::New( + env, message.c_str(), message.size(), napi_create_type_error); +} + +inline TypeError::TypeError() : Error() {} + +inline TypeError::TypeError(napi_env env, napi_value value) + : Error(env, value) {} + +inline RangeError RangeError::New(napi_env env, const char* message) { + return Error::New( + env, message, std::strlen(message), napi_create_range_error); +} + +inline RangeError RangeError::New(napi_env env, const std::string& message) { + return Error::New( + env, message.c_str(), message.size(), napi_create_range_error); +} + +inline RangeError::RangeError() : Error() {} + +inline RangeError::RangeError(napi_env env, napi_value value) + : Error(env, value) {} + +//////////////////////////////////////////////////////////////////////////////// +// Reference class +//////////////////////////////////////////////////////////////////////////////// + +template +inline Reference Reference::New(const T& value, + uint32_t initialRefcount) { + napi_env env = value.Env(); + napi_value val = value; + + if (val == nullptr) { + return Reference(env, nullptr); + } + + napi_ref ref; + napi_status status = napi_create_reference(env, value, initialRefcount, &ref); + NAPI_THROW_IF_FAILED(env, status, Reference()); + + return Reference(env, ref); +} + +template +inline Reference::Reference() + : _env(nullptr), _ref(nullptr), _suppressDestruct(false) {} + +template +inline Reference::Reference(napi_env env, napi_ref ref) + : _env(env), _ref(ref), _suppressDestruct(false) {} + +template +inline Reference::~Reference() { + if (_ref != nullptr) { + if (!_suppressDestruct) { + napi_delete_reference(_env, _ref); + } + + _ref = nullptr; + } +} + +template +inline Reference::Reference(Reference&& other) + : _env(other._env), + _ref(other._ref), + _suppressDestruct(other._suppressDestruct) { + other._env = nullptr; + other._ref = nullptr; + other._suppressDestruct = false; +} + +template +inline Reference& Reference::operator=(Reference&& other) { + Reset(); + _env = other._env; + _ref = other._ref; + _suppressDestruct = other._suppressDestruct; + other._env = nullptr; + other._ref = nullptr; + other._suppressDestruct = false; + return *this; +} + +template +inline Reference::Reference(const Reference& other) + : _env(other._env), _ref(nullptr), _suppressDestruct(false) { + HandleScope scope(_env); + + napi_value value = other.Value(); + if (value != nullptr) { + // Copying is a limited scenario (currently only used for Error object) and + // always creates a strong reference to the given value even if the incoming + // reference is weak. + napi_status status = napi_create_reference(_env, value, 1, &_ref); + NAPI_FATAL_IF_FAILED( + status, "Reference::Reference", "napi_create_reference"); + } +} + +template +inline Reference::operator napi_ref() const { + return _ref; +} + +template +inline bool Reference::operator==(const Reference& other) const { + HandleScope scope(_env); + return this->Value().StrictEquals(other.Value()); +} + +template +inline bool Reference::operator!=(const Reference& other) const { + return !this->operator==(other); +} + +template +inline Napi::Env Reference::Env() const { + return Napi::Env(_env); +} + +template +inline bool Reference::IsEmpty() const { + return _ref == nullptr; +} + +template +inline T Reference::Value() const { + if (_ref == nullptr) { + return T(_env, nullptr); + } + + napi_value value; + napi_status status = napi_get_reference_value(_env, _ref, &value); + NAPI_THROW_IF_FAILED(_env, status, T()); + return T(_env, value); +} + +template +inline uint32_t Reference::Ref() const { + uint32_t result; + napi_status status = napi_reference_ref(_env, _ref, &result); + NAPI_THROW_IF_FAILED(_env, status, 0); + return result; +} + +template +inline uint32_t Reference::Unref() const { + uint32_t result; + napi_status status = napi_reference_unref(_env, _ref, &result); + NAPI_THROW_IF_FAILED(_env, status, 0); + return result; +} + +template +inline void Reference::Reset() { + if (_ref != nullptr) { + napi_status status = napi_delete_reference(_env, _ref); + NAPI_THROW_IF_FAILED_VOID(_env, status); + _ref = nullptr; + } +} + +template +inline void Reference::Reset(const T& value, uint32_t refcount) { + Reset(); + _env = value.Env(); + + napi_value val = value; + if (val != nullptr) { + napi_status status = napi_create_reference(_env, value, refcount, &_ref); + NAPI_THROW_IF_FAILED_VOID(_env, status); + } +} + +template +inline void Reference::SuppressDestruct() { + _suppressDestruct = true; +} + +template +inline Reference Weak(T value) { + return Reference::New(value, 0); +} + +inline ObjectReference Weak(Object value) { + return Reference::New(value, 0); +} + +inline FunctionReference Weak(Function value) { + return Reference::New(value, 0); +} + +template +inline Reference Persistent(T value) { + return Reference::New(value, 1); +} + +inline ObjectReference Persistent(Object value) { + return Reference::New(value, 1); +} + +inline FunctionReference Persistent(Function value) { + return Reference::New(value, 1); +} + +//////////////////////////////////////////////////////////////////////////////// +// ObjectReference class +//////////////////////////////////////////////////////////////////////////////// + +inline ObjectReference::ObjectReference() : Reference() {} + +inline ObjectReference::ObjectReference(napi_env env, napi_ref ref) + : Reference(env, ref) {} + +inline ObjectReference::ObjectReference(Reference&& other) + : Reference(std::move(other)) {} + +inline ObjectReference& ObjectReference::operator=(Reference&& other) { + static_cast*>(this)->operator=(std::move(other)); + return *this; +} + +inline ObjectReference::ObjectReference(ObjectReference&& other) + : Reference(std::move(other)) {} + +inline ObjectReference& ObjectReference::operator=(ObjectReference&& other) { + static_cast*>(this)->operator=(std::move(other)); + return *this; +} + +inline ObjectReference::ObjectReference(const ObjectReference& other) + : Reference(other) {} + +inline MaybeOrValue ObjectReference::Get( + const char* utf8name) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value().Get(utf8name); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap())); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Value(); + } + return scope.Escape(result); +#endif +} + +inline MaybeOrValue ObjectReference::Get( + const std::string& utf8name) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value().Get(utf8name); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap())); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Value(); + } + return scope.Escape(result); +#endif +} + +inline MaybeOrValue ObjectReference::Set(const char* utf8name, + napi_value value) const { + HandleScope scope(_env); + return Value().Set(utf8name, value); +} + +inline MaybeOrValue ObjectReference::Set(const char* utf8name, + Napi::Value value) const { + HandleScope scope(_env); + return Value().Set(utf8name, value); +} + +inline MaybeOrValue ObjectReference::Set(const char* utf8name, + const char* utf8value) const { + HandleScope scope(_env); + return Value().Set(utf8name, utf8value); +} + +inline MaybeOrValue ObjectReference::Set(const char* utf8name, + bool boolValue) const { + HandleScope scope(_env); + return Value().Set(utf8name, boolValue); +} + +inline MaybeOrValue ObjectReference::Set(const char* utf8name, + double numberValue) const { + HandleScope scope(_env); + return Value().Set(utf8name, numberValue); +} + +inline MaybeOrValue ObjectReference::Set(const std::string& utf8name, + napi_value value) const { + HandleScope scope(_env); + return Value().Set(utf8name, value); +} + +inline MaybeOrValue ObjectReference::Set(const std::string& utf8name, + Napi::Value value) const { + HandleScope scope(_env); + return Value().Set(utf8name, value); +} + +inline MaybeOrValue ObjectReference::Set(const std::string& utf8name, + std::string& utf8value) const { + HandleScope scope(_env); + return Value().Set(utf8name, utf8value); +} + +inline MaybeOrValue ObjectReference::Set(const std::string& utf8name, + bool boolValue) const { + HandleScope scope(_env); + return Value().Set(utf8name, boolValue); +} + +inline MaybeOrValue ObjectReference::Set(const std::string& utf8name, + double numberValue) const { + HandleScope scope(_env); + return Value().Set(utf8name, numberValue); +} + +inline MaybeOrValue ObjectReference::Get(uint32_t index) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value().Get(index); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap())); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Value(); + } + return scope.Escape(result); +#endif +} + +inline MaybeOrValue ObjectReference::Set(uint32_t index, + napi_value value) const { + HandleScope scope(_env); + return Value().Set(index, value); +} + +inline MaybeOrValue ObjectReference::Set(uint32_t index, + Napi::Value value) const { + HandleScope scope(_env); + return Value().Set(index, value); +} + +inline MaybeOrValue ObjectReference::Set(uint32_t index, + const char* utf8value) const { + HandleScope scope(_env); + return Value().Set(index, utf8value); +} + +inline MaybeOrValue ObjectReference::Set( + uint32_t index, const std::string& utf8value) const { + HandleScope scope(_env); + return Value().Set(index, utf8value); +} + +inline MaybeOrValue ObjectReference::Set(uint32_t index, + bool boolValue) const { + HandleScope scope(_env); + return Value().Set(index, boolValue); +} + +inline MaybeOrValue ObjectReference::Set(uint32_t index, + double numberValue) const { + HandleScope scope(_env); + return Value().Set(index, numberValue); +} + +//////////////////////////////////////////////////////////////////////////////// +// FunctionReference class +//////////////////////////////////////////////////////////////////////////////// + +inline FunctionReference::FunctionReference() : Reference() {} + +inline FunctionReference::FunctionReference(napi_env env, napi_ref ref) + : Reference(env, ref) {} + +inline FunctionReference::FunctionReference(Reference&& other) + : Reference(std::move(other)) {} + +inline FunctionReference& FunctionReference::operator=( + Reference&& other) { + static_cast*>(this)->operator=(std::move(other)); + return *this; +} + +inline FunctionReference::FunctionReference(FunctionReference&& other) + : Reference(std::move(other)) {} + +inline FunctionReference& FunctionReference::operator=( + FunctionReference&& other) { + static_cast*>(this)->operator=(std::move(other)); + return *this; +} + +inline MaybeOrValue FunctionReference::operator()( + const std::initializer_list& args) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value()(args); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap())); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Value(); + } + return scope.Escape(result); +#endif +} + +inline MaybeOrValue FunctionReference::Call( + const std::initializer_list& args) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value().Call(args); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap())); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Value(); + } + return scope.Escape(result); +#endif +} + +inline MaybeOrValue FunctionReference::Call( + const std::vector& args) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value().Call(args); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap())); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Value(); + } + return scope.Escape(result); +#endif +} + +inline MaybeOrValue FunctionReference::Call( + napi_value recv, const std::initializer_list& args) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value().Call(recv, args); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap())); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Value(); + } + return scope.Escape(result); +#endif +} + +inline MaybeOrValue FunctionReference::Call( + napi_value recv, const std::vector& args) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value().Call(recv, args); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap())); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Value(); + } + return scope.Escape(result); +#endif +} + +inline MaybeOrValue FunctionReference::Call( + napi_value recv, size_t argc, const napi_value* args) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value().Call(recv, argc, args); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap())); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Value(); + } + return scope.Escape(result); +#endif +} + +inline MaybeOrValue FunctionReference::MakeCallback( + napi_value recv, + const std::initializer_list& args, + napi_async_context context) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value().MakeCallback(recv, args, context); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap())); + } + + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Value(); + } + return scope.Escape(result); +#endif +} + +inline MaybeOrValue FunctionReference::MakeCallback( + napi_value recv, + const std::vector& args, + napi_async_context context) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value().MakeCallback(recv, args, context); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap())); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Value(); + } + return scope.Escape(result); +#endif +} + +inline MaybeOrValue FunctionReference::MakeCallback( + napi_value recv, + size_t argc, + const napi_value* args, + napi_async_context context) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = + Value().MakeCallback(recv, argc, args, context); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap())); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Value(); + } + return scope.Escape(result); +#endif +} + +inline MaybeOrValue FunctionReference::New( + const std::initializer_list& args) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value().New(args); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap()).As()); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Object(); + } + return scope.Escape(result).As(); +#endif +} + +inline MaybeOrValue FunctionReference::New( + const std::vector& args) const { + EscapableHandleScope scope(_env); + MaybeOrValue result = Value().New(args); +#ifdef NODE_ADDON_API_ENABLE_MAYBE + if (result.IsJust()) { + return Just(scope.Escape(result.Unwrap()).As()); + } + return result; +#else + if (scope.Env().IsExceptionPending()) { + return Object(); + } + return scope.Escape(result).As(); +#endif +} + +//////////////////////////////////////////////////////////////////////////////// +// CallbackInfo class +//////////////////////////////////////////////////////////////////////////////// + +inline CallbackInfo::CallbackInfo(napi_env env, napi_callback_info info) + : _env(env), + _info(info), + _this(nullptr), + _dynamicArgs(nullptr), + _data(nullptr) { + _argc = _staticArgCount; + _argv = _staticArgs; + napi_status status = + napi_get_cb_info(env, info, &_argc, _argv, &_this, &_data); + NAPI_THROW_IF_FAILED_VOID(_env, status); + + if (_argc > _staticArgCount) { + // Use either a fixed-size array (on the stack) or a dynamically-allocated + // array (on the heap) depending on the number of args. + _dynamicArgs = new napi_value[_argc]; + _argv = _dynamicArgs; + + status = napi_get_cb_info(env, info, &_argc, _argv, nullptr, nullptr); + NAPI_THROW_IF_FAILED_VOID(_env, status); + } +} + +inline CallbackInfo::~CallbackInfo() { + if (_dynamicArgs != nullptr) { + delete[] _dynamicArgs; + } +} + +inline CallbackInfo::operator napi_callback_info() const { + return _info; +} + +inline Value CallbackInfo::NewTarget() const { + napi_value newTarget; + napi_status status = napi_get_new_target(_env, _info, &newTarget); + NAPI_THROW_IF_FAILED(_env, status, Value()); + return Value(_env, newTarget); +} + +inline bool CallbackInfo::IsConstructCall() const { + return !NewTarget().IsEmpty(); +} + +inline Napi::Env CallbackInfo::Env() const { + return Napi::Env(_env); +} + +inline size_t CallbackInfo::Length() const { + return _argc; +} + +inline const Value CallbackInfo::operator[](size_t index) const { + return index < _argc ? Value(_env, _argv[index]) : Env().Undefined(); +} + +inline Value CallbackInfo::This() const { + if (_this == nullptr) { + return Env().Undefined(); + } + return Object(_env, _this); +} + +inline void* CallbackInfo::Data() const { + return _data; +} + +inline void CallbackInfo::SetData(void* data) { + _data = data; +} + +//////////////////////////////////////////////////////////////////////////////// +// PropertyDescriptor class +//////////////////////////////////////////////////////////////////////////////// + +template +PropertyDescriptor PropertyDescriptor::Accessor( + const char* utf8name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + + desc.utf8name = utf8name; + desc.getter = details::TemplatedCallback; + desc.attributes = attributes; + desc.data = data; + + return desc; +} + +template +PropertyDescriptor PropertyDescriptor::Accessor( + const std::string& utf8name, + napi_property_attributes attributes, + void* data) { + return Accessor(utf8name.c_str(), attributes, data); +} + +template +PropertyDescriptor PropertyDescriptor::Accessor( + Name name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + + desc.name = name; + desc.getter = details::TemplatedCallback; + desc.attributes = attributes; + desc.data = data; + + return desc; +} + +template +PropertyDescriptor PropertyDescriptor::Accessor( + const char* utf8name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + + desc.utf8name = utf8name; + desc.getter = details::TemplatedCallback; + desc.setter = details::TemplatedVoidCallback; + desc.attributes = attributes; + desc.data = data; + + return desc; +} + +template +PropertyDescriptor PropertyDescriptor::Accessor( + const std::string& utf8name, + napi_property_attributes attributes, + void* data) { + return Accessor(utf8name.c_str(), attributes, data); +} + +template +PropertyDescriptor PropertyDescriptor::Accessor( + Name name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + + desc.name = name; + desc.getter = details::TemplatedCallback; + desc.setter = details::TemplatedVoidCallback; + desc.attributes = attributes; + desc.data = data; + + return desc; +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + Napi::Env env, + Napi::Object object, + const char* utf8name, + Getter getter, + napi_property_attributes attributes, + void* data) { + using CbData = details::CallbackData; + auto callbackData = new CbData({getter, data}); + + napi_status status = AttachData(env, object, callbackData); + if (status != napi_ok) { + delete callbackData; + NAPI_THROW_IF_FAILED(env, status, napi_property_descriptor()); + } + + return PropertyDescriptor({utf8name, + nullptr, + nullptr, + CbData::Wrapper, + nullptr, + nullptr, + attributes, + callbackData}); +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + Napi::Env env, + Napi::Object object, + const std::string& utf8name, + Getter getter, + napi_property_attributes attributes, + void* data) { + return Accessor(env, object, utf8name.c_str(), getter, attributes, data); +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + Napi::Env env, + Napi::Object object, + Name name, + Getter getter, + napi_property_attributes attributes, + void* data) { + using CbData = details::CallbackData; + auto callbackData = new CbData({getter, data}); + + napi_status status = AttachData(env, object, callbackData); + if (status != napi_ok) { + delete callbackData; + NAPI_THROW_IF_FAILED(env, status, napi_property_descriptor()); + } + + return PropertyDescriptor({nullptr, + name, + nullptr, + CbData::Wrapper, + nullptr, + nullptr, + attributes, + callbackData}); +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + Napi::Env env, + Napi::Object object, + const char* utf8name, + Getter getter, + Setter setter, + napi_property_attributes attributes, + void* data) { + using CbData = details::AccessorCallbackData; + auto callbackData = new CbData({getter, setter, data}); + + napi_status status = AttachData(env, object, callbackData); + if (status != napi_ok) { + delete callbackData; + NAPI_THROW_IF_FAILED(env, status, napi_property_descriptor()); + } + + return PropertyDescriptor({utf8name, + nullptr, + nullptr, + CbData::GetterWrapper, + CbData::SetterWrapper, + nullptr, + attributes, + callbackData}); +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + Napi::Env env, + Napi::Object object, + const std::string& utf8name, + Getter getter, + Setter setter, + napi_property_attributes attributes, + void* data) { + return Accessor( + env, object, utf8name.c_str(), getter, setter, attributes, data); +} + +template +inline PropertyDescriptor PropertyDescriptor::Accessor( + Napi::Env env, + Napi::Object object, + Name name, + Getter getter, + Setter setter, + napi_property_attributes attributes, + void* data) { + using CbData = details::AccessorCallbackData; + auto callbackData = new CbData({getter, setter, data}); + + napi_status status = AttachData(env, object, callbackData); + if (status != napi_ok) { + delete callbackData; + NAPI_THROW_IF_FAILED(env, status, napi_property_descriptor()); + } + + return PropertyDescriptor({nullptr, + name, + nullptr, + CbData::GetterWrapper, + CbData::SetterWrapper, + nullptr, + attributes, + callbackData}); +} + +template +inline PropertyDescriptor PropertyDescriptor::Function( + Napi::Env env, + Napi::Object /*object*/, + const char* utf8name, + Callable cb, + napi_property_attributes attributes, + void* data) { + return PropertyDescriptor({utf8name, + nullptr, + nullptr, + nullptr, + nullptr, + Napi::Function::New(env, cb, utf8name, data), + attributes, + nullptr}); +} + +template +inline PropertyDescriptor PropertyDescriptor::Function( + Napi::Env env, + Napi::Object object, + const std::string& utf8name, + Callable cb, + napi_property_attributes attributes, + void* data) { + return Function(env, object, utf8name.c_str(), cb, attributes, data); +} + +template +inline PropertyDescriptor PropertyDescriptor::Function( + Napi::Env env, + Napi::Object /*object*/, + Name name, + Callable cb, + napi_property_attributes attributes, + void* data) { + return PropertyDescriptor({nullptr, + name, + nullptr, + nullptr, + nullptr, + Napi::Function::New(env, cb, nullptr, data), + attributes, + nullptr}); +} + +inline PropertyDescriptor PropertyDescriptor::Value( + const char* utf8name, + napi_value value, + napi_property_attributes attributes) { + return PropertyDescriptor({utf8name, + nullptr, + nullptr, + nullptr, + nullptr, + value, + attributes, + nullptr}); +} + +inline PropertyDescriptor PropertyDescriptor::Value( + const std::string& utf8name, + napi_value value, + napi_property_attributes attributes) { + return Value(utf8name.c_str(), value, attributes); +} + +inline PropertyDescriptor PropertyDescriptor::Value( + napi_value name, napi_value value, napi_property_attributes attributes) { + return PropertyDescriptor( + {nullptr, name, nullptr, nullptr, nullptr, value, attributes, nullptr}); +} + +inline PropertyDescriptor PropertyDescriptor::Value( + Name name, Napi::Value value, napi_property_attributes attributes) { + napi_value nameValue = name; + napi_value valueValue = value; + return PropertyDescriptor::Value(nameValue, valueValue, attributes); +} + +inline PropertyDescriptor::PropertyDescriptor(napi_property_descriptor desc) + : _desc(desc) {} + +inline PropertyDescriptor::operator napi_property_descriptor&() { + return _desc; +} + +inline PropertyDescriptor::operator const napi_property_descriptor&() const { + return _desc; +} + +//////////////////////////////////////////////////////////////////////////////// +// InstanceWrap class +//////////////////////////////////////////////////////////////////////////////// + +template +inline void InstanceWrap::AttachPropData( + napi_env env, napi_value value, const napi_property_descriptor* prop) { + napi_status status; + if (!(prop->attributes & napi_static)) { + if (prop->method == T::InstanceVoidMethodCallbackWrapper) { + status = Napi::details::AttachData( + env, value, static_cast(prop->data)); + NAPI_THROW_IF_FAILED_VOID(env, status); + } else if (prop->method == T::InstanceMethodCallbackWrapper) { + status = Napi::details::AttachData( + env, value, static_cast(prop->data)); + NAPI_THROW_IF_FAILED_VOID(env, status); + } else if (prop->getter == T::InstanceGetterCallbackWrapper || + prop->setter == T::InstanceSetterCallbackWrapper) { + status = Napi::details::AttachData( + env, value, static_cast(prop->data)); + NAPI_THROW_IF_FAILED_VOID(env, status); + } + } +} + +template +inline ClassPropertyDescriptor InstanceWrap::InstanceMethod( + const char* utf8name, + InstanceVoidMethodCallback method, + napi_property_attributes attributes, + void* data) { + InstanceVoidMethodCallbackData* callbackData = + new InstanceVoidMethodCallbackData({method, data}); + + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.method = T::InstanceVoidMethodCallbackWrapper; + desc.data = callbackData; + desc.attributes = attributes; + return desc; +} + +template +inline ClassPropertyDescriptor InstanceWrap::InstanceMethod( + const char* utf8name, + InstanceMethodCallback method, + napi_property_attributes attributes, + void* data) { + InstanceMethodCallbackData* callbackData = + new InstanceMethodCallbackData({method, data}); + + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.method = T::InstanceMethodCallbackWrapper; + desc.data = callbackData; + desc.attributes = attributes; + return desc; +} + +template +inline ClassPropertyDescriptor InstanceWrap::InstanceMethod( + Symbol name, + InstanceVoidMethodCallback method, + napi_property_attributes attributes, + void* data) { + InstanceVoidMethodCallbackData* callbackData = + new InstanceVoidMethodCallbackData({method, data}); + + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.method = T::InstanceVoidMethodCallbackWrapper; + desc.data = callbackData; + desc.attributes = attributes; + return desc; +} + +template +inline ClassPropertyDescriptor InstanceWrap::InstanceMethod( + Symbol name, + InstanceMethodCallback method, + napi_property_attributes attributes, + void* data) { + InstanceMethodCallbackData* callbackData = + new InstanceMethodCallbackData({method, data}); + + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.method = T::InstanceMethodCallbackWrapper; + desc.data = callbackData; + desc.attributes = attributes; + return desc; +} + +template +template ::InstanceVoidMethodCallback method> +inline ClassPropertyDescriptor InstanceWrap::InstanceMethod( + const char* utf8name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.method = details::TemplatedInstanceVoidCallback; + desc.data = data; + desc.attributes = attributes; + return desc; +} + +template +template ::InstanceMethodCallback method> +inline ClassPropertyDescriptor InstanceWrap::InstanceMethod( + const char* utf8name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.method = details::TemplatedInstanceCallback; + desc.data = data; + desc.attributes = attributes; + return desc; +} + +template +template ::InstanceVoidMethodCallback method> +inline ClassPropertyDescriptor InstanceWrap::InstanceMethod( + Symbol name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.method = details::TemplatedInstanceVoidCallback; + desc.data = data; + desc.attributes = attributes; + return desc; +} + +template +template ::InstanceMethodCallback method> +inline ClassPropertyDescriptor InstanceWrap::InstanceMethod( + Symbol name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.method = details::TemplatedInstanceCallback; + desc.data = data; + desc.attributes = attributes; + return desc; +} + +template +inline ClassPropertyDescriptor InstanceWrap::InstanceAccessor( + const char* utf8name, + InstanceGetterCallback getter, + InstanceSetterCallback setter, + napi_property_attributes attributes, + void* data) { + InstanceAccessorCallbackData* callbackData = + new InstanceAccessorCallbackData({getter, setter, data}); + + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.getter = getter != nullptr ? T::InstanceGetterCallbackWrapper : nullptr; + desc.setter = setter != nullptr ? T::InstanceSetterCallbackWrapper : nullptr; + desc.data = callbackData; + desc.attributes = attributes; + return desc; +} + +template +inline ClassPropertyDescriptor InstanceWrap::InstanceAccessor( + Symbol name, + InstanceGetterCallback getter, + InstanceSetterCallback setter, + napi_property_attributes attributes, + void* data) { + InstanceAccessorCallbackData* callbackData = + new InstanceAccessorCallbackData({getter, setter, data}); + + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.getter = getter != nullptr ? T::InstanceGetterCallbackWrapper : nullptr; + desc.setter = setter != nullptr ? T::InstanceSetterCallbackWrapper : nullptr; + desc.data = callbackData; + desc.attributes = attributes; + return desc; +} + +template +template ::InstanceGetterCallback getter, + typename InstanceWrap::InstanceSetterCallback setter> +inline ClassPropertyDescriptor InstanceWrap::InstanceAccessor( + const char* utf8name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.getter = details::TemplatedInstanceCallback; + desc.setter = This::WrapSetter(This::SetterTag()); + desc.data = data; + desc.attributes = attributes; + return desc; +} + +template +template ::InstanceGetterCallback getter, + typename InstanceWrap::InstanceSetterCallback setter> +inline ClassPropertyDescriptor InstanceWrap::InstanceAccessor( + Symbol name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.getter = details::TemplatedInstanceCallback; + desc.setter = This::WrapSetter(This::SetterTag()); + desc.data = data; + desc.attributes = attributes; + return desc; +} + +template +inline ClassPropertyDescriptor InstanceWrap::InstanceValue( + const char* utf8name, + Napi::Value value, + napi_property_attributes attributes) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.value = value; + desc.attributes = attributes; + return desc; +} + +template +inline ClassPropertyDescriptor InstanceWrap::InstanceValue( + Symbol name, Napi::Value value, napi_property_attributes attributes) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.value = value; + desc.attributes = attributes; + return desc; +} + +template +inline napi_value InstanceWrap::InstanceVoidMethodCallbackWrapper( + napi_env env, napi_callback_info info) { + return details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + InstanceVoidMethodCallbackData* callbackData = + reinterpret_cast(callbackInfo.Data()); + callbackInfo.SetData(callbackData->data); + T* instance = T::Unwrap(callbackInfo.This().As()); + auto cb = callbackData->callback; + (instance->*cb)(callbackInfo); + return nullptr; + }); +} + +template +inline napi_value InstanceWrap::InstanceMethodCallbackWrapper( + napi_env env, napi_callback_info info) { + return details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + InstanceMethodCallbackData* callbackData = + reinterpret_cast(callbackInfo.Data()); + callbackInfo.SetData(callbackData->data); + T* instance = T::Unwrap(callbackInfo.This().As()); + auto cb = callbackData->callback; + return (instance->*cb)(callbackInfo); + }); +} + +template +inline napi_value InstanceWrap::InstanceGetterCallbackWrapper( + napi_env env, napi_callback_info info) { + return details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + InstanceAccessorCallbackData* callbackData = + reinterpret_cast(callbackInfo.Data()); + callbackInfo.SetData(callbackData->data); + T* instance = T::Unwrap(callbackInfo.This().As()); + auto cb = callbackData->getterCallback; + return (instance->*cb)(callbackInfo); + }); +} + +template +inline napi_value InstanceWrap::InstanceSetterCallbackWrapper( + napi_env env, napi_callback_info info) { + return details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + InstanceAccessorCallbackData* callbackData = + reinterpret_cast(callbackInfo.Data()); + callbackInfo.SetData(callbackData->data); + T* instance = T::Unwrap(callbackInfo.This().As()); + auto cb = callbackData->setterCallback; + (instance->*cb)(callbackInfo, callbackInfo[0]); + return nullptr; + }); +} + +template +template ::InstanceSetterCallback method> +inline napi_value InstanceWrap::WrappedMethod( + napi_env env, napi_callback_info info) NAPI_NOEXCEPT { + return details::WrapCallback([&] { + const CallbackInfo cbInfo(env, info); + T* instance = T::Unwrap(cbInfo.This().As()); + (instance->*method)(cbInfo, cbInfo[0]); + return nullptr; + }); +} + +//////////////////////////////////////////////////////////////////////////////// +// ObjectWrap class +//////////////////////////////////////////////////////////////////////////////// + +template +inline ObjectWrap::ObjectWrap(const Napi::CallbackInfo& callbackInfo) { + napi_env env = callbackInfo.Env(); + napi_value wrapper = callbackInfo.This(); + napi_status status; + napi_ref ref; + T* instance = static_cast(this); + status = napi_wrap(env, wrapper, instance, FinalizeCallback, nullptr, &ref); + NAPI_THROW_IF_FAILED_VOID(env, status); + + Reference* instanceRef = instance; + *instanceRef = Reference(env, ref); +} + +template +inline ObjectWrap::~ObjectWrap() { + // If the JS object still exists at this point, remove the finalizer added + // through `napi_wrap()`. + if (!IsEmpty()) { + Object object = Value(); + // It is not valid to call `napi_remove_wrap()` with an empty `object`. + // This happens e.g. during garbage collection. + if (!object.IsEmpty() && _construction_failed) { + napi_remove_wrap(Env(), object, nullptr); + } + } +} + +template +inline T* ObjectWrap::Unwrap(Object wrapper) { + void* unwrapped; + napi_status status = napi_unwrap(wrapper.Env(), wrapper, &unwrapped); + NAPI_THROW_IF_FAILED(wrapper.Env(), status, nullptr); + return static_cast(unwrapped); +} + +template +inline Function ObjectWrap::DefineClass( + Napi::Env env, + const char* utf8name, + const size_t props_count, + const napi_property_descriptor* descriptors, + void* data) { + napi_status status; + std::vector props(props_count); + + // We copy the descriptors to a local array because before defining the class + // we must replace static method property descriptors with value property + // descriptors such that the value is a function-valued `napi_value` created + // with `CreateFunction()`. + // + // This replacement could be made for instance methods as well, but V8 aborts + // if we do that, because it expects methods defined on the prototype template + // to have `FunctionTemplate`s. + for (size_t index = 0; index < props_count; index++) { + props[index] = descriptors[index]; + napi_property_descriptor* prop = &props[index]; + if (prop->method == T::StaticMethodCallbackWrapper) { + status = + CreateFunction(env, + utf8name, + prop->method, + static_cast(prop->data), + &(prop->value)); + NAPI_THROW_IF_FAILED(env, status, Function()); + prop->method = nullptr; + prop->data = nullptr; + } else if (prop->method == T::StaticVoidMethodCallbackWrapper) { + status = + CreateFunction(env, + utf8name, + prop->method, + static_cast(prop->data), + &(prop->value)); + NAPI_THROW_IF_FAILED(env, status, Function()); + prop->method = nullptr; + prop->data = nullptr; + } + } + + napi_value value; + status = napi_define_class(env, + utf8name, + NAPI_AUTO_LENGTH, + T::ConstructorCallbackWrapper, + data, + props_count, + props.data(), + &value); + NAPI_THROW_IF_FAILED(env, status, Function()); + + // After defining the class we iterate once more over the property descriptors + // and attach the data associated with accessors and instance methods to the + // newly created JavaScript class. + for (size_t idx = 0; idx < props_count; idx++) { + const napi_property_descriptor* prop = &props[idx]; + + if (prop->getter == T::StaticGetterCallbackWrapper || + prop->setter == T::StaticSetterCallbackWrapper) { + status = Napi::details::AttachData( + env, value, static_cast(prop->data)); + NAPI_THROW_IF_FAILED(env, status, Function()); + } else { + // InstanceWrap::AttachPropData is responsible for attaching the data + // of instance methods and accessors. + T::AttachPropData(env, value, prop); + } + } + + return Function(env, value); +} + +template +inline Function ObjectWrap::DefineClass( + Napi::Env env, + const char* utf8name, + const std::initializer_list>& properties, + void* data) { + return DefineClass( + env, + utf8name, + properties.size(), + reinterpret_cast(properties.begin()), + data); +} + +template +inline Function ObjectWrap::DefineClass( + Napi::Env env, + const char* utf8name, + const std::vector>& properties, + void* data) { + return DefineClass( + env, + utf8name, + properties.size(), + reinterpret_cast(properties.data()), + data); +} + +template +inline ClassPropertyDescriptor ObjectWrap::StaticMethod( + const char* utf8name, + StaticVoidMethodCallback method, + napi_property_attributes attributes, + void* data) { + StaticVoidMethodCallbackData* callbackData = + new StaticVoidMethodCallbackData({method, data}); + + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.method = T::StaticVoidMethodCallbackWrapper; + desc.data = callbackData; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +inline ClassPropertyDescriptor ObjectWrap::StaticMethod( + const char* utf8name, + StaticMethodCallback method, + napi_property_attributes attributes, + void* data) { + StaticMethodCallbackData* callbackData = + new StaticMethodCallbackData({method, data}); + + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.method = T::StaticMethodCallbackWrapper; + desc.data = callbackData; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +inline ClassPropertyDescriptor ObjectWrap::StaticMethod( + Symbol name, + StaticVoidMethodCallback method, + napi_property_attributes attributes, + void* data) { + StaticVoidMethodCallbackData* callbackData = + new StaticVoidMethodCallbackData({method, data}); + + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.method = T::StaticVoidMethodCallbackWrapper; + desc.data = callbackData; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +inline ClassPropertyDescriptor ObjectWrap::StaticMethod( + Symbol name, + StaticMethodCallback method, + napi_property_attributes attributes, + void* data) { + StaticMethodCallbackData* callbackData = + new StaticMethodCallbackData({method, data}); + + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.method = T::StaticMethodCallbackWrapper; + desc.data = callbackData; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +template ::StaticVoidMethodCallback method> +inline ClassPropertyDescriptor ObjectWrap::StaticMethod( + const char* utf8name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.method = details::TemplatedVoidCallback; + desc.data = data; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +template ::StaticVoidMethodCallback method> +inline ClassPropertyDescriptor ObjectWrap::StaticMethod( + Symbol name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.method = details::TemplatedVoidCallback; + desc.data = data; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +template ::StaticMethodCallback method> +inline ClassPropertyDescriptor ObjectWrap::StaticMethod( + const char* utf8name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.method = details::TemplatedCallback; + desc.data = data; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +template ::StaticMethodCallback method> +inline ClassPropertyDescriptor ObjectWrap::StaticMethod( + Symbol name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.method = details::TemplatedCallback; + desc.data = data; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +inline ClassPropertyDescriptor ObjectWrap::StaticAccessor( + const char* utf8name, + StaticGetterCallback getter, + StaticSetterCallback setter, + napi_property_attributes attributes, + void* data) { + StaticAccessorCallbackData* callbackData = + new StaticAccessorCallbackData({getter, setter, data}); + + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.getter = getter != nullptr ? T::StaticGetterCallbackWrapper : nullptr; + desc.setter = setter != nullptr ? T::StaticSetterCallbackWrapper : nullptr; + desc.data = callbackData; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +inline ClassPropertyDescriptor ObjectWrap::StaticAccessor( + Symbol name, + StaticGetterCallback getter, + StaticSetterCallback setter, + napi_property_attributes attributes, + void* data) { + StaticAccessorCallbackData* callbackData = + new StaticAccessorCallbackData({getter, setter, data}); + + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.getter = getter != nullptr ? T::StaticGetterCallbackWrapper : nullptr; + desc.setter = setter != nullptr ? T::StaticSetterCallbackWrapper : nullptr; + desc.data = callbackData; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +template ::StaticGetterCallback getter, + typename ObjectWrap::StaticSetterCallback setter> +inline ClassPropertyDescriptor ObjectWrap::StaticAccessor( + const char* utf8name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.getter = details::TemplatedCallback; + desc.setter = This::WrapStaticSetter(This::StaticSetterTag()); + desc.data = data; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +template ::StaticGetterCallback getter, + typename ObjectWrap::StaticSetterCallback setter> +inline ClassPropertyDescriptor ObjectWrap::StaticAccessor( + Symbol name, napi_property_attributes attributes, void* data) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.getter = details::TemplatedCallback; + desc.setter = This::WrapStaticSetter(This::StaticSetterTag()); + desc.data = data; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +inline ClassPropertyDescriptor ObjectWrap::StaticValue( + const char* utf8name, + Napi::Value value, + napi_property_attributes attributes) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.utf8name = utf8name; + desc.value = value; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +inline ClassPropertyDescriptor ObjectWrap::StaticValue( + Symbol name, Napi::Value value, napi_property_attributes attributes) { + napi_property_descriptor desc = napi_property_descriptor(); + desc.name = name; + desc.value = value; + desc.attributes = + static_cast(attributes | napi_static); + return desc; +} + +template +inline Value ObjectWrap::OnCalledAsFunction( + const Napi::CallbackInfo& callbackInfo) { + NAPI_THROW( + TypeError::New(callbackInfo.Env(), + "Class constructors cannot be invoked without 'new'"), + Napi::Value()); +} + +template +inline void ObjectWrap::Finalize(Napi::Env /*env*/) {} + +template +inline napi_value ObjectWrap::ConstructorCallbackWrapper( + napi_env env, napi_callback_info info) { + napi_value new_target; + napi_status status = napi_get_new_target(env, info, &new_target); + if (status != napi_ok) return nullptr; + + bool isConstructCall = (new_target != nullptr); + if (!isConstructCall) { + return details::WrapCallback( + [&] { return T::OnCalledAsFunction(CallbackInfo(env, info)); }); + } + + napi_value wrapper = details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + T* instance = new T(callbackInfo); +#ifdef NAPI_CPP_EXCEPTIONS + instance->_construction_failed = false; +#else + if (callbackInfo.Env().IsExceptionPending()) { + // We need to clear the exception so that removing the wrap might work. + Error e = callbackInfo.Env().GetAndClearPendingException(); + delete instance; + e.ThrowAsJavaScriptException(); + } else { + instance->_construction_failed = false; + } +#endif // NAPI_CPP_EXCEPTIONS + return callbackInfo.This(); + }); + + return wrapper; +} + +template +inline napi_value ObjectWrap::StaticVoidMethodCallbackWrapper( + napi_env env, napi_callback_info info) { + return details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + StaticVoidMethodCallbackData* callbackData = + reinterpret_cast(callbackInfo.Data()); + callbackInfo.SetData(callbackData->data); + callbackData->callback(callbackInfo); + return nullptr; + }); +} + +template +inline napi_value ObjectWrap::StaticMethodCallbackWrapper( + napi_env env, napi_callback_info info) { + return details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + StaticMethodCallbackData* callbackData = + reinterpret_cast(callbackInfo.Data()); + callbackInfo.SetData(callbackData->data); + return callbackData->callback(callbackInfo); + }); +} + +template +inline napi_value ObjectWrap::StaticGetterCallbackWrapper( + napi_env env, napi_callback_info info) { + return details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + StaticAccessorCallbackData* callbackData = + reinterpret_cast(callbackInfo.Data()); + callbackInfo.SetData(callbackData->data); + return callbackData->getterCallback(callbackInfo); + }); +} + +template +inline napi_value ObjectWrap::StaticSetterCallbackWrapper( + napi_env env, napi_callback_info info) { + return details::WrapCallback([&] { + CallbackInfo callbackInfo(env, info); + StaticAccessorCallbackData* callbackData = + reinterpret_cast(callbackInfo.Data()); + callbackInfo.SetData(callbackData->data); + callbackData->setterCallback(callbackInfo, callbackInfo[0]); + return nullptr; + }); +} + +template +inline void ObjectWrap::FinalizeCallback(napi_env env, + void* data, + void* /*hint*/) { + HandleScope scope(env); + T* instance = static_cast(data); + instance->Finalize(Napi::Env(env)); + delete instance; +} + +template +template ::StaticSetterCallback method> +inline napi_value ObjectWrap::WrappedMethod( + napi_env env, napi_callback_info info) NAPI_NOEXCEPT { + return details::WrapCallback([&] { + const CallbackInfo cbInfo(env, info); + method(cbInfo, cbInfo[0]); + return nullptr; + }); +} + +//////////////////////////////////////////////////////////////////////////////// +// HandleScope class +//////////////////////////////////////////////////////////////////////////////// + +inline HandleScope::HandleScope(napi_env env, napi_handle_scope scope) + : _env(env), _scope(scope) {} + +inline HandleScope::HandleScope(Napi::Env env) : _env(env) { + napi_status status = napi_open_handle_scope(_env, &_scope); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +inline HandleScope::~HandleScope() { + napi_status status = napi_close_handle_scope(_env, _scope); + NAPI_FATAL_IF_FAILED( + status, "HandleScope::~HandleScope", "napi_close_handle_scope"); +} + +inline HandleScope::operator napi_handle_scope() const { + return _scope; +} + +inline Napi::Env HandleScope::Env() const { + return Napi::Env(_env); +} + +//////////////////////////////////////////////////////////////////////////////// +// EscapableHandleScope class +//////////////////////////////////////////////////////////////////////////////// + +inline EscapableHandleScope::EscapableHandleScope( + napi_env env, napi_escapable_handle_scope scope) + : _env(env), _scope(scope) {} + +inline EscapableHandleScope::EscapableHandleScope(Napi::Env env) : _env(env) { + napi_status status = napi_open_escapable_handle_scope(_env, &_scope); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +inline EscapableHandleScope::~EscapableHandleScope() { + napi_status status = napi_close_escapable_handle_scope(_env, _scope); + NAPI_FATAL_IF_FAILED(status, + "EscapableHandleScope::~EscapableHandleScope", + "napi_close_escapable_handle_scope"); +} + +inline EscapableHandleScope::operator napi_escapable_handle_scope() const { + return _scope; +} + +inline Napi::Env EscapableHandleScope::Env() const { + return Napi::Env(_env); +} + +inline Value EscapableHandleScope::Escape(napi_value escapee) { + napi_value result; + napi_status status = napi_escape_handle(_env, _scope, escapee, &result); + NAPI_THROW_IF_FAILED(_env, status, Value()); + return Value(_env, result); +} + +#if (NAPI_VERSION > 2) +//////////////////////////////////////////////////////////////////////////////// +// CallbackScope class +//////////////////////////////////////////////////////////////////////////////// + +inline CallbackScope::CallbackScope(napi_env env, napi_callback_scope scope) + : _env(env), _scope(scope) {} + +inline CallbackScope::CallbackScope(napi_env env, napi_async_context context) + : _env(env) { + napi_status status = + napi_open_callback_scope(_env, Object::New(env), context, &_scope); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +inline CallbackScope::~CallbackScope() { + napi_status status = napi_close_callback_scope(_env, _scope); + NAPI_FATAL_IF_FAILED( + status, "CallbackScope::~CallbackScope", "napi_close_callback_scope"); +} + +inline CallbackScope::operator napi_callback_scope() const { + return _scope; +} + +inline Napi::Env CallbackScope::Env() const { + return Napi::Env(_env); +} +#endif + +//////////////////////////////////////////////////////////////////////////////// +// AsyncContext class +//////////////////////////////////////////////////////////////////////////////// + +inline AsyncContext::AsyncContext(napi_env env, const char* resource_name) + : AsyncContext(env, resource_name, Object::New(env)) {} + +inline AsyncContext::AsyncContext(napi_env env, + const char* resource_name, + const Object& resource) + : _env(env), _context(nullptr) { + napi_value resource_id; + napi_status status = napi_create_string_utf8( + _env, resource_name, NAPI_AUTO_LENGTH, &resource_id); + NAPI_THROW_IF_FAILED_VOID(_env, status); + + status = napi_async_init(_env, resource, resource_id, &_context); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +inline AsyncContext::~AsyncContext() { + if (_context != nullptr) { + napi_async_destroy(_env, _context); + _context = nullptr; + } +} + +inline AsyncContext::AsyncContext(AsyncContext&& other) { + _env = other._env; + other._env = nullptr; + _context = other._context; + other._context = nullptr; +} + +inline AsyncContext& AsyncContext::operator=(AsyncContext&& other) { + _env = other._env; + other._env = nullptr; + _context = other._context; + other._context = nullptr; + return *this; +} + +inline AsyncContext::operator napi_async_context() const { + return _context; +} + +inline Napi::Env AsyncContext::Env() const { + return Napi::Env(_env); +} + +//////////////////////////////////////////////////////////////////////////////// +// AsyncWorker class +//////////////////////////////////////////////////////////////////////////////// + +inline AsyncWorker::AsyncWorker(const Function& callback) + : AsyncWorker(callback, "generic") {} + +inline AsyncWorker::AsyncWorker(const Function& callback, + const char* resource_name) + : AsyncWorker(callback, resource_name, Object::New(callback.Env())) {} + +inline AsyncWorker::AsyncWorker(const Function& callback, + const char* resource_name, + const Object& resource) + : AsyncWorker( + Object::New(callback.Env()), callback, resource_name, resource) {} + +inline AsyncWorker::AsyncWorker(const Object& receiver, + const Function& callback) + : AsyncWorker(receiver, callback, "generic") {} + +inline AsyncWorker::AsyncWorker(const Object& receiver, + const Function& callback, + const char* resource_name) + : AsyncWorker( + receiver, callback, resource_name, Object::New(callback.Env())) {} + +inline AsyncWorker::AsyncWorker(const Object& receiver, + const Function& callback, + const char* resource_name, + const Object& resource) + : _env(callback.Env()), + _receiver(Napi::Persistent(receiver)), + _callback(Napi::Persistent(callback)), + _suppress_destruct(false) { + napi_value resource_id; + napi_status status = napi_create_string_latin1( + _env, resource_name, NAPI_AUTO_LENGTH, &resource_id); + NAPI_THROW_IF_FAILED_VOID(_env, status); + + status = napi_create_async_work(_env, + resource, + resource_id, + OnAsyncWorkExecute, + OnAsyncWorkComplete, + this, + &_work); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +inline AsyncWorker::AsyncWorker(Napi::Env env) : AsyncWorker(env, "generic") {} + +inline AsyncWorker::AsyncWorker(Napi::Env env, const char* resource_name) + : AsyncWorker(env, resource_name, Object::New(env)) {} + +inline AsyncWorker::AsyncWorker(Napi::Env env, + const char* resource_name, + const Object& resource) + : _env(env), _receiver(), _callback(), _suppress_destruct(false) { + napi_value resource_id; + napi_status status = napi_create_string_latin1( + _env, resource_name, NAPI_AUTO_LENGTH, &resource_id); + NAPI_THROW_IF_FAILED_VOID(_env, status); + + status = napi_create_async_work(_env, + resource, + resource_id, + OnAsyncWorkExecute, + OnAsyncWorkComplete, + this, + &_work); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +inline AsyncWorker::~AsyncWorker() { + if (_work != nullptr) { + napi_delete_async_work(_env, _work); + _work = nullptr; + } +} + +inline void AsyncWorker::Destroy() { + delete this; +} + +inline AsyncWorker::AsyncWorker(AsyncWorker&& other) { + _env = other._env; + other._env = nullptr; + _work = other._work; + other._work = nullptr; + _receiver = std::move(other._receiver); + _callback = std::move(other._callback); + _error = std::move(other._error); + _suppress_destruct = other._suppress_destruct; +} + +inline AsyncWorker& AsyncWorker::operator=(AsyncWorker&& other) { + _env = other._env; + other._env = nullptr; + _work = other._work; + other._work = nullptr; + _receiver = std::move(other._receiver); + _callback = std::move(other._callback); + _error = std::move(other._error); + _suppress_destruct = other._suppress_destruct; + return *this; +} + +inline AsyncWorker::operator napi_async_work() const { + return _work; +} + +inline Napi::Env AsyncWorker::Env() const { + return Napi::Env(_env); +} + +inline void AsyncWorker::Queue() { + napi_status status = napi_queue_async_work(_env, _work); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +inline void AsyncWorker::Cancel() { + napi_status status = napi_cancel_async_work(_env, _work); + NAPI_THROW_IF_FAILED_VOID(_env, status); +} + +inline ObjectReference& AsyncWorker::Receiver() { + return _receiver; +} + +inline FunctionReference& AsyncWorker::Callback() { + return _callback; +} + +inline void AsyncWorker::SuppressDestruct() { + _suppress_destruct = true; +} + +inline void AsyncWorker::OnOK() { + if (!_callback.IsEmpty()) { + _callback.Call(_receiver.Value(), GetResult(_callback.Env())); + } +} + +inline void AsyncWorker::OnError(const Error& e) { + if (!_callback.IsEmpty()) { + _callback.Call(_receiver.Value(), + std::initializer_list{e.Value()}); + } +} + +inline void AsyncWorker::SetError(const std::string& error) { + _error = error; +} + +inline std::vector AsyncWorker::GetResult(Napi::Env /*env*/) { + return {}; +} +// The OnAsyncWorkExecute method receives an napi_env argument. However, do NOT +// use it within this method, as it does not run on the JavaScript thread and +// must not run any method that would cause JavaScript to run. In practice, +// this means that almost any use of napi_env will be incorrect. +inline void AsyncWorker::OnAsyncWorkExecute(napi_env env, void* asyncworker) { + AsyncWorker* self = static_cast(asyncworker); + self->OnExecute(env); +} +// The OnExecute method receives an napi_env argument. However, do NOT +// use it within this method, as it does not run on the JavaScript thread and +// must not run any method that would cause JavaScript to run. In practice, +// this means that almost any use of napi_env will be incorrect. +inline void AsyncWorker::OnExecute(Napi::Env /*DO_NOT_USE*/) { +#ifdef NAPI_CPP_EXCEPTIONS + try { + Execute(); + } catch (const std::exception& e) { + SetError(e.what()); + } +#else // NAPI_CPP_EXCEPTIONS + Execute(); +#endif // NAPI_CPP_EXCEPTIONS +} + +inline void AsyncWorker::OnAsyncWorkComplete(napi_env env, + napi_status status, + void* asyncworker) { + AsyncWorker* self = static_cast(asyncworker); + self->OnWorkComplete(env, status); +} +inline void AsyncWorker::OnWorkComplete(Napi::Env /*env*/, napi_status status) { + if (status != napi_cancelled) { + HandleScope scope(_env); + details::WrapCallback([&] { + if (_error.size() == 0) { + OnOK(); + } else { + OnError(Error::New(_env, _error)); + } + return nullptr; + }); + } + if (!_suppress_destruct) { + Destroy(); + } +} + +#if (NAPI_VERSION > 3 && !defined(__wasm32__)) +//////////////////////////////////////////////////////////////////////////////// +// TypedThreadSafeFunction class +//////////////////////////////////////////////////////////////////////////////// + +// Starting with NAPI 5, the JavaScript function `func` parameter of +// `napi_create_threadsafe_function` is optional. +#if NAPI_VERSION > 4 +// static, with Callback [missing] Resource [missing] Finalizer [missing] +template +template +inline TypedThreadSafeFunction +TypedThreadSafeFunction::New( + napi_env env, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context) { + TypedThreadSafeFunction tsfn; + + napi_status status = + napi_create_threadsafe_function(env, + nullptr, + nullptr, + String::From(env, resourceName), + maxQueueSize, + initialThreadCount, + nullptr, + nullptr, + context, + CallJsInternal, + &tsfn._tsfn); + if (status != napi_ok) { + NAPI_THROW_IF_FAILED( + env, status, TypedThreadSafeFunction()); + } + + return tsfn; +} + +// static, with Callback [missing] Resource [passed] Finalizer [missing] +template +template +inline TypedThreadSafeFunction +TypedThreadSafeFunction::New( + napi_env env, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context) { + TypedThreadSafeFunction tsfn; + + napi_status status = + napi_create_threadsafe_function(env, + nullptr, + resource, + String::From(env, resourceName), + maxQueueSize, + initialThreadCount, + nullptr, + nullptr, + context, + CallJsInternal, + &tsfn._tsfn); + if (status != napi_ok) { + NAPI_THROW_IF_FAILED( + env, status, TypedThreadSafeFunction()); + } + + return tsfn; +} + +// static, with Callback [missing] Resource [missing] Finalizer [passed] +template +template +inline TypedThreadSafeFunction +TypedThreadSafeFunction::New( + napi_env env, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data) { + TypedThreadSafeFunction tsfn; + + auto* finalizeData = new details:: + ThreadSafeFinalize( + {data, finalizeCallback}); + napi_status status = napi_create_threadsafe_function( + env, + nullptr, + nullptr, + String::From(env, resourceName), + maxQueueSize, + initialThreadCount, + finalizeData, + details::ThreadSafeFinalize:: + FinalizeFinalizeWrapperWithDataAndContext, + context, + CallJsInternal, + &tsfn._tsfn); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED( + env, status, TypedThreadSafeFunction()); + } + + return tsfn; +} + +// static, with Callback [missing] Resource [passed] Finalizer [passed] +template +template +inline TypedThreadSafeFunction +TypedThreadSafeFunction::New( + napi_env env, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data) { + TypedThreadSafeFunction tsfn; + + auto* finalizeData = new details:: + ThreadSafeFinalize( + {data, finalizeCallback}); + napi_status status = napi_create_threadsafe_function( + env, + nullptr, + resource, + String::From(env, resourceName), + maxQueueSize, + initialThreadCount, + finalizeData, + details::ThreadSafeFinalize:: + FinalizeFinalizeWrapperWithDataAndContext, + context, + CallJsInternal, + &tsfn._tsfn); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED( + env, status, TypedThreadSafeFunction()); + } + + return tsfn; +} +#endif + +// static, with Callback [passed] Resource [missing] Finalizer [missing] +template +template +inline TypedThreadSafeFunction +TypedThreadSafeFunction::New( + napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context) { + TypedThreadSafeFunction tsfn; + + napi_status status = + napi_create_threadsafe_function(env, + callback, + nullptr, + String::From(env, resourceName), + maxQueueSize, + initialThreadCount, + nullptr, + nullptr, + context, + CallJsInternal, + &tsfn._tsfn); + if (status != napi_ok) { + NAPI_THROW_IF_FAILED( + env, status, TypedThreadSafeFunction()); + } + + return tsfn; +} + +// static, with Callback [passed] Resource [passed] Finalizer [missing] +template +template +inline TypedThreadSafeFunction +TypedThreadSafeFunction::New( + napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context) { + TypedThreadSafeFunction tsfn; + + napi_status status = + napi_create_threadsafe_function(env, + callback, + resource, + String::From(env, resourceName), + maxQueueSize, + initialThreadCount, + nullptr, + nullptr, + context, + CallJsInternal, + &tsfn._tsfn); + if (status != napi_ok) { + NAPI_THROW_IF_FAILED( + env, status, TypedThreadSafeFunction()); + } + + return tsfn; +} + +// static, with Callback [passed] Resource [missing] Finalizer [passed] +template +template +inline TypedThreadSafeFunction +TypedThreadSafeFunction::New( + napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data) { + TypedThreadSafeFunction tsfn; + + auto* finalizeData = new details:: + ThreadSafeFinalize( + {data, finalizeCallback}); + napi_status status = napi_create_threadsafe_function( + env, + callback, + nullptr, + String::From(env, resourceName), + maxQueueSize, + initialThreadCount, + finalizeData, + details::ThreadSafeFinalize:: + FinalizeFinalizeWrapperWithDataAndContext, + context, + CallJsInternal, + &tsfn._tsfn); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED( + env, status, TypedThreadSafeFunction()); + } + + return tsfn; +} + +// static, with: Callback [passed] Resource [passed] Finalizer [passed] +template +template +inline TypedThreadSafeFunction +TypedThreadSafeFunction::New( + napi_env env, + CallbackType callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data) { + TypedThreadSafeFunction tsfn; + + auto* finalizeData = new details:: + ThreadSafeFinalize( + {data, finalizeCallback}); + napi_status status = napi_create_threadsafe_function( + env, + details::DefaultCallbackWrapper< + CallbackType, + TypedThreadSafeFunction>(env, + callback), + resource, + String::From(env, resourceName), + maxQueueSize, + initialThreadCount, + finalizeData, + details::ThreadSafeFinalize:: + FinalizeFinalizeWrapperWithDataAndContext, + context, + CallJsInternal, + &tsfn._tsfn); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED( + env, status, TypedThreadSafeFunction()); + } + + return tsfn; +} + +template +inline TypedThreadSafeFunction:: + TypedThreadSafeFunction() + : _tsfn() {} + +template +inline TypedThreadSafeFunction:: + TypedThreadSafeFunction(napi_threadsafe_function tsfn) + : _tsfn(tsfn) {} + +template +inline TypedThreadSafeFunction:: +operator napi_threadsafe_function() const { + return _tsfn; +} + +template +inline napi_status +TypedThreadSafeFunction::BlockingCall( + DataType* data) const { + return napi_call_threadsafe_function(_tsfn, data, napi_tsfn_blocking); +} + +template +inline napi_status +TypedThreadSafeFunction::NonBlockingCall( + DataType* data) const { + return napi_call_threadsafe_function(_tsfn, data, napi_tsfn_nonblocking); +} + +template +inline void TypedThreadSafeFunction::Ref( + napi_env env) const { + if (_tsfn != nullptr) { + napi_status status = napi_ref_threadsafe_function(env, _tsfn); + NAPI_THROW_IF_FAILED_VOID(env, status); + } +} + +template +inline void TypedThreadSafeFunction::Unref( + napi_env env) const { + if (_tsfn != nullptr) { + napi_status status = napi_unref_threadsafe_function(env, _tsfn); + NAPI_THROW_IF_FAILED_VOID(env, status); + } +} + +template +inline napi_status +TypedThreadSafeFunction::Acquire() const { + return napi_acquire_threadsafe_function(_tsfn); +} + +template +inline napi_status +TypedThreadSafeFunction::Release() const { + return napi_release_threadsafe_function(_tsfn, napi_tsfn_release); +} + +template +inline napi_status +TypedThreadSafeFunction::Abort() const { + return napi_release_threadsafe_function(_tsfn, napi_tsfn_abort); +} + +template +inline ContextType* +TypedThreadSafeFunction::GetContext() const { + void* context; + napi_status status = napi_get_threadsafe_function_context(_tsfn, &context); + NAPI_FATAL_IF_FAILED(status, + "TypedThreadSafeFunction::GetContext", + "napi_get_threadsafe_function_context"); + return static_cast(context); +} + +// static +template +void TypedThreadSafeFunction::CallJsInternal( + napi_env env, napi_value jsCallback, void* context, void* data) { + details::CallJsWrapper( + env, jsCallback, context, data); +} + +#if NAPI_VERSION == 4 +// static +template +Napi::Function +TypedThreadSafeFunction::EmptyFunctionFactory( + Napi::Env env) { + return Napi::Function::New(env, [](const CallbackInfo& cb) {}); +} + +// static +template +Napi::Function +TypedThreadSafeFunction::FunctionOrEmpty( + Napi::Env env, Napi::Function& callback) { + if (callback.IsEmpty()) { + return EmptyFunctionFactory(env); + } + return callback; +} + +#else +// static +template +std::nullptr_t +TypedThreadSafeFunction::EmptyFunctionFactory( + Napi::Env /*env*/) { + return nullptr; +} + +// static +template +Napi::Function +TypedThreadSafeFunction::FunctionOrEmpty( + Napi::Env /*env*/, Napi::Function& callback) { + return callback; +} + +#endif + +//////////////////////////////////////////////////////////////////////////////// +// ThreadSafeFunction class +//////////////////////////////////////////////////////////////////////////////// + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount) { + return New( + env, callback, Object(), resourceName, maxQueueSize, initialThreadCount); +} + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context) { + return New(env, + callback, + Object(), + resourceName, + maxQueueSize, + initialThreadCount, + context); +} + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + Finalizer finalizeCallback) { + return New(env, + callback, + Object(), + resourceName, + maxQueueSize, + initialThreadCount, + finalizeCallback); +} + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + Finalizer finalizeCallback, + FinalizerDataType* data) { + return New(env, + callback, + Object(), + resourceName, + maxQueueSize, + initialThreadCount, + finalizeCallback, + data); +} + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback) { + return New(env, + callback, + Object(), + resourceName, + maxQueueSize, + initialThreadCount, + context, + finalizeCallback); +} + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data) { + return New(env, + callback, + Object(), + resourceName, + maxQueueSize, + initialThreadCount, + context, + finalizeCallback, + data); +} + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount) { + return New(env, + callback, + resource, + resourceName, + maxQueueSize, + initialThreadCount, + static_cast(nullptr) /* context */); +} + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context) { + return New(env, + callback, + resource, + resourceName, + maxQueueSize, + initialThreadCount, + context, + [](Env, ContextType*) {} /* empty finalizer */); +} + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + Finalizer finalizeCallback) { + return New(env, + callback, + resource, + resourceName, + maxQueueSize, + initialThreadCount, + static_cast(nullptr) /* context */, + finalizeCallback, + static_cast(nullptr) /* data */, + details::ThreadSafeFinalize::Wrapper); +} + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + Finalizer finalizeCallback, + FinalizerDataType* data) { + return New(env, + callback, + resource, + resourceName, + maxQueueSize, + initialThreadCount, + static_cast(nullptr) /* context */, + finalizeCallback, + data, + details::ThreadSafeFinalize:: + FinalizeWrapperWithData); +} + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback) { + return New( + env, + callback, + resource, + resourceName, + maxQueueSize, + initialThreadCount, + context, + finalizeCallback, + static_cast(nullptr) /* data */, + details::ThreadSafeFinalize::FinalizeWrapperWithContext); +} + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data) { + return New( + env, + callback, + resource, + resourceName, + maxQueueSize, + initialThreadCount, + context, + finalizeCallback, + data, + details::ThreadSafeFinalize:: + FinalizeFinalizeWrapperWithDataAndContext); +} + +inline ThreadSafeFunction::ThreadSafeFunction() : _tsfn() {} + +inline ThreadSafeFunction::ThreadSafeFunction(napi_threadsafe_function tsfn) + : _tsfn(tsfn) {} + +inline ThreadSafeFunction::operator napi_threadsafe_function() const { + return _tsfn; +} + +inline napi_status ThreadSafeFunction::BlockingCall() const { + return CallInternal(nullptr, napi_tsfn_blocking); +} + +template <> +inline napi_status ThreadSafeFunction::BlockingCall(void* data) const { + return napi_call_threadsafe_function(_tsfn, data, napi_tsfn_blocking); +} + +template +inline napi_status ThreadSafeFunction::BlockingCall(Callback callback) const { + return CallInternal(new CallbackWrapper(callback), napi_tsfn_blocking); +} + +template +inline napi_status ThreadSafeFunction::BlockingCall(DataType* data, + Callback callback) const { + auto wrapper = [data, callback](Env env, Function jsCallback) { + callback(env, jsCallback, data); + }; + return CallInternal(new CallbackWrapper(wrapper), napi_tsfn_blocking); +} + +inline napi_status ThreadSafeFunction::NonBlockingCall() const { + return CallInternal(nullptr, napi_tsfn_nonblocking); +} + +template <> +inline napi_status ThreadSafeFunction::NonBlockingCall(void* data) const { + return napi_call_threadsafe_function(_tsfn, data, napi_tsfn_nonblocking); +} + +template +inline napi_status ThreadSafeFunction::NonBlockingCall( + Callback callback) const { + return CallInternal(new CallbackWrapper(callback), napi_tsfn_nonblocking); +} + +template +inline napi_status ThreadSafeFunction::NonBlockingCall( + DataType* data, Callback callback) const { + auto wrapper = [data, callback](Env env, Function jsCallback) { + callback(env, jsCallback, data); + }; + return CallInternal(new CallbackWrapper(wrapper), napi_tsfn_nonblocking); +} + +inline void ThreadSafeFunction::Ref(napi_env env) const { + if (_tsfn != nullptr) { + napi_status status = napi_ref_threadsafe_function(env, _tsfn); + NAPI_THROW_IF_FAILED_VOID(env, status); + } +} + +inline void ThreadSafeFunction::Unref(napi_env env) const { + if (_tsfn != nullptr) { + napi_status status = napi_unref_threadsafe_function(env, _tsfn); + NAPI_THROW_IF_FAILED_VOID(env, status); + } +} + +inline napi_status ThreadSafeFunction::Acquire() const { + return napi_acquire_threadsafe_function(_tsfn); +} + +inline napi_status ThreadSafeFunction::Release() const { + return napi_release_threadsafe_function(_tsfn, napi_tsfn_release); +} + +inline napi_status ThreadSafeFunction::Abort() const { + return napi_release_threadsafe_function(_tsfn, napi_tsfn_abort); +} + +inline ThreadSafeFunction::ConvertibleContext ThreadSafeFunction::GetContext() + const { + void* context; + napi_status status = napi_get_threadsafe_function_context(_tsfn, &context); + NAPI_FATAL_IF_FAILED(status, + "ThreadSafeFunction::GetContext", + "napi_get_threadsafe_function_context"); + return ConvertibleContext({context}); +} + +// static +template +inline ThreadSafeFunction ThreadSafeFunction::New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data, + napi_finalize wrapper) { + static_assert(details::can_make_string::value || + std::is_convertible::value, + "Resource name should be convertible to the string type"); + + ThreadSafeFunction tsfn; + auto* finalizeData = new details:: + ThreadSafeFinalize( + {data, finalizeCallback}); + napi_status status = + napi_create_threadsafe_function(env, + callback, + resource, + Value::From(env, resourceName), + maxQueueSize, + initialThreadCount, + finalizeData, + wrapper, + context, + CallJS, + &tsfn._tsfn); + if (status != napi_ok) { + delete finalizeData; + NAPI_THROW_IF_FAILED(env, status, ThreadSafeFunction()); + } + + return tsfn; +} + +inline napi_status ThreadSafeFunction::CallInternal( + CallbackWrapper* callbackWrapper, + napi_threadsafe_function_call_mode mode) const { + napi_status status = + napi_call_threadsafe_function(_tsfn, callbackWrapper, mode); + if (status != napi_ok && callbackWrapper != nullptr) { + delete callbackWrapper; + } + + return status; +} + +// static +inline void ThreadSafeFunction::CallJS(napi_env env, + napi_value jsCallback, + void* /* context */, + void* data) { + if (env == nullptr && jsCallback == nullptr) { + return; + } + + if (data != nullptr) { + auto* callbackWrapper = static_cast(data); + (*callbackWrapper)(env, Function(env, jsCallback)); + delete callbackWrapper; + } else if (jsCallback != nullptr) { + Function(env, jsCallback).Call({}); + } +} + +//////////////////////////////////////////////////////////////////////////////// +// Async Progress Worker Base class +//////////////////////////////////////////////////////////////////////////////// +template +inline AsyncProgressWorkerBase::AsyncProgressWorkerBase( + const Object& receiver, + const Function& callback, + const char* resource_name, + const Object& resource, + size_t queue_size) + : AsyncWorker(receiver, callback, resource_name, resource) { + // Fill all possible arguments to work around ambiguous + // ThreadSafeFunction::New signatures. + _tsfn = ThreadSafeFunction::New(callback.Env(), + callback, + resource, + resource_name, + queue_size, + /** initialThreadCount */ 1, + /** context */ this, + OnThreadSafeFunctionFinalize, + /** finalizeData */ this); +} + +#if NAPI_VERSION > 4 +template +inline AsyncProgressWorkerBase::AsyncProgressWorkerBase( + Napi::Env env, + const char* resource_name, + const Object& resource, + size_t queue_size) + : AsyncWorker(env, resource_name, resource) { + // TODO: Once the changes to make the callback optional for threadsafe + // functions are available on all versions we can remove the dummy Function + // here. + Function callback; + // Fill all possible arguments to work around ambiguous + // ThreadSafeFunction::New signatures. + _tsfn = ThreadSafeFunction::New(env, + callback, + resource, + resource_name, + queue_size, + /** initialThreadCount */ 1, + /** context */ this, + OnThreadSafeFunctionFinalize, + /** finalizeData */ this); +} +#endif + +template +inline AsyncProgressWorkerBase::~AsyncProgressWorkerBase() { + // Abort pending tsfn call. + // Don't send progress events after we've already completed. + // It's ok to call ThreadSafeFunction::Abort and ThreadSafeFunction::Release + // duplicated. + _tsfn.Abort(); +} + +template +inline void AsyncProgressWorkerBase::OnAsyncWorkProgress( + Napi::Env /* env */, Napi::Function /* jsCallback */, void* data) { + ThreadSafeData* tsd = static_cast(data); + tsd->asyncprogressworker()->OnWorkProgress(tsd->data()); + delete tsd; +} + +template +inline napi_status AsyncProgressWorkerBase::NonBlockingCall( + DataType* data) { + auto tsd = new AsyncProgressWorkerBase::ThreadSafeData(this, data); + auto ret = _tsfn.NonBlockingCall(tsd, OnAsyncWorkProgress); + if (ret != napi_ok) { + delete tsd; + } + return ret; +} + +template +inline void AsyncProgressWorkerBase::OnWorkComplete( + Napi::Env /* env */, napi_status status) { + _work_completed = true; + _complete_status = status; + _tsfn.Release(); +} + +template +inline void AsyncProgressWorkerBase::OnThreadSafeFunctionFinalize( + Napi::Env env, void* /* data */, AsyncProgressWorkerBase* context) { + if (context->_work_completed) { + context->AsyncWorker::OnWorkComplete(env, context->_complete_status); + } +} + +//////////////////////////////////////////////////////////////////////////////// +// Async Progress Worker class +//////////////////////////////////////////////////////////////////////////////// +template +inline AsyncProgressWorker::AsyncProgressWorker(const Function& callback) + : AsyncProgressWorker(callback, "generic") {} + +template +inline AsyncProgressWorker::AsyncProgressWorker(const Function& callback, + const char* resource_name) + : AsyncProgressWorker( + callback, resource_name, Object::New(callback.Env())) {} + +template +inline AsyncProgressWorker::AsyncProgressWorker(const Function& callback, + const char* resource_name, + const Object& resource) + : AsyncProgressWorker( + Object::New(callback.Env()), callback, resource_name, resource) {} + +template +inline AsyncProgressWorker::AsyncProgressWorker(const Object& receiver, + const Function& callback) + : AsyncProgressWorker(receiver, callback, "generic") {} + +template +inline AsyncProgressWorker::AsyncProgressWorker(const Object& receiver, + const Function& callback, + const char* resource_name) + : AsyncProgressWorker( + receiver, callback, resource_name, Object::New(callback.Env())) {} + +template +inline AsyncProgressWorker::AsyncProgressWorker(const Object& receiver, + const Function& callback, + const char* resource_name, + const Object& resource) + : AsyncProgressWorkerBase(receiver, callback, resource_name, resource), + _asyncdata(nullptr), + _asyncsize(0), + _signaled(false) {} + +#if NAPI_VERSION > 4 +template +inline AsyncProgressWorker::AsyncProgressWorker(Napi::Env env) + : AsyncProgressWorker(env, "generic") {} + +template +inline AsyncProgressWorker::AsyncProgressWorker(Napi::Env env, + const char* resource_name) + : AsyncProgressWorker(env, resource_name, Object::New(env)) {} + +template +inline AsyncProgressWorker::AsyncProgressWorker(Napi::Env env, + const char* resource_name, + const Object& resource) + : AsyncProgressWorkerBase(env, resource_name, resource), + _asyncdata(nullptr), + _asyncsize(0) {} +#endif + +template +inline AsyncProgressWorker::~AsyncProgressWorker() { + { + std::lock_guard lock(this->_mutex); + _asyncdata = nullptr; + _asyncsize = 0; + } +} + +template +inline void AsyncProgressWorker::Execute() { + ExecutionProgress progress(this); + Execute(progress); +} + +template +inline void AsyncProgressWorker::OnWorkProgress(void*) { + T* data; + size_t size; + bool signaled; + { + std::lock_guard lock(this->_mutex); + data = this->_asyncdata; + size = this->_asyncsize; + signaled = this->_signaled; + this->_asyncdata = nullptr; + this->_asyncsize = 0; + this->_signaled = false; + } + + /** + * The callback of ThreadSafeFunction is not been invoked immediately on the + * callback of uv_async_t (uv io poll), rather the callback of TSFN is + * invoked on the right next uv idle callback. There are chances that during + * the deferring the signal of uv_async_t is been sent again, i.e. potential + * not coalesced two calls of the TSFN callback. + */ + if (data == nullptr && !signaled) { + return; + } + + this->OnProgress(data, size); + delete[] data; +} + +template +inline void AsyncProgressWorker::SendProgress_(const T* data, size_t count) { + T* new_data = new T[count]; + std::copy(data, data + count, new_data); + + T* old_data; + { + std::lock_guard lock(this->_mutex); + old_data = _asyncdata; + _asyncdata = new_data; + _asyncsize = count; + _signaled = false; + } + this->NonBlockingCall(nullptr); + + delete[] old_data; +} + +template +inline void AsyncProgressWorker::Signal() { + { + std::lock_guard lock(this->_mutex); + _signaled = true; + } + this->NonBlockingCall(static_cast(nullptr)); +} + +template +inline void AsyncProgressWorker::ExecutionProgress::Signal() const { + this->_worker->Signal(); +} + +template +inline void AsyncProgressWorker::ExecutionProgress::Send( + const T* data, size_t count) const { + _worker->SendProgress_(data, count); +} + +//////////////////////////////////////////////////////////////////////////////// +// Async Progress Queue Worker class +//////////////////////////////////////////////////////////////////////////////// +template +inline AsyncProgressQueueWorker::AsyncProgressQueueWorker( + const Function& callback) + : AsyncProgressQueueWorker(callback, "generic") {} + +template +inline AsyncProgressQueueWorker::AsyncProgressQueueWorker( + const Function& callback, const char* resource_name) + : AsyncProgressQueueWorker( + callback, resource_name, Object::New(callback.Env())) {} + +template +inline AsyncProgressQueueWorker::AsyncProgressQueueWorker( + const Function& callback, const char* resource_name, const Object& resource) + : AsyncProgressQueueWorker( + Object::New(callback.Env()), callback, resource_name, resource) {} + +template +inline AsyncProgressQueueWorker::AsyncProgressQueueWorker( + const Object& receiver, const Function& callback) + : AsyncProgressQueueWorker(receiver, callback, "generic") {} + +template +inline AsyncProgressQueueWorker::AsyncProgressQueueWorker( + const Object& receiver, const Function& callback, const char* resource_name) + : AsyncProgressQueueWorker( + receiver, callback, resource_name, Object::New(callback.Env())) {} + +template +inline AsyncProgressQueueWorker::AsyncProgressQueueWorker( + const Object& receiver, + const Function& callback, + const char* resource_name, + const Object& resource) + : AsyncProgressWorkerBase>( + receiver, + callback, + resource_name, + resource, + /** unlimited queue size */ 0) {} + +#if NAPI_VERSION > 4 +template +inline AsyncProgressQueueWorker::AsyncProgressQueueWorker(Napi::Env env) + : AsyncProgressQueueWorker(env, "generic") {} + +template +inline AsyncProgressQueueWorker::AsyncProgressQueueWorker( + Napi::Env env, const char* resource_name) + : AsyncProgressQueueWorker(env, resource_name, Object::New(env)) {} + +template +inline AsyncProgressQueueWorker::AsyncProgressQueueWorker( + Napi::Env env, const char* resource_name, const Object& resource) + : AsyncProgressWorkerBase>( + env, resource_name, resource, /** unlimited queue size */ 0) {} +#endif + +template +inline void AsyncProgressQueueWorker::Execute() { + ExecutionProgress progress(this); + Execute(progress); +} + +template +inline void AsyncProgressQueueWorker::OnWorkProgress( + std::pair* datapair) { + if (datapair == nullptr) { + return; + } + + T* data = datapair->first; + size_t size = datapair->second; + + this->OnProgress(data, size); + delete datapair; + delete[] data; +} + +template +inline void AsyncProgressQueueWorker::SendProgress_(const T* data, + size_t count) { + T* new_data = new T[count]; + std::copy(data, data + count, new_data); + + auto pair = new std::pair(new_data, count); + this->NonBlockingCall(pair); +} + +template +inline void AsyncProgressQueueWorker::Signal() const { + this->SendProgress_(static_cast(nullptr), 0); +} + +template +inline void AsyncProgressQueueWorker::OnWorkComplete(Napi::Env env, + napi_status status) { + // Draining queued items in TSFN. + AsyncProgressWorkerBase>::OnWorkComplete(env, status); +} + +template +inline void AsyncProgressQueueWorker::ExecutionProgress::Signal() const { + _worker->SendProgress_(static_cast(nullptr), 0); +} + +template +inline void AsyncProgressQueueWorker::ExecutionProgress::Send( + const T* data, size_t count) const { + _worker->SendProgress_(data, count); +} +#endif // NAPI_VERSION > 3 && !defined(__wasm32__) + +//////////////////////////////////////////////////////////////////////////////// +// Memory Management class +//////////////////////////////////////////////////////////////////////////////// + +inline int64_t MemoryManagement::AdjustExternalMemory(Env env, + int64_t change_in_bytes) { + int64_t result; + napi_status status = + napi_adjust_external_memory(env, change_in_bytes, &result); + NAPI_THROW_IF_FAILED(env, status, 0); + return result; +} + +//////////////////////////////////////////////////////////////////////////////// +// Version Management class +//////////////////////////////////////////////////////////////////////////////// + +inline uint32_t VersionManagement::GetNapiVersion(Env env) { + uint32_t result; + napi_status status = napi_get_version(env, &result); + NAPI_THROW_IF_FAILED(env, status, 0); + return result; +} + +inline const napi_node_version* VersionManagement::GetNodeVersion(Env env) { + const napi_node_version* result; + napi_status status = napi_get_node_version(env, &result); + NAPI_THROW_IF_FAILED(env, status, 0); + return result; +} + +#if NAPI_VERSION > 5 +//////////////////////////////////////////////////////////////////////////////// +// Addon class +//////////////////////////////////////////////////////////////////////////////// + +template +inline Object Addon::Init(Env env, Object exports) { + T* addon = new T(env, exports); + env.SetInstanceData(addon); + return addon->entry_point_; +} + +template +inline T* Addon::Unwrap(Object wrapper) { + return wrapper.Env().GetInstanceData(); +} + +template +inline void Addon::DefineAddon( + Object exports, const std::initializer_list& props) { + DefineProperties(exports, props); + entry_point_ = exports; +} + +template +inline Napi::Object Addon::DefineProperties( + Object object, const std::initializer_list& props) { + const napi_property_descriptor* properties = + reinterpret_cast(props.begin()); + size_t size = props.size(); + napi_status status = + napi_define_properties(object.Env(), object, size, properties); + NAPI_THROW_IF_FAILED(object.Env(), status, object); + for (size_t idx = 0; idx < size; idx++) + T::AttachPropData(object.Env(), object, &properties[idx]); + return object; +} +#endif // NAPI_VERSION > 5 + +#if NAPI_VERSION > 2 +template +Env::CleanupHook Env::AddCleanupHook(Hook hook, Arg* arg) { + return CleanupHook(*this, hook, arg); +} + +template +Env::CleanupHook Env::AddCleanupHook(Hook hook) { + return CleanupHook(*this, hook); +} + +template +Env::CleanupHook::CleanupHook() { + data = nullptr; +} + +template +Env::CleanupHook::CleanupHook(Napi::Env env, Hook hook) + : wrapper(Env::CleanupHook::Wrapper) { + data = new CleanupData{std::move(hook), nullptr}; + napi_status status = napi_add_env_cleanup_hook(env, wrapper, data); + if (status != napi_ok) { + delete data; + data = nullptr; + } +} + +template +Env::CleanupHook::CleanupHook(Napi::Env env, Hook hook, Arg* arg) + : wrapper(Env::CleanupHook::WrapperWithArg) { + data = new CleanupData{std::move(hook), arg}; + napi_status status = napi_add_env_cleanup_hook(env, wrapper, data); + if (status != napi_ok) { + delete data; + data = nullptr; + } +} + +template +bool Env::CleanupHook::Remove(Env env) { + napi_status status = napi_remove_env_cleanup_hook(env, wrapper, data); + delete data; + data = nullptr; + return status == napi_ok; +} + +template +bool Env::CleanupHook::IsEmpty() const { + return data == nullptr; +} +#endif // NAPI_VERSION > 2 + +#ifdef NAPI_CPP_CUSTOM_NAMESPACE +} // namespace NAPI_CPP_CUSTOM_NAMESPACE +#endif + +} // namespace Napi + +#endif // SRC_NAPI_INL_H_ diff --git a/node_modules/node-addon-api/napi.h b/node_modules/node-addon-api/napi.h new file mode 100644 index 0000000..831b3b6 --- /dev/null +++ b/node_modules/node-addon-api/napi.h @@ -0,0 +1,3114 @@ +#ifndef SRC_NAPI_H_ +#define SRC_NAPI_H_ + +#include +#include +#include +#include +#include +#include +#include + +// VS2015 RTM has bugs with constexpr, so require min of VS2015 Update 3 (known +// good version) +#if !defined(_MSC_VER) || _MSC_FULL_VER >= 190024210 +#define NAPI_HAS_CONSTEXPR 1 +#endif + +// VS2013 does not support char16_t literal strings, so we'll work around it +// using wchar_t strings and casting them. This is safe as long as the character +// sizes are the same. +#if defined(_MSC_VER) && _MSC_VER <= 1800 +static_assert(sizeof(char16_t) == sizeof(wchar_t), + "Size mismatch between char16_t and wchar_t"); +#define NAPI_WIDE_TEXT(x) reinterpret_cast(L##x) +#else +#define NAPI_WIDE_TEXT(x) u##x +#endif + +// If C++ exceptions are not explicitly enabled or disabled, enable them +// if exceptions were enabled in the compiler settings. +#if !defined(NAPI_CPP_EXCEPTIONS) && !defined(NAPI_DISABLE_CPP_EXCEPTIONS) +#if defined(_CPPUNWIND) || defined(__EXCEPTIONS) +#define NAPI_CPP_EXCEPTIONS +#else +#error Exception support not detected. \ + Define either NAPI_CPP_EXCEPTIONS or NAPI_DISABLE_CPP_EXCEPTIONS. +#endif +#endif + +// If C++ NAPI_CPP_EXCEPTIONS are enabled, NODE_ADDON_API_ENABLE_MAYBE should +// not be set +#if defined(NAPI_CPP_EXCEPTIONS) && defined(NODE_ADDON_API_ENABLE_MAYBE) +#error NODE_ADDON_API_ENABLE_MAYBE should not be set when \ + NAPI_CPP_EXCEPTIONS is defined. +#endif + +#ifdef _NOEXCEPT +#define NAPI_NOEXCEPT _NOEXCEPT +#else +#define NAPI_NOEXCEPT noexcept +#endif + +#ifdef NAPI_CPP_EXCEPTIONS + +// When C++ exceptions are enabled, Errors are thrown directly. There is no need +// to return anything after the throw statements. The variadic parameter is an +// optional return value that is ignored. +// We need _VOID versions of the macros to avoid warnings resulting from +// leaving the NAPI_THROW_* `...` argument empty. + +#define NAPI_THROW(e, ...) throw e +#define NAPI_THROW_VOID(e) throw e + +#define NAPI_THROW_IF_FAILED(env, status, ...) \ + if ((status) != napi_ok) throw Napi::Error::New(env); + +#define NAPI_THROW_IF_FAILED_VOID(env, status) \ + if ((status) != napi_ok) throw Napi::Error::New(env); + +#else // NAPI_CPP_EXCEPTIONS + +// When C++ exceptions are disabled, Errors are thrown as JavaScript exceptions, +// which are pending until the callback returns to JS. The variadic parameter +// is an optional return value; usually it is an empty result. +// We need _VOID versions of the macros to avoid warnings resulting from +// leaving the NAPI_THROW_* `...` argument empty. + +#define NAPI_THROW(e, ...) \ + do { \ + (e).ThrowAsJavaScriptException(); \ + return __VA_ARGS__; \ + } while (0) + +#define NAPI_THROW_VOID(e) \ + do { \ + (e).ThrowAsJavaScriptException(); \ + return; \ + } while (0) + +#define NAPI_THROW_IF_FAILED(env, status, ...) \ + if ((status) != napi_ok) { \ + Napi::Error::New(env).ThrowAsJavaScriptException(); \ + return __VA_ARGS__; \ + } + +#define NAPI_THROW_IF_FAILED_VOID(env, status) \ + if ((status) != napi_ok) { \ + Napi::Error::New(env).ThrowAsJavaScriptException(); \ + return; \ + } + +#endif // NAPI_CPP_EXCEPTIONS + +#ifdef NODE_ADDON_API_ENABLE_MAYBE +#define NAPI_MAYBE_THROW_IF_FAILED(env, status, type) \ + NAPI_THROW_IF_FAILED(env, status, Napi::Nothing()) + +#define NAPI_RETURN_OR_THROW_IF_FAILED(env, status, result, type) \ + NAPI_MAYBE_THROW_IF_FAILED(env, status, type); \ + return Napi::Just(result); +#else +#define NAPI_MAYBE_THROW_IF_FAILED(env, status, type) \ + NAPI_THROW_IF_FAILED(env, status, type()) + +#define NAPI_RETURN_OR_THROW_IF_FAILED(env, status, result, type) \ + NAPI_MAYBE_THROW_IF_FAILED(env, status, type); \ + return result; +#endif + +#define NAPI_DISALLOW_ASSIGN(CLASS) void operator=(const CLASS&) = delete; +#define NAPI_DISALLOW_COPY(CLASS) CLASS(const CLASS&) = delete; + +#define NAPI_DISALLOW_ASSIGN_COPY(CLASS) \ + NAPI_DISALLOW_ASSIGN(CLASS) \ + NAPI_DISALLOW_COPY(CLASS) + +#define NAPI_CHECK(condition, location, message) \ + do { \ + if (!(condition)) { \ + Napi::Error::Fatal((location), (message)); \ + } \ + } while (0) + +#define NAPI_FATAL_IF_FAILED(status, location, message) \ + NAPI_CHECK((status) == napi_ok, location, message) + +//////////////////////////////////////////////////////////////////////////////// +/// Node-API C++ Wrapper Classes +/// +/// These classes wrap the "Node-API" ABI-stable C APIs for Node.js, providing a +/// C++ object model and C++ exception-handling semantics with low overhead. +/// The wrappers are all header-only so that they do not affect the ABI. +//////////////////////////////////////////////////////////////////////////////// +namespace Napi { + +#ifdef NAPI_CPP_CUSTOM_NAMESPACE +// NAPI_CPP_CUSTOM_NAMESPACE can be #define'd per-addon to avoid symbol +// conflicts between different instances of node-addon-api + +// First dummy definition of the namespace to make sure that Napi::(name) still +// refers to the right things inside this file. +namespace NAPI_CPP_CUSTOM_NAMESPACE {} +using namespace NAPI_CPP_CUSTOM_NAMESPACE; + +namespace NAPI_CPP_CUSTOM_NAMESPACE { +#endif + +// Forward declarations +class Env; +class Value; +class Boolean; +class Number; +#if NAPI_VERSION > 5 +class BigInt; +#endif // NAPI_VERSION > 5 +#if (NAPI_VERSION > 4) +class Date; +#endif +class String; +class Object; +class Array; +class ArrayBuffer; +class Function; +class Error; +class PropertyDescriptor; +class CallbackInfo; +class TypedArray; +template +class TypedArrayOf; + +using Int8Array = + TypedArrayOf; ///< Typed-array of signed 8-bit integers +using Uint8Array = + TypedArrayOf; ///< Typed-array of unsigned 8-bit integers +using Int16Array = + TypedArrayOf; ///< Typed-array of signed 16-bit integers +using Uint16Array = + TypedArrayOf; ///< Typed-array of unsigned 16-bit integers +using Int32Array = + TypedArrayOf; ///< Typed-array of signed 32-bit integers +using Uint32Array = + TypedArrayOf; ///< Typed-array of unsigned 32-bit integers +using Float32Array = + TypedArrayOf; ///< Typed-array of 32-bit floating-point values +using Float64Array = + TypedArrayOf; ///< Typed-array of 64-bit floating-point values +#if NAPI_VERSION > 5 +using BigInt64Array = + TypedArrayOf; ///< Typed array of signed 64-bit integers +using BigUint64Array = + TypedArrayOf; ///< Typed array of unsigned 64-bit integers +#endif // NAPI_VERSION > 5 + +/// Defines the signature of a Node-API C++ module's registration callback +/// (init) function. +using ModuleRegisterCallback = Object (*)(Env env, Object exports); + +class MemoryManagement; + +/// A simple Maybe type, representing an object which may or may not have a +/// value. +/// +/// If an API method returns a Maybe<>, the API method can potentially fail +/// either because an exception is thrown, or because an exception is pending, +/// e.g. because a previous API call threw an exception that hasn't been +/// caught yet. In that case, a "Nothing" value is returned. +template +class Maybe { + public: + bool IsNothing() const; + bool IsJust() const; + + /// Short-hand for Unwrap(), which doesn't return a value. Could be used + /// where the actual value of the Maybe is not needed like Object::Set. + /// If this Maybe is nothing (empty), node-addon-api will crash the + /// process. + void Check() const; + + /// Return the value of type T contained in the Maybe. If this Maybe is + /// nothing (empty), node-addon-api will crash the process. + T Unwrap() const; + + /// Return the value of type T contained in the Maybe, or using a default + /// value if this Maybe is nothing (empty). + T UnwrapOr(const T& default_value) const; + + /// Converts this Maybe to a value of type T in the out. If this Maybe is + /// nothing (empty), `false` is returned and `out` is left untouched. + bool UnwrapTo(T* out) const; + + bool operator==(const Maybe& other) const; + bool operator!=(const Maybe& other) const; + + private: + Maybe(); + explicit Maybe(const T& t); + + bool _has_value; + T _value; + + template + friend Maybe Nothing(); + template + friend Maybe Just(const U& u); +}; + +template +inline Maybe Nothing(); + +template +inline Maybe Just(const T& t); + +#if defined(NODE_ADDON_API_ENABLE_MAYBE) +template +using MaybeOrValue = Maybe; +#else +template +using MaybeOrValue = T; +#endif + +/// Environment for Node-API values and operations. +/// +/// All Node-API values and operations must be associated with an environment. +/// An environment instance is always provided to callback functions; that +/// environment must then be used for any creation of Node-API values or other +/// Node-API operations within the callback. (Many methods infer the +/// environment from the `this` instance that the method is called on.) +/// +/// In the future, multiple environments per process may be supported, +/// although current implementations only support one environment per process. +/// +/// In the V8 JavaScript engine, a Node-API environment approximately +/// corresponds to an Isolate. +class Env { + private: + napi_env _env; +#if NAPI_VERSION > 5 + template + static void DefaultFini(Env, T* data); + template + static void DefaultFiniWithHint(Env, DataType* data, HintType* hint); +#endif // NAPI_VERSION > 5 + public: + Env(napi_env env); + + operator napi_env() const; + + Object Global() const; + Value Undefined() const; + Value Null() const; + + bool IsExceptionPending() const; + Error GetAndClearPendingException() const; + + MaybeOrValue RunScript(const char* utf8script) const; + MaybeOrValue RunScript(const std::string& utf8script) const; + MaybeOrValue RunScript(String script) const; + +#if NAPI_VERSION > 2 + template + class CleanupHook; + + template + CleanupHook AddCleanupHook(Hook hook); + + template + CleanupHook AddCleanupHook(Hook hook, Arg* arg); +#endif // NAPI_VERSION > 2 + +#if NAPI_VERSION > 5 + template + T* GetInstanceData() const; + + template + using Finalizer = void (*)(Env, T*); + template fini = Env::DefaultFini> + void SetInstanceData(T* data) const; + + template + using FinalizerWithHint = void (*)(Env, DataType*, HintType*); + template fini = + Env::DefaultFiniWithHint> + void SetInstanceData(DataType* data, HintType* hint) const; +#endif // NAPI_VERSION > 5 + +#if NAPI_VERSION > 2 + template + class CleanupHook { + public: + CleanupHook(); + CleanupHook(Env env, Hook hook, Arg* arg); + CleanupHook(Env env, Hook hook); + bool Remove(Env env); + bool IsEmpty() const; + + private: + static inline void Wrapper(void* data) NAPI_NOEXCEPT; + static inline void WrapperWithArg(void* data) NAPI_NOEXCEPT; + + void (*wrapper)(void* arg); + struct CleanupData { + Hook hook; + Arg* arg; + } * data; + }; +#endif // NAPI_VERSION > 2 +}; + +/// A JavaScript value of unknown type. +/// +/// For type-specific operations, convert to one of the Value subclasses using a +/// `To*` or `As()` method. The `To*` methods do type coercion; the `As()` +/// method does not. +/// +/// Napi::Value value = ... +/// if (!value.IsString()) throw Napi::TypeError::New(env, "Invalid +/// arg..."); Napi::String str = value.As(); // Cast to a +/// string value +/// +/// Napi::Value anotherValue = ... +/// bool isTruthy = anotherValue.ToBoolean(); // Coerce to a boolean value +class Value { + public: + Value(); ///< Creates a new _empty_ Value instance. + Value(napi_env env, + napi_value value); ///< Wraps a Node-API value primitive. + + /// Creates a JS value from a C++ primitive. + /// + /// `value` may be any of: + /// - bool + /// - Any integer type + /// - Any floating point type + /// - const char* (encoded using UTF-8, null-terminated) + /// - const char16_t* (encoded using UTF-16-LE, null-terminated) + /// - std::string (encoded using UTF-8) + /// - std::u16string + /// - napi::Value + /// - napi_value + template + static Value From(napi_env env, const T& value); + + /// Converts to a Node-API value primitive. + /// + /// If the instance is _empty_, this returns `nullptr`. + operator napi_value() const; + + /// Tests if this value strictly equals another value. + bool operator==(const Value& other) const; + + /// Tests if this value does not strictly equal another value. + bool operator!=(const Value& other) const; + + /// Tests if this value strictly equals another value. + bool StrictEquals(const Value& other) const; + + /// Gets the environment the value is associated with. + Napi::Env Env() const; + + /// Checks if the value is empty (uninitialized). + /// + /// An empty value is invalid, and most attempts to perform an operation on an + /// empty value will result in an exception. Note an empty value is distinct + /// from JavaScript `null` or `undefined`, which are valid values. + /// + /// When C++ exceptions are disabled at compile time, a method with a `Value` + /// return type may return an empty value to indicate a pending exception. So + /// when not using C++ exceptions, callers should check whether the value is + /// empty before attempting to use it. + bool IsEmpty() const; + + napi_valuetype Type() const; ///< Gets the type of the value. + + bool IsUndefined() + const; ///< Tests if a value is an undefined JavaScript value. + bool IsNull() const; ///< Tests if a value is a null JavaScript value. + bool IsBoolean() const; ///< Tests if a value is a JavaScript boolean. + bool IsNumber() const; ///< Tests if a value is a JavaScript number. +#if NAPI_VERSION > 5 + bool IsBigInt() const; ///< Tests if a value is a JavaScript bigint. +#endif // NAPI_VERSION > 5 +#if (NAPI_VERSION > 4) + bool IsDate() const; ///< Tests if a value is a JavaScript date. +#endif + bool IsString() const; ///< Tests if a value is a JavaScript string. + bool IsSymbol() const; ///< Tests if a value is a JavaScript symbol. + bool IsArray() const; ///< Tests if a value is a JavaScript array. + bool IsArrayBuffer() + const; ///< Tests if a value is a JavaScript array buffer. + bool IsTypedArray() const; ///< Tests if a value is a JavaScript typed array. + bool IsObject() const; ///< Tests if a value is a JavaScript object. + bool IsFunction() const; ///< Tests if a value is a JavaScript function. + bool IsPromise() const; ///< Tests if a value is a JavaScript promise. + bool IsDataView() const; ///< Tests if a value is a JavaScript data view. + bool IsBuffer() const; ///< Tests if a value is a Node buffer. + bool IsExternal() const; ///< Tests if a value is a pointer to external data. + + /// Casts to another type of `Napi::Value`, when the actual type is known or + /// assumed. + /// + /// This conversion does NOT coerce the type. Calling any methods + /// inappropriate for the actual value type will throw `Napi::Error`. + template + T As() const; + + MaybeOrValue ToBoolean() + const; ///< Coerces a value to a JavaScript boolean. + MaybeOrValue ToNumber() + const; ///< Coerces a value to a JavaScript number. + MaybeOrValue ToString() + const; ///< Coerces a value to a JavaScript string. + MaybeOrValue ToObject() + const; ///< Coerces a value to a JavaScript object. + + protected: + /// !cond INTERNAL + napi_env _env; + napi_value _value; + /// !endcond +}; + +/// A JavaScript boolean value. +class Boolean : public Value { + public: + static Boolean New(napi_env env, ///< Node-API environment + bool value ///< Boolean value + ); + + Boolean(); ///< Creates a new _empty_ Boolean instance. + Boolean(napi_env env, + napi_value value); ///< Wraps a Node-API value primitive. + + operator bool() const; ///< Converts a Boolean value to a boolean primitive. + bool Value() const; ///< Converts a Boolean value to a boolean primitive. +}; + +/// A JavaScript number value. +class Number : public Value { + public: + static Number New(napi_env env, ///< Node-API environment + double value ///< Number value + ); + + Number(); ///< Creates a new _empty_ Number instance. + Number(napi_env env, + napi_value value); ///< Wraps a Node-API value primitive. + + operator int32_t() + const; ///< Converts a Number value to a 32-bit signed integer value. + operator uint32_t() + const; ///< Converts a Number value to a 32-bit unsigned integer value. + operator int64_t() + const; ///< Converts a Number value to a 64-bit signed integer value. + operator float() + const; ///< Converts a Number value to a 32-bit floating-point value. + operator double() + const; ///< Converts a Number value to a 64-bit floating-point value. + + int32_t Int32Value() + const; ///< Converts a Number value to a 32-bit signed integer value. + uint32_t Uint32Value() + const; ///< Converts a Number value to a 32-bit unsigned integer value. + int64_t Int64Value() + const; ///< Converts a Number value to a 64-bit signed integer value. + float FloatValue() + const; ///< Converts a Number value to a 32-bit floating-point value. + double DoubleValue() + const; ///< Converts a Number value to a 64-bit floating-point value. +}; + +#if NAPI_VERSION > 5 +/// A JavaScript bigint value. +class BigInt : public Value { + public: + static BigInt New(napi_env env, ///< Node-API environment + int64_t value ///< Number value + ); + static BigInt New(napi_env env, ///< Node-API environment + uint64_t value ///< Number value + ); + + /// Creates a new BigInt object using a specified sign bit and a + /// specified list of digits/words. + /// The resulting number is calculated as: + /// (-1)^sign_bit * (words[0] * (2^64)^0 + words[1] * (2^64)^1 + ...) + static BigInt New(napi_env env, ///< Node-API environment + int sign_bit, ///< Sign bit. 1 if negative. + size_t word_count, ///< Number of words in array + const uint64_t* words ///< Array of words + ); + + BigInt(); ///< Creates a new _empty_ BigInt instance. + BigInt(napi_env env, + napi_value value); ///< Wraps a Node-API value primitive. + + int64_t Int64Value(bool* lossless) + const; ///< Converts a BigInt value to a 64-bit signed integer value. + uint64_t Uint64Value(bool* lossless) + const; ///< Converts a BigInt value to a 64-bit unsigned integer value. + + size_t WordCount() const; ///< The number of 64-bit words needed to store + ///< the result of ToWords(). + + /// Writes the contents of this BigInt to a specified memory location. + /// `sign_bit` must be provided and will be set to 1 if this BigInt is + /// negative. + /// `*word_count` has to be initialized to the length of the `words` array. + /// Upon return, it will be set to the actual number of words that would + /// be needed to store this BigInt (i.e. the return value of `WordCount()`). + void ToWords(int* sign_bit, size_t* word_count, uint64_t* words); +}; +#endif // NAPI_VERSION > 5 + +#if (NAPI_VERSION > 4) +/// A JavaScript date value. +class Date : public Value { + public: + /// Creates a new Date value from a double primitive. + static Date New(napi_env env, ///< Node-API environment + double value ///< Number value + ); + + Date(); ///< Creates a new _empty_ Date instance. + Date(napi_env env, napi_value value); ///< Wraps a Node-API value primitive. + operator double() const; ///< Converts a Date value to double primitive + + double ValueOf() const; ///< Converts a Date value to a double primitive. +}; +#endif + +/// A JavaScript string or symbol value (that can be used as a property name). +class Name : public Value { + public: + Name(); ///< Creates a new _empty_ Name instance. + Name(napi_env env, + napi_value value); ///< Wraps a Node-API value primitive. +}; + +/// A JavaScript string value. +class String : public Name { + public: + /// Creates a new String value from a UTF-8 encoded C++ string. + static String New(napi_env env, ///< Node-API environment + const std::string& value ///< UTF-8 encoded C++ string + ); + + /// Creates a new String value from a UTF-16 encoded C++ string. + static String New(napi_env env, ///< Node-API environment + const std::u16string& value ///< UTF-16 encoded C++ string + ); + + /// Creates a new String value from a UTF-8 encoded C string. + static String New( + napi_env env, ///< Node-API environment + const char* value ///< UTF-8 encoded null-terminated C string + ); + + /// Creates a new String value from a UTF-16 encoded C string. + static String New( + napi_env env, ///< Node-API environment + const char16_t* value ///< UTF-16 encoded null-terminated C string + ); + + /// Creates a new String value from a UTF-8 encoded C string with specified + /// length. + static String New(napi_env env, ///< Node-API environment + const char* value, ///< UTF-8 encoded C string (not + ///< necessarily null-terminated) + size_t length ///< length of the string in bytes + ); + + /// Creates a new String value from a UTF-16 encoded C string with specified + /// length. + static String New( + napi_env env, ///< Node-API environment + const char16_t* value, ///< UTF-16 encoded C string (not necessarily + ///< null-terminated) + size_t length ///< Length of the string in 2-byte code units + ); + + /// Creates a new String based on the original object's type. + /// + /// `value` may be any of: + /// - const char* (encoded using UTF-8, null-terminated) + /// - const char16_t* (encoded using UTF-16-LE, null-terminated) + /// - std::string (encoded using UTF-8) + /// - std::u16string + template + static String From(napi_env env, const T& value); + + String(); ///< Creates a new _empty_ String instance. + String(napi_env env, + napi_value value); ///< Wraps a Node-API value primitive. + + operator std::string() + const; ///< Converts a String value to a UTF-8 encoded C++ string. + operator std::u16string() + const; ///< Converts a String value to a UTF-16 encoded C++ string. + std::string Utf8Value() + const; ///< Converts a String value to a UTF-8 encoded C++ string. + std::u16string Utf16Value() + const; ///< Converts a String value to a UTF-16 encoded C++ string. +}; + +/// A JavaScript symbol value. +class Symbol : public Name { + public: + /// Creates a new Symbol value with an optional description. + static Symbol New( + napi_env env, ///< Node-API environment + const char* description = + nullptr ///< Optional UTF-8 encoded null-terminated C string + /// describing the symbol + ); + + /// Creates a new Symbol value with a description. + static Symbol New( + napi_env env, ///< Node-API environment + const std::string& + description ///< UTF-8 encoded C++ string describing the symbol + ); + + /// Creates a new Symbol value with a description. + static Symbol New(napi_env env, ///< Node-API environment + String description ///< String value describing the symbol + ); + + /// Creates a new Symbol value with a description. + static Symbol New( + napi_env env, ///< Node-API environment + napi_value description ///< String value describing the symbol + ); + + /// Get a public Symbol (e.g. Symbol.iterator). + static MaybeOrValue WellKnown(napi_env, const std::string& name); + + // Create a symbol in the global registry, UTF-8 Encoded cpp string + static MaybeOrValue For(napi_env env, const std::string& description); + + // Create a symbol in the global registry, C style string (null terminated) + static MaybeOrValue For(napi_env env, const char* description); + + // Create a symbol in the global registry, String value describing the symbol + static MaybeOrValue For(napi_env env, String description); + + // Create a symbol in the global registry, napi_value describing the symbol + static MaybeOrValue For(napi_env env, napi_value description); + + Symbol(); ///< Creates a new _empty_ Symbol instance. + Symbol(napi_env env, + napi_value value); ///< Wraps a Node-API value primitive. +}; + +/// A JavaScript object value. +class Object : public Value { + public: + /// Enables property and element assignments using indexing syntax. + /// + /// This is a convenient helper to get and set object properties. As + /// getting and setting object properties may throw with JavaScript + /// exceptions, it is notable that these operations may fail. + /// When NODE_ADDON_API_ENABLE_MAYBE is defined, the process will abort + /// on JavaScript exceptions. + /// + /// Example: + /// + /// Napi::Value propertyValue = object1['A']; + /// object2['A'] = propertyValue; + /// Napi::Value elementValue = array[0]; + /// array[1] = elementValue; + template + class PropertyLValue { + public: + /// Converts an L-value to a value. + operator Value() const; + + /// Assigns a value to the property. The type of value can be + /// anything supported by `Object::Set`. + template + PropertyLValue& operator=(ValueType value); + + private: + PropertyLValue() = delete; + PropertyLValue(Object object, Key key); + napi_env _env; + napi_value _object; + Key _key; + + friend class Napi::Object; + }; + + /// Creates a new Object value. + static Object New(napi_env env ///< Node-API environment + ); + + Object(); ///< Creates a new _empty_ Object instance. + Object(napi_env env, + napi_value value); ///< Wraps a Node-API value primitive. + + /// Gets or sets a named property. + PropertyLValue operator[]( + const char* utf8name ///< UTF-8 encoded null-terminated property name + ); + + /// Gets or sets a named property. + PropertyLValue operator[]( + const std::string& utf8name ///< UTF-8 encoded property name + ); + + /// Gets or sets an indexed property or array element. + PropertyLValue operator[]( + uint32_t index /// Property / element index + ); + + /// Gets or sets an indexed property or array element. + PropertyLValue operator[](Value index /// Property / element index + ) const; + + /// Gets a named property. + MaybeOrValue operator[]( + const char* utf8name ///< UTF-8 encoded null-terminated property name + ) const; + + /// Gets a named property. + MaybeOrValue operator[]( + const std::string& utf8name ///< UTF-8 encoded property name + ) const; + + /// Gets an indexed property or array element. + MaybeOrValue operator[](uint32_t index ///< Property / element index + ) const; + + /// Checks whether a property is present. + MaybeOrValue Has(napi_value key ///< Property key primitive + ) const; + + /// Checks whether a property is present. + MaybeOrValue Has(Value key ///< Property key + ) const; + + /// Checks whether a named property is present. + MaybeOrValue Has( + const char* utf8name ///< UTF-8 encoded null-terminated property name + ) const; + + /// Checks whether a named property is present. + MaybeOrValue Has( + const std::string& utf8name ///< UTF-8 encoded property name + ) const; + + /// Checks whether a own property is present. + MaybeOrValue HasOwnProperty(napi_value key ///< Property key primitive + ) const; + + /// Checks whether a own property is present. + MaybeOrValue HasOwnProperty(Value key ///< Property key + ) const; + + /// Checks whether a own property is present. + MaybeOrValue HasOwnProperty( + const char* utf8name ///< UTF-8 encoded null-terminated property name + ) const; + + /// Checks whether a own property is present. + MaybeOrValue HasOwnProperty( + const std::string& utf8name ///< UTF-8 encoded property name + ) const; + + /// Gets a property. + MaybeOrValue Get(napi_value key ///< Property key primitive + ) const; + + /// Gets a property. + MaybeOrValue Get(Value key ///< Property key + ) const; + + /// Gets a named property. + MaybeOrValue Get( + const char* utf8name ///< UTF-8 encoded null-terminated property name + ) const; + + /// Gets a named property. + MaybeOrValue Get( + const std::string& utf8name ///< UTF-8 encoded property name + ) const; + + /// Sets a property. + template + MaybeOrValue Set(napi_value key, ///< Property key primitive + const ValueType& value ///< Property value primitive + ) const; + + /// Sets a property. + template + MaybeOrValue Set(Value key, ///< Property key + const ValueType& value ///< Property value + ) const; + + /// Sets a named property. + template + MaybeOrValue Set( + const char* utf8name, ///< UTF-8 encoded null-terminated property name + const ValueType& value) const; + + /// Sets a named property. + template + MaybeOrValue Set( + const std::string& utf8name, ///< UTF-8 encoded property name + const ValueType& value ///< Property value primitive + ) const; + + /// Delete property. + MaybeOrValue Delete(napi_value key ///< Property key primitive + ) const; + + /// Delete property. + MaybeOrValue Delete(Value key ///< Property key + ) const; + + /// Delete property. + MaybeOrValue Delete( + const char* utf8name ///< UTF-8 encoded null-terminated property name + ) const; + + /// Delete property. + MaybeOrValue Delete( + const std::string& utf8name ///< UTF-8 encoded property name + ) const; + + /// Checks whether an indexed property is present. + MaybeOrValue Has(uint32_t index ///< Property / element index + ) const; + + /// Gets an indexed property or array element. + MaybeOrValue Get(uint32_t index ///< Property / element index + ) const; + + /// Sets an indexed property or array element. + template + MaybeOrValue Set(uint32_t index, ///< Property / element index + const ValueType& value ///< Property value primitive + ) const; + + /// Deletes an indexed property or array element. + MaybeOrValue Delete(uint32_t index ///< Property / element index + ) const; + + /// This operation can fail in case of Proxy.[[OwnPropertyKeys]] and + /// Proxy.[[GetOwnProperty]] calling into JavaScript. See: + /// - + /// https://tc39.es/ecma262/#sec-proxy-object-internal-methods-and-internal-slots-ownpropertykeys + /// - + /// https://tc39.es/ecma262/#sec-proxy-object-internal-methods-and-internal-slots-getownproperty-p + MaybeOrValue GetPropertyNames() const; ///< Get all property names + + /// Defines a property on the object. + /// + /// This operation can fail in case of Proxy.[[DefineOwnProperty]] calling + /// into JavaScript. See + /// https://tc39.es/ecma262/#sec-proxy-object-internal-methods-and-internal-slots-defineownproperty-p-desc + MaybeOrValue DefineProperty( + const PropertyDescriptor& + property ///< Descriptor for the property to be defined + ) const; + + /// Defines properties on the object. + /// + /// This operation can fail in case of Proxy.[[DefineOwnProperty]] calling + /// into JavaScript. See + /// https://tc39.es/ecma262/#sec-proxy-object-internal-methods-and-internal-slots-defineownproperty-p-desc + MaybeOrValue DefineProperties( + const std::initializer_list& properties + ///< List of descriptors for the properties to be defined + ) const; + + /// Defines properties on the object. + /// + /// This operation can fail in case of Proxy.[[DefineOwnProperty]] calling + /// into JavaScript. See + /// https://tc39.es/ecma262/#sec-proxy-object-internal-methods-and-internal-slots-defineownproperty-p-desc + MaybeOrValue DefineProperties( + const std::vector& properties + ///< Vector of descriptors for the properties to be defined + ) const; + + /// Checks if an object is an instance created by a constructor function. + /// + /// This is equivalent to the JavaScript `instanceof` operator. + /// + /// This operation can fail in case of Proxy.[[GetPrototypeOf]] calling into + /// JavaScript. + /// See + /// https://tc39.es/ecma262/#sec-proxy-object-internal-methods-and-internal-slots-getprototypeof + MaybeOrValue InstanceOf( + const Function& constructor ///< Constructor function + ) const; + + template + inline void AddFinalizer(Finalizer finalizeCallback, T* data) const; + + template + inline void AddFinalizer(Finalizer finalizeCallback, + T* data, + Hint* finalizeHint) const; + +#ifdef NAPI_CPP_EXCEPTIONS + class const_iterator; + + inline const_iterator begin() const; + + inline const_iterator end() const; + + class iterator; + + inline iterator begin(); + + inline iterator end(); +#endif // NAPI_CPP_EXCEPTIONS + +#if NAPI_VERSION >= 8 + /// This operation can fail in case of Proxy.[[GetPrototypeOf]] calling into + /// JavaScript. + /// See + /// https://tc39.es/ecma262/#sec-proxy-object-internal-methods-and-internal-slots-getprototypeof + MaybeOrValue Freeze() const; + /// This operation can fail in case of Proxy.[[GetPrototypeOf]] calling into + /// JavaScript. + /// See + /// https://tc39.es/ecma262/#sec-proxy-object-internal-methods-and-internal-slots-getprototypeof + MaybeOrValue Seal() const; +#endif // NAPI_VERSION >= 8 +}; + +template +class External : public Value { + public: + static External New(napi_env env, T* data); + + // Finalizer must implement `void operator()(Env env, T* data)`. + template + static External New(napi_env env, T* data, Finalizer finalizeCallback); + // Finalizer must implement `void operator()(Env env, T* data, Hint* hint)`. + template + static External New(napi_env env, + T* data, + Finalizer finalizeCallback, + Hint* finalizeHint); + + External(); + External(napi_env env, napi_value value); + + T* Data() const; +}; + +class Array : public Object { + public: + static Array New(napi_env env); + static Array New(napi_env env, size_t length); + + Array(); + Array(napi_env env, napi_value value); + + uint32_t Length() const; +}; + +#ifdef NAPI_CPP_EXCEPTIONS +class Object::const_iterator { + private: + enum class Type { BEGIN, END }; + + inline const_iterator(const Object* object, const Type type); + + public: + inline const_iterator& operator++(); + + inline bool operator==(const const_iterator& other) const; + + inline bool operator!=(const const_iterator& other) const; + + inline const std::pair> operator*() + const; + + private: + const Napi::Object* _object; + Array _keys; + uint32_t _index; + + friend class Object; +}; + +class Object::iterator { + private: + enum class Type { BEGIN, END }; + + inline iterator(Object* object, const Type type); + + public: + inline iterator& operator++(); + + inline bool operator==(const iterator& other) const; + + inline bool operator!=(const iterator& other) const; + + inline std::pair> operator*(); + + private: + Napi::Object* _object; + Array _keys; + uint32_t _index; + + friend class Object; +}; +#endif // NAPI_CPP_EXCEPTIONS + +/// A JavaScript array buffer value. +class ArrayBuffer : public Object { + public: + /// Creates a new ArrayBuffer instance over a new automatically-allocated + /// buffer. + static ArrayBuffer New( + napi_env env, ///< Node-API environment + size_t byteLength ///< Length of the buffer to be allocated, in bytes + ); + + /// Creates a new ArrayBuffer instance, using an external buffer with + /// specified byte length. + static ArrayBuffer New( + napi_env env, ///< Node-API environment + void* externalData, ///< Pointer to the external buffer to be used by + ///< the array + size_t byteLength ///< Length of the external buffer to be used by the + ///< array, in bytes + ); + + /// Creates a new ArrayBuffer instance, using an external buffer with + /// specified byte length. + template + static ArrayBuffer New( + napi_env env, ///< Node-API environment + void* externalData, ///< Pointer to the external buffer to be used by + ///< the array + size_t byteLength, ///< Length of the external buffer to be used by the + ///< array, + /// in bytes + Finalizer finalizeCallback ///< Function to be called when the array + ///< buffer is destroyed; + /// must implement `void operator()(Env env, + /// void* externalData)` + ); + + /// Creates a new ArrayBuffer instance, using an external buffer with + /// specified byte length. + template + static ArrayBuffer New( + napi_env env, ///< Node-API environment + void* externalData, ///< Pointer to the external buffer to be used by + ///< the array + size_t byteLength, ///< Length of the external buffer to be used by the + ///< array, + /// in bytes + Finalizer finalizeCallback, ///< Function to be called when the array + ///< buffer is destroyed; + /// must implement `void operator()(Env + /// env, void* externalData, Hint* hint)` + Hint* finalizeHint ///< Hint (second parameter) to be passed to the + ///< finalize callback + ); + + ArrayBuffer(); ///< Creates a new _empty_ ArrayBuffer instance. + ArrayBuffer(napi_env env, + napi_value value); ///< Wraps a Node-API value primitive. + + void* Data(); ///< Gets a pointer to the data buffer. + size_t ByteLength(); ///< Gets the length of the array buffer in bytes. + +#if NAPI_VERSION >= 7 + bool IsDetached() const; + void Detach(); +#endif // NAPI_VERSION >= 7 +}; + +/// A JavaScript typed-array value with unknown array type. +/// +/// For type-specific operations, cast to a `TypedArrayOf` instance using the +/// `As()` method: +/// +/// Napi::TypedArray array = ... +/// if (t.TypedArrayType() == napi_int32_array) { +/// Napi::Int32Array int32Array = t.As(); +/// } +class TypedArray : public Object { + public: + TypedArray(); ///< Creates a new _empty_ TypedArray instance. + TypedArray(napi_env env, + napi_value value); ///< Wraps a Node-API value primitive. + + napi_typedarray_type TypedArrayType() + const; ///< Gets the type of this typed-array. + Napi::ArrayBuffer ArrayBuffer() const; ///< Gets the backing array buffer. + + uint8_t ElementSize() + const; ///< Gets the size in bytes of one element in the array. + size_t ElementLength() const; ///< Gets the number of elements in the array. + size_t ByteOffset() + const; ///< Gets the offset into the buffer where the array starts. + size_t ByteLength() const; ///< Gets the length of the array in bytes. + + protected: + /// !cond INTERNAL + napi_typedarray_type _type; + size_t _length; + + TypedArray(napi_env env, + napi_value value, + napi_typedarray_type type, + size_t length); + + template + static +#if defined(NAPI_HAS_CONSTEXPR) + constexpr +#endif + napi_typedarray_type + TypedArrayTypeForPrimitiveType() { + return std::is_same::value ? napi_int8_array + : std::is_same::value ? napi_uint8_array + : std::is_same::value ? napi_int16_array + : std::is_same::value ? napi_uint16_array + : std::is_same::value ? napi_int32_array + : std::is_same::value ? napi_uint32_array + : std::is_same::value ? napi_float32_array + : std::is_same::value ? napi_float64_array +#if NAPI_VERSION > 5 + : std::is_same::value ? napi_bigint64_array + : std::is_same::value ? napi_biguint64_array +#endif // NAPI_VERSION > 5 + : napi_int8_array; + } + /// !endcond +}; + +/// A JavaScript typed-array value with known array type. +/// +/// Note while it is possible to create and access Uint8 "clamped" arrays using +/// this class, the _clamping_ behavior is only applied in JavaScript. +template +class TypedArrayOf : public TypedArray { + public: + /// Creates a new TypedArray instance over a new automatically-allocated array + /// buffer. + /// + /// The array type parameter can normally be omitted (because it is inferred + /// from the template parameter T), except when creating a "clamped" array: + /// + /// Uint8Array::New(env, length, napi_uint8_clamped_array) + static TypedArrayOf New( + napi_env env, ///< Node-API environment + size_t elementLength, ///< Length of the created array, as a number of + ///< elements +#if defined(NAPI_HAS_CONSTEXPR) + napi_typedarray_type type = + TypedArray::TypedArrayTypeForPrimitiveType() +#else + napi_typedarray_type type +#endif + ///< Type of array, if different from the default array type for the + ///< template parameter T. + ); + + /// Creates a new TypedArray instance over a provided array buffer. + /// + /// The array type parameter can normally be omitted (because it is inferred + /// from the template parameter T), except when creating a "clamped" array: + /// + /// Uint8Array::New(env, length, buffer, 0, napi_uint8_clamped_array) + static TypedArrayOf New( + napi_env env, ///< Node-API environment + size_t elementLength, ///< Length of the created array, as a number of + ///< elements + Napi::ArrayBuffer arrayBuffer, ///< Backing array buffer instance to use + size_t bufferOffset, ///< Offset into the array buffer where the + ///< typed-array starts +#if defined(NAPI_HAS_CONSTEXPR) + napi_typedarray_type type = + TypedArray::TypedArrayTypeForPrimitiveType() +#else + napi_typedarray_type type +#endif + ///< Type of array, if different from the default array type for the + ///< template parameter T. + ); + + TypedArrayOf(); ///< Creates a new _empty_ TypedArrayOf instance. + TypedArrayOf(napi_env env, + napi_value value); ///< Wraps a Node-API value primitive. + + T& operator[](size_t index); ///< Gets or sets an element in the array. + const T& operator[](size_t index) const; ///< Gets an element in the array. + + /// Gets a pointer to the array's backing buffer. + /// + /// This is not necessarily the same as the `ArrayBuffer::Data()` pointer, + /// because the typed-array may have a non-zero `ByteOffset()` into the + /// `ArrayBuffer`. + T* Data(); + + /// Gets a pointer to the array's backing buffer. + /// + /// This is not necessarily the same as the `ArrayBuffer::Data()` pointer, + /// because the typed-array may have a non-zero `ByteOffset()` into the + /// `ArrayBuffer`. + const T* Data() const; + + private: + T* _data; + + TypedArrayOf(napi_env env, + napi_value value, + napi_typedarray_type type, + size_t length, + T* data); +}; + +/// The DataView provides a low-level interface for reading/writing multiple +/// number types in an ArrayBuffer irrespective of the platform's endianness. +class DataView : public Object { + public: + static DataView New(napi_env env, Napi::ArrayBuffer arrayBuffer); + static DataView New(napi_env env, + Napi::ArrayBuffer arrayBuffer, + size_t byteOffset); + static DataView New(napi_env env, + Napi::ArrayBuffer arrayBuffer, + size_t byteOffset, + size_t byteLength); + + DataView(); ///< Creates a new _empty_ DataView instance. + DataView(napi_env env, + napi_value value); ///< Wraps a Node-API value primitive. + + Napi::ArrayBuffer ArrayBuffer() const; ///< Gets the backing array buffer. + size_t ByteOffset() + const; ///< Gets the offset into the buffer where the array starts. + size_t ByteLength() const; ///< Gets the length of the array in bytes. + + void* Data() const; + + float GetFloat32(size_t byteOffset) const; + double GetFloat64(size_t byteOffset) const; + int8_t GetInt8(size_t byteOffset) const; + int16_t GetInt16(size_t byteOffset) const; + int32_t GetInt32(size_t byteOffset) const; + uint8_t GetUint8(size_t byteOffset) const; + uint16_t GetUint16(size_t byteOffset) const; + uint32_t GetUint32(size_t byteOffset) const; + + void SetFloat32(size_t byteOffset, float value) const; + void SetFloat64(size_t byteOffset, double value) const; + void SetInt8(size_t byteOffset, int8_t value) const; + void SetInt16(size_t byteOffset, int16_t value) const; + void SetInt32(size_t byteOffset, int32_t value) const; + void SetUint8(size_t byteOffset, uint8_t value) const; + void SetUint16(size_t byteOffset, uint16_t value) const; + void SetUint32(size_t byteOffset, uint32_t value) const; + + private: + template + T ReadData(size_t byteOffset) const; + + template + void WriteData(size_t byteOffset, T value) const; + + void* _data; + size_t _length; +}; + +class Function : public Object { + public: + using VoidCallback = void (*)(const CallbackInfo& info); + using Callback = Value (*)(const CallbackInfo& info); + + template + static Function New(napi_env env, + const char* utf8name = nullptr, + void* data = nullptr); + + template + static Function New(napi_env env, + const char* utf8name = nullptr, + void* data = nullptr); + + template + static Function New(napi_env env, + const std::string& utf8name, + void* data = nullptr); + + template + static Function New(napi_env env, + const std::string& utf8name, + void* data = nullptr); + + /// Callable must implement operator() accepting a const CallbackInfo& + /// and return either void or Value. + template + static Function New(napi_env env, + Callable cb, + const char* utf8name = nullptr, + void* data = nullptr); + /// Callable must implement operator() accepting a const CallbackInfo& + /// and return either void or Value. + template + static Function New(napi_env env, + Callable cb, + const std::string& utf8name, + void* data = nullptr); + + Function(); + Function(napi_env env, napi_value value); + + MaybeOrValue operator()( + const std::initializer_list& args) const; + + MaybeOrValue Call(const std::initializer_list& args) const; + MaybeOrValue Call(const std::vector& args) const; + MaybeOrValue Call(const std::vector& args) const; + MaybeOrValue Call(size_t argc, const napi_value* args) const; + MaybeOrValue Call(napi_value recv, + const std::initializer_list& args) const; + MaybeOrValue Call(napi_value recv, + const std::vector& args) const; + MaybeOrValue Call(napi_value recv, + const std::vector& args) const; + MaybeOrValue Call(napi_value recv, + size_t argc, + const napi_value* args) const; + + MaybeOrValue MakeCallback( + napi_value recv, + const std::initializer_list& args, + napi_async_context context = nullptr) const; + MaybeOrValue MakeCallback(napi_value recv, + const std::vector& args, + napi_async_context context = nullptr) const; + MaybeOrValue MakeCallback(napi_value recv, + size_t argc, + const napi_value* args, + napi_async_context context = nullptr) const; + + MaybeOrValue New(const std::initializer_list& args) const; + MaybeOrValue New(const std::vector& args) const; + MaybeOrValue New(size_t argc, const napi_value* args) const; +}; + +class Promise : public Object { + public: + class Deferred { + public: + static Deferred New(napi_env env); + Deferred(napi_env env); + + Napi::Promise Promise() const; + Napi::Env Env() const; + + void Resolve(napi_value value) const; + void Reject(napi_value value) const; + + private: + napi_env _env; + napi_deferred _deferred; + napi_value _promise; + }; + + Promise(napi_env env, napi_value value); +}; + +template +class Buffer : public Uint8Array { + public: + static Buffer New(napi_env env, size_t length); + static Buffer New(napi_env env, T* data, size_t length); + + // Finalizer must implement `void operator()(Env env, T* data)`. + template + static Buffer New(napi_env env, + T* data, + size_t length, + Finalizer finalizeCallback); + // Finalizer must implement `void operator()(Env env, T* data, Hint* hint)`. + template + static Buffer New(napi_env env, + T* data, + size_t length, + Finalizer finalizeCallback, + Hint* finalizeHint); + + static Buffer Copy(napi_env env, const T* data, size_t length); + + Buffer(); + Buffer(napi_env env, napi_value value); + size_t Length() const; + T* Data() const; + + private: + mutable size_t _length; + mutable T* _data; + + Buffer(napi_env env, napi_value value, size_t length, T* data); + void EnsureInfo() const; +}; + +/// Holds a counted reference to a value; initially a weak reference unless +/// otherwise specified, may be changed to/from a strong reference by adjusting +/// the refcount. +/// +/// The referenced value is not immediately destroyed when the reference count +/// is zero; it is merely then eligible for garbage-collection if there are no +/// other references to the value. +template +class Reference { + public: + static Reference New(const T& value, uint32_t initialRefcount = 0); + + Reference(); + Reference(napi_env env, napi_ref ref); + ~Reference(); + + // A reference can be moved but cannot be copied. + Reference(Reference&& other); + Reference& operator=(Reference&& other); + NAPI_DISALLOW_ASSIGN(Reference) + + operator napi_ref() const; + bool operator==(const Reference& other) const; + bool operator!=(const Reference& other) const; + + Napi::Env Env() const; + bool IsEmpty() const; + + // Note when getting the value of a Reference it is usually correct to do so + // within a HandleScope so that the value handle gets cleaned up efficiently. + T Value() const; + + uint32_t Ref() const; + uint32_t Unref() const; + void Reset(); + void Reset(const T& value, uint32_t refcount = 0); + + // Call this on a reference that is declared as static data, to prevent its + // destructor from running at program shutdown time, which would attempt to + // reset the reference when the environment is no longer valid. Avoid using + // this if at all possible. If you do need to use static data, MAKE SURE to + // warn your users that your addon is NOT threadsafe. + void SuppressDestruct(); + + protected: + Reference(const Reference&); + + /// !cond INTERNAL + napi_env _env; + napi_ref _ref; + /// !endcond + + private: + bool _suppressDestruct; +}; + +class ObjectReference : public Reference { + public: + ObjectReference(); + ObjectReference(napi_env env, napi_ref ref); + + // A reference can be moved but cannot be copied. + ObjectReference(Reference&& other); + ObjectReference& operator=(Reference&& other); + ObjectReference(ObjectReference&& other); + ObjectReference& operator=(ObjectReference&& other); + NAPI_DISALLOW_ASSIGN(ObjectReference) + + MaybeOrValue Get(const char* utf8name) const; + MaybeOrValue Get(const std::string& utf8name) const; + MaybeOrValue Set(const char* utf8name, napi_value value) const; + MaybeOrValue Set(const char* utf8name, Napi::Value value) const; + MaybeOrValue Set(const char* utf8name, const char* utf8value) const; + MaybeOrValue Set(const char* utf8name, bool boolValue) const; + MaybeOrValue Set(const char* utf8name, double numberValue) const; + MaybeOrValue Set(const std::string& utf8name, napi_value value) const; + MaybeOrValue Set(const std::string& utf8name, Napi::Value value) const; + MaybeOrValue Set(const std::string& utf8name, + std::string& utf8value) const; + MaybeOrValue Set(const std::string& utf8name, bool boolValue) const; + MaybeOrValue Set(const std::string& utf8name, double numberValue) const; + + MaybeOrValue Get(uint32_t index) const; + MaybeOrValue Set(uint32_t index, const napi_value value) const; + MaybeOrValue Set(uint32_t index, const Napi::Value value) const; + MaybeOrValue Set(uint32_t index, const char* utf8value) const; + MaybeOrValue Set(uint32_t index, const std::string& utf8value) const; + MaybeOrValue Set(uint32_t index, bool boolValue) const; + MaybeOrValue Set(uint32_t index, double numberValue) const; + + protected: + ObjectReference(const ObjectReference&); +}; + +class FunctionReference : public Reference { + public: + FunctionReference(); + FunctionReference(napi_env env, napi_ref ref); + + // A reference can be moved but cannot be copied. + FunctionReference(Reference&& other); + FunctionReference& operator=(Reference&& other); + FunctionReference(FunctionReference&& other); + FunctionReference& operator=(FunctionReference&& other); + NAPI_DISALLOW_ASSIGN_COPY(FunctionReference) + + MaybeOrValue operator()( + const std::initializer_list& args) const; + + MaybeOrValue Call( + const std::initializer_list& args) const; + MaybeOrValue Call(const std::vector& args) const; + MaybeOrValue Call( + napi_value recv, const std::initializer_list& args) const; + MaybeOrValue Call(napi_value recv, + const std::vector& args) const; + MaybeOrValue Call(napi_value recv, + size_t argc, + const napi_value* args) const; + + MaybeOrValue MakeCallback( + napi_value recv, + const std::initializer_list& args, + napi_async_context context = nullptr) const; + MaybeOrValue MakeCallback( + napi_value recv, + const std::vector& args, + napi_async_context context = nullptr) const; + MaybeOrValue MakeCallback( + napi_value recv, + size_t argc, + const napi_value* args, + napi_async_context context = nullptr) const; + + MaybeOrValue New(const std::initializer_list& args) const; + MaybeOrValue New(const std::vector& args) const; +}; + +// Shortcuts to creating a new reference with inferred type and refcount = 0. +template +Reference Weak(T value); +ObjectReference Weak(Object value); +FunctionReference Weak(Function value); + +// Shortcuts to creating a new reference with inferred type and refcount = 1. +template +Reference Persistent(T value); +ObjectReference Persistent(Object value); +FunctionReference Persistent(Function value); + +/// A persistent reference to a JavaScript error object. Use of this class +/// depends somewhat on whether C++ exceptions are enabled at compile time. +/// +/// ### Handling Errors With C++ Exceptions +/// +/// If C++ exceptions are enabled, then the `Error` class extends +/// `std::exception` and enables integrated error-handling for C++ exceptions +/// and JavaScript exceptions. +/// +/// If a Node-API call fails without executing any JavaScript code (for +/// example due to an invalid argument), then the Node-API wrapper +/// automatically converts and throws the error as a C++ exception of type +/// `Napi::Error`. Or if a JavaScript function called by C++ code via Node-API +/// throws a JavaScript exception, then the Node-API wrapper automatically +/// converts and throws it as a C++ exception of type `Napi::Error`. +/// +/// If a C++ exception of type `Napi::Error` escapes from a Node-API C++ +/// callback, then the Node-API wrapper automatically converts and throws it +/// as a JavaScript exception. Therefore, catching a C++ exception of type +/// `Napi::Error` prevents a JavaScript exception from being thrown. +/// +/// #### Example 1A - Throwing a C++ exception: +/// +/// Napi::Env env = ... +/// throw Napi::Error::New(env, "Example exception"); +/// +/// Following C++ statements will not be executed. The exception will bubble +/// up as a C++ exception of type `Napi::Error`, until it is either caught +/// while still in C++, or else automatically propataged as a JavaScript +/// exception when the callback returns to JavaScript. +/// +/// #### Example 2A - Propagating a Node-API C++ exception: +/// +/// Napi::Function jsFunctionThatThrows = someObj.As(); +/// Napi::Value result = jsFunctionThatThrows({ arg1, arg2 }); +/// +/// Following C++ statements will not be executed. The exception will bubble +/// up as a C++ exception of type `Napi::Error`, until it is either caught +/// while still in C++, or else automatically propagated as a JavaScript +/// exception when the callback returns to JavaScript. +/// +/// #### Example 3A - Handling a Node-API C++ exception: +/// +/// Napi::Function jsFunctionThatThrows = someObj.As(); +/// Napi::Value result; +/// try { +/// result = jsFunctionThatThrows({ arg1, arg2 }); +/// } catch (const Napi::Error& e) { +/// cerr << "Caught JavaScript exception: " + e.what(); +/// } +/// +/// Since the exception was caught here, it will not be propagated as a +/// JavaScript exception. +/// +/// ### Handling Errors Without C++ Exceptions +/// +/// If C++ exceptions are disabled (by defining `NAPI_DISABLE_CPP_EXCEPTIONS`) +/// then this class does not extend `std::exception`, and APIs in the `Napi` +/// namespace do not throw C++ exceptions when they fail. Instead, they raise +/// _pending_ JavaScript exceptions and return _empty_ `Value`s. Calling code +/// should check `Value::IsEmpty()` before attempting to use a returned value, +/// and may use methods on the `Env` class to check for, get, and clear a +/// pending JavaScript exception. If the pending exception is not cleared, it +/// will be thrown when the native callback returns to JavaScript. +/// +/// #### Example 1B - Throwing a JS exception +/// +/// Napi::Env env = ... +/// Napi::Error::New(env, "Example +/// exception").ThrowAsJavaScriptException(); return; +/// +/// After throwing a JS exception, the code should generally return +/// immediately from the native callback, after performing any necessary +/// cleanup. +/// +/// #### Example 2B - Propagating a Node-API JS exception: +/// +/// Napi::Function jsFunctionThatThrows = someObj.As(); +/// Napi::Value result = jsFunctionThatThrows({ arg1, arg2 }); +/// if (result.IsEmpty()) return; +/// +/// An empty value result from a Node-API call indicates an error occurred, +/// and a JavaScript exception is pending. To let the exception propagate, the +/// code should generally return immediately from the native callback, after +/// performing any necessary cleanup. +/// +/// #### Example 3B - Handling a Node-API JS exception: +/// +/// Napi::Function jsFunctionThatThrows = someObj.As(); +/// Napi::Value result = jsFunctionThatThrows({ arg1, arg2 }); +/// if (result.IsEmpty()) { +/// Napi::Error e = env.GetAndClearPendingException(); +/// cerr << "Caught JavaScript exception: " + e.Message(); +/// } +/// +/// Since the exception was cleared here, it will not be propagated as a +/// JavaScript exception after the native callback returns. +class Error : public ObjectReference +#ifdef NAPI_CPP_EXCEPTIONS + , + public std::exception +#endif // NAPI_CPP_EXCEPTIONS +{ + public: + static Error New(napi_env env); + static Error New(napi_env env, const char* message); + static Error New(napi_env env, const std::string& message); + + static NAPI_NO_RETURN void Fatal(const char* location, const char* message); + + Error(); + Error(napi_env env, napi_value value); + + // An error can be moved or copied. + Error(Error&& other); + Error& operator=(Error&& other); + Error(const Error&); + Error& operator=(const Error&); + + const std::string& Message() const NAPI_NOEXCEPT; + void ThrowAsJavaScriptException() const; + + Object Value() const; + +#ifdef NAPI_CPP_EXCEPTIONS + const char* what() const NAPI_NOEXCEPT override; +#endif // NAPI_CPP_EXCEPTIONS + + protected: + /// !cond INTERNAL + using create_error_fn = napi_status (*)(napi_env envb, + napi_value code, + napi_value msg, + napi_value* result); + + template + static TError New(napi_env env, + const char* message, + size_t length, + create_error_fn create_error); + /// !endcond + + private: + static inline const char* ERROR_WRAP_VALUE() NAPI_NOEXCEPT; + mutable std::string _message; +}; + +class TypeError : public Error { + public: + static TypeError New(napi_env env, const char* message); + static TypeError New(napi_env env, const std::string& message); + + TypeError(); + TypeError(napi_env env, napi_value value); +}; + +class RangeError : public Error { + public: + static RangeError New(napi_env env, const char* message); + static RangeError New(napi_env env, const std::string& message); + + RangeError(); + RangeError(napi_env env, napi_value value); +}; + +class CallbackInfo { + public: + CallbackInfo(napi_env env, napi_callback_info info); + ~CallbackInfo(); + + // Disallow copying to prevent multiple free of _dynamicArgs + NAPI_DISALLOW_ASSIGN_COPY(CallbackInfo) + + Napi::Env Env() const; + Value NewTarget() const; + bool IsConstructCall() const; + size_t Length() const; + const Value operator[](size_t index) const; + Value This() const; + void* Data() const; + void SetData(void* data); + operator napi_callback_info() const; + + private: + const size_t _staticArgCount = 6; + napi_env _env; + napi_callback_info _info; + napi_value _this; + size_t _argc; + napi_value* _argv; + napi_value _staticArgs[6]; + napi_value* _dynamicArgs; + void* _data; +}; + +class PropertyDescriptor { + public: + using GetterCallback = Napi::Value (*)(const Napi::CallbackInfo& info); + using SetterCallback = void (*)(const Napi::CallbackInfo& info); + +#ifndef NODE_ADDON_API_DISABLE_DEPRECATED + template + static PropertyDescriptor Accessor( + const char* utf8name, + Getter getter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Accessor( + const std::string& utf8name, + Getter getter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Accessor( + napi_value name, + Getter getter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Accessor( + Name name, + Getter getter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Accessor( + const char* utf8name, + Getter getter, + Setter setter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Accessor( + const std::string& utf8name, + Getter getter, + Setter setter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Accessor( + napi_value name, + Getter getter, + Setter setter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Accessor( + Name name, + Getter getter, + Setter setter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Function( + const char* utf8name, + Callable cb, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Function( + const std::string& utf8name, + Callable cb, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Function( + napi_value name, + Callable cb, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Function( + Name name, + Callable cb, + napi_property_attributes attributes = napi_default, + void* data = nullptr); +#endif // !NODE_ADDON_API_DISABLE_DEPRECATED + + template + static PropertyDescriptor Accessor( + const char* utf8name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + + template + static PropertyDescriptor Accessor( + const std::string& utf8name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + + template + static PropertyDescriptor Accessor( + Name name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + + template + static PropertyDescriptor Accessor( + const char* utf8name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + + template + static PropertyDescriptor Accessor( + const std::string& utf8name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + + template + static PropertyDescriptor Accessor( + Name name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + + template + static PropertyDescriptor Accessor( + Napi::Env env, + Napi::Object object, + const char* utf8name, + Getter getter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Accessor( + Napi::Env env, + Napi::Object object, + const std::string& utf8name, + Getter getter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Accessor( + Napi::Env env, + Napi::Object object, + Name name, + Getter getter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Accessor( + Napi::Env env, + Napi::Object object, + const char* utf8name, + Getter getter, + Setter setter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Accessor( + Napi::Env env, + Napi::Object object, + const std::string& utf8name, + Getter getter, + Setter setter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Accessor( + Napi::Env env, + Napi::Object object, + Name name, + Getter getter, + Setter setter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Function( + Napi::Env env, + Napi::Object object, + const char* utf8name, + Callable cb, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Function( + Napi::Env env, + Napi::Object object, + const std::string& utf8name, + Callable cb, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor Function( + Napi::Env env, + Napi::Object object, + Name name, + Callable cb, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor Value( + const char* utf8name, + napi_value value, + napi_property_attributes attributes = napi_default); + static PropertyDescriptor Value( + const std::string& utf8name, + napi_value value, + napi_property_attributes attributes = napi_default); + static PropertyDescriptor Value( + napi_value name, + napi_value value, + napi_property_attributes attributes = napi_default); + static PropertyDescriptor Value( + Name name, + Napi::Value value, + napi_property_attributes attributes = napi_default); + + PropertyDescriptor(napi_property_descriptor desc); + + operator napi_property_descriptor&(); + operator const napi_property_descriptor&() const; + + private: + napi_property_descriptor _desc; +}; + +/// Property descriptor for use with `ObjectWrap::DefineClass()`. +/// +/// This is different from the standalone `PropertyDescriptor` because it is +/// specific to each `ObjectWrap` subclass. This prevents using descriptors +/// from a different class when defining a new class (preventing the callbacks +/// from having incorrect `this` pointers). +template +class ClassPropertyDescriptor { + public: + ClassPropertyDescriptor(napi_property_descriptor desc) : _desc(desc) {} + + operator napi_property_descriptor&() { return _desc; } + operator const napi_property_descriptor&() const { return _desc; } + + private: + napi_property_descriptor _desc; +}; + +template +struct MethodCallbackData { + TCallback callback; + void* data; +}; + +template +struct AccessorCallbackData { + TGetterCallback getterCallback; + TSetterCallback setterCallback; + void* data; +}; + +template +class InstanceWrap { + public: + using InstanceVoidMethodCallback = void (T::*)(const CallbackInfo& info); + using InstanceMethodCallback = Napi::Value (T::*)(const CallbackInfo& info); + using InstanceGetterCallback = Napi::Value (T::*)(const CallbackInfo& info); + using InstanceSetterCallback = void (T::*)(const CallbackInfo& info, + const Napi::Value& value); + + using PropertyDescriptor = ClassPropertyDescriptor; + + static PropertyDescriptor InstanceMethod( + const char* utf8name, + InstanceVoidMethodCallback method, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor InstanceMethod( + const char* utf8name, + InstanceMethodCallback method, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor InstanceMethod( + Symbol name, + InstanceVoidMethodCallback method, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor InstanceMethod( + Symbol name, + InstanceMethodCallback method, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor InstanceMethod( + const char* utf8name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor InstanceMethod( + const char* utf8name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor InstanceMethod( + Symbol name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor InstanceMethod( + Symbol name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor InstanceAccessor( + const char* utf8name, + InstanceGetterCallback getter, + InstanceSetterCallback setter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor InstanceAccessor( + Symbol name, + InstanceGetterCallback getter, + InstanceSetterCallback setter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor InstanceAccessor( + const char* utf8name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor InstanceAccessor( + Symbol name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor InstanceValue( + const char* utf8name, + Napi::Value value, + napi_property_attributes attributes = napi_default); + static PropertyDescriptor InstanceValue( + Symbol name, + Napi::Value value, + napi_property_attributes attributes = napi_default); + + protected: + static void AttachPropData(napi_env env, + napi_value value, + const napi_property_descriptor* prop); + + private: + using This = InstanceWrap; + + using InstanceVoidMethodCallbackData = + MethodCallbackData; + using InstanceMethodCallbackData = + MethodCallbackData; + using InstanceAccessorCallbackData = + AccessorCallbackData; + + static napi_value InstanceVoidMethodCallbackWrapper(napi_env env, + napi_callback_info info); + static napi_value InstanceMethodCallbackWrapper(napi_env env, + napi_callback_info info); + static napi_value InstanceGetterCallbackWrapper(napi_env env, + napi_callback_info info); + static napi_value InstanceSetterCallbackWrapper(napi_env env, + napi_callback_info info); + + template + static napi_value WrappedMethod(napi_env env, + napi_callback_info info) NAPI_NOEXCEPT; + + template + struct SetterTag {}; + + template + static napi_callback WrapSetter(SetterTag) NAPI_NOEXCEPT { + return &This::WrappedMethod; + } + static napi_callback WrapSetter(SetterTag) NAPI_NOEXCEPT { + return nullptr; + } +}; + +/// Base class to be extended by C++ classes exposed to JavaScript; each C++ +/// class instance gets "wrapped" by a JavaScript object that is managed by this +/// class. +/// +/// At initialization time, the `DefineClass()` method must be used to +/// hook up the accessor and method callbacks. It takes a list of +/// property descriptors, which can be constructed via the various +/// static methods on the base class. +/// +/// #### Example: +/// +/// class Example: public Napi::ObjectWrap { +/// public: +/// static void Initialize(Napi::Env& env, Napi::Object& target) { +/// Napi::Function constructor = DefineClass(env, "Example", { +/// InstanceAccessor<&Example::GetSomething, +/// &Example::SetSomething>("value"), +/// InstanceMethod<&Example::DoSomething>("doSomething"), +/// }); +/// target.Set("Example", constructor); +/// } +/// +/// Example(const Napi::CallbackInfo& info); // Constructor +/// Napi::Value GetSomething(const Napi::CallbackInfo& info); +/// void SetSomething(const Napi::CallbackInfo& info, const Napi::Value& +/// value); Napi::Value DoSomething(const Napi::CallbackInfo& info); +/// } +template +class ObjectWrap : public InstanceWrap, public Reference { + public: + ObjectWrap(const CallbackInfo& callbackInfo); + virtual ~ObjectWrap(); + + static T* Unwrap(Object wrapper); + + // Methods exposed to JavaScript must conform to one of these callback + // signatures. + using StaticVoidMethodCallback = void (*)(const CallbackInfo& info); + using StaticMethodCallback = Napi::Value (*)(const CallbackInfo& info); + using StaticGetterCallback = Napi::Value (*)(const CallbackInfo& info); + using StaticSetterCallback = void (*)(const CallbackInfo& info, + const Napi::Value& value); + + using PropertyDescriptor = ClassPropertyDescriptor; + + static Function DefineClass( + Napi::Env env, + const char* utf8name, + const std::initializer_list& properties, + void* data = nullptr); + static Function DefineClass(Napi::Env env, + const char* utf8name, + const std::vector& properties, + void* data = nullptr); + static PropertyDescriptor StaticMethod( + const char* utf8name, + StaticVoidMethodCallback method, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor StaticMethod( + const char* utf8name, + StaticMethodCallback method, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor StaticMethod( + Symbol name, + StaticVoidMethodCallback method, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor StaticMethod( + Symbol name, + StaticMethodCallback method, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor StaticMethod( + const char* utf8name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor StaticMethod( + Symbol name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor StaticMethod( + const char* utf8name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor StaticMethod( + Symbol name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor StaticAccessor( + const char* utf8name, + StaticGetterCallback getter, + StaticSetterCallback setter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor StaticAccessor( + Symbol name, + StaticGetterCallback getter, + StaticSetterCallback setter, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor StaticAccessor( + const char* utf8name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + template + static PropertyDescriptor StaticAccessor( + Symbol name, + napi_property_attributes attributes = napi_default, + void* data = nullptr); + static PropertyDescriptor StaticValue( + const char* utf8name, + Napi::Value value, + napi_property_attributes attributes = napi_default); + static PropertyDescriptor StaticValue( + Symbol name, + Napi::Value value, + napi_property_attributes attributes = napi_default); + static Napi::Value OnCalledAsFunction(const Napi::CallbackInfo& callbackInfo); + virtual void Finalize(Napi::Env env); + + private: + using This = ObjectWrap; + + static napi_value ConstructorCallbackWrapper(napi_env env, + napi_callback_info info); + static napi_value StaticVoidMethodCallbackWrapper(napi_env env, + napi_callback_info info); + static napi_value StaticMethodCallbackWrapper(napi_env env, + napi_callback_info info); + static napi_value StaticGetterCallbackWrapper(napi_env env, + napi_callback_info info); + static napi_value StaticSetterCallbackWrapper(napi_env env, + napi_callback_info info); + static void FinalizeCallback(napi_env env, void* data, void* hint); + static Function DefineClass(Napi::Env env, + const char* utf8name, + const size_t props_count, + const napi_property_descriptor* props, + void* data = nullptr); + + using StaticVoidMethodCallbackData = + MethodCallbackData; + using StaticMethodCallbackData = MethodCallbackData; + + using StaticAccessorCallbackData = + AccessorCallbackData; + + template + static napi_value WrappedMethod(napi_env env, + napi_callback_info info) NAPI_NOEXCEPT; + + template + struct StaticSetterTag {}; + + template + static napi_callback WrapStaticSetter(StaticSetterTag) NAPI_NOEXCEPT { + return &This::WrappedMethod; + } + static napi_callback WrapStaticSetter(StaticSetterTag) + NAPI_NOEXCEPT { + return nullptr; + } + + bool _construction_failed = true; +}; + +class HandleScope { + public: + HandleScope(napi_env env, napi_handle_scope scope); + explicit HandleScope(Napi::Env env); + ~HandleScope(); + + // Disallow copying to prevent double close of napi_handle_scope + NAPI_DISALLOW_ASSIGN_COPY(HandleScope) + + operator napi_handle_scope() const; + + Napi::Env Env() const; + + private: + napi_env _env; + napi_handle_scope _scope; +}; + +class EscapableHandleScope { + public: + EscapableHandleScope(napi_env env, napi_escapable_handle_scope scope); + explicit EscapableHandleScope(Napi::Env env); + ~EscapableHandleScope(); + + // Disallow copying to prevent double close of napi_escapable_handle_scope + NAPI_DISALLOW_ASSIGN_COPY(EscapableHandleScope) + + operator napi_escapable_handle_scope() const; + + Napi::Env Env() const; + Value Escape(napi_value escapee); + + private: + napi_env _env; + napi_escapable_handle_scope _scope; +}; + +#if (NAPI_VERSION > 2) +class CallbackScope { + public: + CallbackScope(napi_env env, napi_callback_scope scope); + CallbackScope(napi_env env, napi_async_context context); + virtual ~CallbackScope(); + + // Disallow copying to prevent double close of napi_callback_scope + NAPI_DISALLOW_ASSIGN_COPY(CallbackScope) + + operator napi_callback_scope() const; + + Napi::Env Env() const; + + private: + napi_env _env; + napi_callback_scope _scope; +}; +#endif + +class AsyncContext { + public: + explicit AsyncContext(napi_env env, const char* resource_name); + explicit AsyncContext(napi_env env, + const char* resource_name, + const Object& resource); + virtual ~AsyncContext(); + + AsyncContext(AsyncContext&& other); + AsyncContext& operator=(AsyncContext&& other); + NAPI_DISALLOW_ASSIGN_COPY(AsyncContext) + + operator napi_async_context() const; + + Napi::Env Env() const; + + private: + napi_env _env; + napi_async_context _context; +}; + +class AsyncWorker { + public: + virtual ~AsyncWorker(); + + // An async worker can be moved but cannot be copied. + AsyncWorker(AsyncWorker&& other); + AsyncWorker& operator=(AsyncWorker&& other); + NAPI_DISALLOW_ASSIGN_COPY(AsyncWorker) + + operator napi_async_work() const; + + Napi::Env Env() const; + + void Queue(); + void Cancel(); + void SuppressDestruct(); + + ObjectReference& Receiver(); + FunctionReference& Callback(); + + virtual void OnExecute(Napi::Env env); + virtual void OnWorkComplete(Napi::Env env, napi_status status); + + protected: + explicit AsyncWorker(const Function& callback); + explicit AsyncWorker(const Function& callback, const char* resource_name); + explicit AsyncWorker(const Function& callback, + const char* resource_name, + const Object& resource); + explicit AsyncWorker(const Object& receiver, const Function& callback); + explicit AsyncWorker(const Object& receiver, + const Function& callback, + const char* resource_name); + explicit AsyncWorker(const Object& receiver, + const Function& callback, + const char* resource_name, + const Object& resource); + + explicit AsyncWorker(Napi::Env env); + explicit AsyncWorker(Napi::Env env, const char* resource_name); + explicit AsyncWorker(Napi::Env env, + const char* resource_name, + const Object& resource); + + virtual void Execute() = 0; + virtual void OnOK(); + virtual void OnError(const Error& e); + virtual void Destroy(); + virtual std::vector GetResult(Napi::Env env); + + void SetError(const std::string& error); + + private: + static inline void OnAsyncWorkExecute(napi_env env, void* asyncworker); + static inline void OnAsyncWorkComplete(napi_env env, + napi_status status, + void* asyncworker); + + napi_env _env; + napi_async_work _work; + ObjectReference _receiver; + FunctionReference _callback; + std::string _error; + bool _suppress_destruct; +}; + +#if (NAPI_VERSION > 3 && !defined(__wasm32__)) +class ThreadSafeFunction { + public: + // This API may only be called from the main thread. + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount); + + // This API may only be called from the main thread. + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context); + + // This API may only be called from the main thread. + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + Finalizer finalizeCallback); + + // This API may only be called from the main thread. + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + Finalizer finalizeCallback, + FinalizerDataType* data); + + // This API may only be called from the main thread. + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback); + + // This API may only be called from the main thread. + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data); + + // This API may only be called from the main thread. + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount); + + // This API may only be called from the main thread. + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context); + + // This API may only be called from the main thread. + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + Finalizer finalizeCallback); + + // This API may only be called from the main thread. + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + Finalizer finalizeCallback, + FinalizerDataType* data); + + // This API may only be called from the main thread. + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback); + + // This API may only be called from the main thread. + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data); + + ThreadSafeFunction(); + ThreadSafeFunction(napi_threadsafe_function tsFunctionValue); + + operator napi_threadsafe_function() const; + + // This API may be called from any thread. + napi_status BlockingCall() const; + + // This API may be called from any thread. + template + napi_status BlockingCall(Callback callback) const; + + // This API may be called from any thread. + template + napi_status BlockingCall(DataType* data, Callback callback) const; + + // This API may be called from any thread. + napi_status NonBlockingCall() const; + + // This API may be called from any thread. + template + napi_status NonBlockingCall(Callback callback) const; + + // This API may be called from any thread. + template + napi_status NonBlockingCall(DataType* data, Callback callback) const; + + // This API may only be called from the main thread. + void Ref(napi_env env) const; + + // This API may only be called from the main thread. + void Unref(napi_env env) const; + + // This API may be called from any thread. + napi_status Acquire() const; + + // This API may be called from any thread. + napi_status Release() const; + + // This API may be called from any thread. + napi_status Abort() const; + + struct ConvertibleContext { + template + operator T*() { + return static_cast(context); + } + void* context; + }; + + // This API may be called from any thread. + ConvertibleContext GetContext() const; + + private: + using CallbackWrapper = std::function; + + template + static ThreadSafeFunction New(napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data, + napi_finalize wrapper); + + napi_status CallInternal(CallbackWrapper* callbackWrapper, + napi_threadsafe_function_call_mode mode) const; + + static void CallJS(napi_env env, + napi_value jsCallback, + void* context, + void* data); + + napi_threadsafe_function _tsfn; +}; + +// A TypedThreadSafeFunction by default has no context (nullptr) and can +// accept any type (void) to its CallJs. +template +class TypedThreadSafeFunction { + public: + // This API may only be called from the main thread. + // Helper function that returns nullptr if running Node-API 5+, otherwise a + // non-empty, no-op Function. This provides the ability to specify at + // compile-time a callback parameter to `New` that safely does no action + // when targeting _any_ Node-API version. +#if NAPI_VERSION > 4 + static std::nullptr_t EmptyFunctionFactory(Napi::Env env); +#else + static Napi::Function EmptyFunctionFactory(Napi::Env env); +#endif + static Napi::Function FunctionOrEmpty(Napi::Env env, + Napi::Function& callback); + +#if NAPI_VERSION > 4 + // This API may only be called from the main thread. + // Creates a new threadsafe function with: + // Callback [missing] Resource [missing] Finalizer [missing] + template + static TypedThreadSafeFunction New( + napi_env env, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context = nullptr); + + // This API may only be called from the main thread. + // Creates a new threadsafe function with: + // Callback [missing] Resource [passed] Finalizer [missing] + template + static TypedThreadSafeFunction New( + napi_env env, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context = nullptr); + + // This API may only be called from the main thread. + // Creates a new threadsafe function with: + // Callback [missing] Resource [missing] Finalizer [passed] + template + static TypedThreadSafeFunction New( + napi_env env, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data = nullptr); + + // This API may only be called from the main thread. + // Creates a new threadsafe function with: + // Callback [missing] Resource [passed] Finalizer [passed] + template + static TypedThreadSafeFunction New( + napi_env env, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data = nullptr); +#endif + + // This API may only be called from the main thread. + // Creates a new threadsafe function with: + // Callback [passed] Resource [missing] Finalizer [missing] + template + static TypedThreadSafeFunction New( + napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context = nullptr); + + // This API may only be called from the main thread. + // Creates a new threadsafe function with: + // Callback [passed] Resource [passed] Finalizer [missing] + template + static TypedThreadSafeFunction New( + napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context = nullptr); + + // This API may only be called from the main thread. + // Creates a new threadsafe function with: + // Callback [passed] Resource [missing] Finalizer [passed] + template + static TypedThreadSafeFunction New( + napi_env env, + const Function& callback, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data = nullptr); + + // This API may only be called from the main thread. + // Creates a new threadsafe function with: + // Callback [passed] Resource [passed] Finalizer [passed] + template + static TypedThreadSafeFunction New( + napi_env env, + CallbackType callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data = nullptr); + + TypedThreadSafeFunction(); + TypedThreadSafeFunction(napi_threadsafe_function tsFunctionValue); + + operator napi_threadsafe_function() const; + + // This API may be called from any thread. + napi_status BlockingCall(DataType* data = nullptr) const; + + // This API may be called from any thread. + napi_status NonBlockingCall(DataType* data = nullptr) const; + + // This API may only be called from the main thread. + void Ref(napi_env env) const; + + // This API may only be called from the main thread. + void Unref(napi_env env) const; + + // This API may be called from any thread. + napi_status Acquire() const; + + // This API may be called from any thread. + napi_status Release() const; + + // This API may be called from any thread. + napi_status Abort() const; + + // This API may be called from any thread. + ContextType* GetContext() const; + + private: + template + static TypedThreadSafeFunction New( + napi_env env, + const Function& callback, + const Object& resource, + ResourceString resourceName, + size_t maxQueueSize, + size_t initialThreadCount, + ContextType* context, + Finalizer finalizeCallback, + FinalizerDataType* data, + napi_finalize wrapper); + + static void CallJsInternal(napi_env env, + napi_value jsCallback, + void* context, + void* data); + + protected: + napi_threadsafe_function _tsfn; +}; +template +class AsyncProgressWorkerBase : public AsyncWorker { + public: + virtual void OnWorkProgress(DataType* data) = 0; + class ThreadSafeData { + public: + ThreadSafeData(AsyncProgressWorkerBase* asyncprogressworker, DataType* data) + : _asyncprogressworker(asyncprogressworker), _data(data) {} + + AsyncProgressWorkerBase* asyncprogressworker() { + return _asyncprogressworker; + }; + DataType* data() { return _data; }; + + private: + AsyncProgressWorkerBase* _asyncprogressworker; + DataType* _data; + }; + void OnWorkComplete(Napi::Env env, napi_status status) override; + + protected: + explicit AsyncProgressWorkerBase(const Object& receiver, + const Function& callback, + const char* resource_name, + const Object& resource, + size_t queue_size = 1); + virtual ~AsyncProgressWorkerBase(); + +// Optional callback of Napi::ThreadSafeFunction only available after +// NAPI_VERSION 4. Refs: https://github.com/nodejs/node/pull/27791 +#if NAPI_VERSION > 4 + explicit AsyncProgressWorkerBase(Napi::Env env, + const char* resource_name, + const Object& resource, + size_t queue_size = 1); +#endif + + static inline void OnAsyncWorkProgress(Napi::Env env, + Napi::Function jsCallback, + void* data); + + napi_status NonBlockingCall(DataType* data); + + private: + ThreadSafeFunction _tsfn; + bool _work_completed = false; + napi_status _complete_status; + static inline void OnThreadSafeFunctionFinalize( + Napi::Env env, void* data, AsyncProgressWorkerBase* context); +}; + +template +class AsyncProgressWorker : public AsyncProgressWorkerBase { + public: + virtual ~AsyncProgressWorker(); + + class ExecutionProgress { + friend class AsyncProgressWorker; + + public: + void Signal() const; + void Send(const T* data, size_t count) const; + + private: + explicit ExecutionProgress(AsyncProgressWorker* worker) : _worker(worker) {} + AsyncProgressWorker* const _worker; + }; + + void OnWorkProgress(void*) override; + + protected: + explicit AsyncProgressWorker(const Function& callback); + explicit AsyncProgressWorker(const Function& callback, + const char* resource_name); + explicit AsyncProgressWorker(const Function& callback, + const char* resource_name, + const Object& resource); + explicit AsyncProgressWorker(const Object& receiver, + const Function& callback); + explicit AsyncProgressWorker(const Object& receiver, + const Function& callback, + const char* resource_name); + explicit AsyncProgressWorker(const Object& receiver, + const Function& callback, + const char* resource_name, + const Object& resource); + +// Optional callback of Napi::ThreadSafeFunction only available after +// NAPI_VERSION 4. Refs: https://github.com/nodejs/node/pull/27791 +#if NAPI_VERSION > 4 + explicit AsyncProgressWorker(Napi::Env env); + explicit AsyncProgressWorker(Napi::Env env, const char* resource_name); + explicit AsyncProgressWorker(Napi::Env env, + const char* resource_name, + const Object& resource); +#endif + virtual void Execute(const ExecutionProgress& progress) = 0; + virtual void OnProgress(const T* data, size_t count) = 0; + + private: + void Execute() override; + void Signal(); + void SendProgress_(const T* data, size_t count); + + std::mutex _mutex; + T* _asyncdata; + size_t _asyncsize; + bool _signaled; +}; + +template +class AsyncProgressQueueWorker + : public AsyncProgressWorkerBase> { + public: + virtual ~AsyncProgressQueueWorker(){}; + + class ExecutionProgress { + friend class AsyncProgressQueueWorker; + + public: + void Signal() const; + void Send(const T* data, size_t count) const; + + private: + explicit ExecutionProgress(AsyncProgressQueueWorker* worker) + : _worker(worker) {} + AsyncProgressQueueWorker* const _worker; + }; + + void OnWorkComplete(Napi::Env env, napi_status status) override; + void OnWorkProgress(std::pair*) override; + + protected: + explicit AsyncProgressQueueWorker(const Function& callback); + explicit AsyncProgressQueueWorker(const Function& callback, + const char* resource_name); + explicit AsyncProgressQueueWorker(const Function& callback, + const char* resource_name, + const Object& resource); + explicit AsyncProgressQueueWorker(const Object& receiver, + const Function& callback); + explicit AsyncProgressQueueWorker(const Object& receiver, + const Function& callback, + const char* resource_name); + explicit AsyncProgressQueueWorker(const Object& receiver, + const Function& callback, + const char* resource_name, + const Object& resource); + +// Optional callback of Napi::ThreadSafeFunction only available after +// NAPI_VERSION 4. Refs: https://github.com/nodejs/node/pull/27791 +#if NAPI_VERSION > 4 + explicit AsyncProgressQueueWorker(Napi::Env env); + explicit AsyncProgressQueueWorker(Napi::Env env, const char* resource_name); + explicit AsyncProgressQueueWorker(Napi::Env env, + const char* resource_name, + const Object& resource); +#endif + virtual void Execute(const ExecutionProgress& progress) = 0; + virtual void OnProgress(const T* data, size_t count) = 0; + + private: + void Execute() override; + void Signal() const; + void SendProgress_(const T* data, size_t count); +}; +#endif // NAPI_VERSION > 3 && !defined(__wasm32__) + +// Memory management. +class MemoryManagement { + public: + static int64_t AdjustExternalMemory(Env env, int64_t change_in_bytes); +}; + +// Version management +class VersionManagement { + public: + static uint32_t GetNapiVersion(Env env); + static const napi_node_version* GetNodeVersion(Env env); +}; + +#if NAPI_VERSION > 5 +template +class Addon : public InstanceWrap { + public: + static inline Object Init(Env env, Object exports); + static T* Unwrap(Object wrapper); + + protected: + using AddonProp = ClassPropertyDescriptor; + void DefineAddon(Object exports, + const std::initializer_list& props); + Napi::Object DefineProperties(Object object, + const std::initializer_list& props); + + private: + Object entry_point_; +}; +#endif // NAPI_VERSION > 5 + +#ifdef NAPI_CPP_CUSTOM_NAMESPACE +} // namespace NAPI_CPP_CUSTOM_NAMESPACE +#endif + +} // namespace Napi + +// Inline implementations of all the above class methods are included here. +#include "napi-inl.h" + +#endif // SRC_NAPI_H_ diff --git a/node_modules/node-addon-api/node_api.gyp b/node_modules/node-addon-api/node_api.gyp new file mode 100644 index 0000000..4ff0ae7 --- /dev/null +++ b/node_modules/node-addon-api/node_api.gyp @@ -0,0 +1,9 @@ +{ + 'targets': [ + { + 'target_name': 'nothing', + 'type': 'static_library', + 'sources': [ 'nothing.c' ] + } + ] +} diff --git a/node_modules/node-addon-api/noexcept.gypi b/node_modules/node-addon-api/noexcept.gypi new file mode 100644 index 0000000..404a05f --- /dev/null +++ b/node_modules/node-addon-api/noexcept.gypi @@ -0,0 +1,26 @@ +{ + 'defines': [ 'NAPI_DISABLE_CPP_EXCEPTIONS' ], + 'cflags': [ '-fno-exceptions' ], + 'cflags_cc': [ '-fno-exceptions' ], + 'conditions': [ + ["OS=='win'", { + # _HAS_EXCEPTIONS is already defined and set to 0 in common.gypi + #"defines": [ + # "_HAS_EXCEPTIONS=0" + #], + "msvs_settings": { + "VCCLCompilerTool": { + 'ExceptionHandling': 0, + 'EnablePREfast': 'true', + }, + }, + }], + ["OS=='mac'", { + 'xcode_settings': { + 'CLANG_CXX_LIBRARY': 'libc++', + 'MACOSX_DEPLOYMENT_TARGET': '10.7', + 'GCC_ENABLE_CPP_EXCEPTIONS': 'NO', + }, + }], + ], +} diff --git a/node_modules/node-addon-api/nothing.c b/node_modules/node-addon-api/nothing.c new file mode 100644 index 0000000..e69de29 diff --git a/node_modules/node-addon-api/package-support.json b/node_modules/node-addon-api/package-support.json new file mode 100644 index 0000000..10d3607 --- /dev/null +++ b/node_modules/node-addon-api/package-support.json @@ -0,0 +1,21 @@ +{ + "versions": [ + { + "version": "*", + "target": { + "node": "active" + }, + "response": { + "type": "time-permitting", + "paid": false, + "contact": { + "name": "node-addon-api team", + "url": "https://github.com/nodejs/node-addon-api/issues" + } + }, + "backing": [ { "project": "https://github.com/nodejs" }, + { "foundation": "https://openjsf.org/" } + ] + } + ] +} diff --git a/node_modules/node-addon-api/package.json b/node_modules/node-addon-api/package.json new file mode 100644 index 0000000..3ec3776 --- /dev/null +++ b/node_modules/node-addon-api/package.json @@ -0,0 +1,456 @@ +{ + "bugs": { + "url": "https://github.com/nodejs/node-addon-api/issues" + }, + "contributors": [ + { + "name": "Abhishek Kumar Singh", + "url": "https://github.com/abhi11210646" + }, + { + "name": "Alba Mendez", + "url": "https://github.com/jmendeth" + }, + { + "name": "Alexander Floh", + "url": "https://github.com/alexanderfloh" + }, + { + "name": "Ammar Faizi", + "url": "https://github.com/ammarfaizi2" + }, + { + "name": "András Timár, Dr", + "url": "https://github.com/timarandras" + }, + { + "name": "Andrew Petersen", + "url": "https://github.com/kirbysayshi" + }, + { + "name": "Anisha Rohra", + "url": "https://github.com/anisha-rohra" + }, + { + "name": "Anna Henningsen", + "url": "https://github.com/addaleax" + }, + { + "name": "Arnaud Botella", + "url": "https://github.com/BotellaA" + }, + { + "name": "Arunesh Chandra", + "url": "https://github.com/aruneshchandra" + }, + { + "name": "Azlan Mukhtar", + "url": "https://github.com/azlan" + }, + { + "name": "Ben Berman", + "url": "https://github.com/rivertam" + }, + { + "name": "Benjamin Byholm", + "url": "https://github.com/kkoopa" + }, + { + "name": "Bill Gallafent", + "url": "https://github.com/gallafent" + }, + { + "name": "blagoev", + "url": "https://github.com/blagoev" + }, + { + "name": "Bruce A. MacNaughton", + "url": "https://github.com/bmacnaughton" + }, + { + "name": "Cory Mickelson", + "url": "https://github.com/corymickelson" + }, + { + "name": "Daniel Bevenius", + "url": "https://github.com/danbev" + }, + { + "name": "Dante Calderón", + "url": "https://github.com/dantehemerson" + }, + { + "name": "Darshan Sen", + "url": "https://github.com/RaisinTen" + }, + { + "name": "David Halls", + "url": "https://github.com/davedoesdev" + }, + { + "name": "Deepak Rajamohan", + "url": "https://github.com/deepakrkris" + }, + { + "name": "Dmitry Ashkadov", + "url": "https://github.com/dmitryash" + }, + { + "name": "Dongjin Na", + "url": "https://github.com/nadongguri" + }, + { + "name": "Doni Rubiagatra", + "url": "https://github.com/rubiagatra" + }, + { + "name": "Eric Bickle", + "url": "https://github.com/ebickle" + }, + { + "name": "extremeheat", + "url": "https://github.com/extremeheat" + }, + { + "name": "Feng Yu", + "url": "https://github.com/F3n67u" + }, + { + "name": "Ferdinand Holzer", + "url": "https://github.com/fholzer" + }, + { + "name": "Gabriel Schulhof", + "url": "https://github.com/gabrielschulhof" + }, + { + "name": "Guenter Sandner", + "url": "https://github.com/gms1" + }, + { + "name": "Gus Caplan", + "url": "https://github.com/devsnek" + }, + { + "name": "Helio Frota", + "url": "https://github.com/helio-frota" + }, + { + "name": "Hitesh Kanwathirtha", + "url": "https://github.com/digitalinfinity" + }, + { + "name": "ikokostya", + "url": "https://github.com/ikokostya" + }, + { + "name": "Jack Xia", + "url": "https://github.com/JckXia" + }, + { + "name": "Jake Barnes", + "url": "https://github.com/DuBistKomisch" + }, + { + "name": "Jake Yoon", + "url": "https://github.com/yjaeseok" + }, + { + "name": "Jason Ginchereau", + "url": "https://github.com/jasongin" + }, + { + "name": "Jenny", + "url": "https://github.com/egg-bread" + }, + { + "name": "Jeroen Janssen", + "url": "https://github.com/japj" + }, + { + "name": "Jim Schlight", + "url": "https://github.com/jschlight" + }, + { + "name": "Jinho Bang", + "url": "https://github.com/romandev" + }, + { + "name": "José Expósito", + "url": "https://github.com/JoseExposito" + }, + { + "name": "joshgarde", + "url": "https://github.com/joshgarde" + }, + { + "name": "Julian Mesa", + "url": "https://github.com/julianmesa-gitkraken" + }, + { + "name": "Kasumi Hanazuki", + "url": "https://github.com/hanazuki" + }, + { + "name": "Kelvin", + "url": "https://github.com/kelvinhammond" + }, + { + "name": "Kevin Eady", + "url": "https://github.com/KevinEady" + }, + { + "name": "Kévin VOYER", + "url": "https://github.com/kecsou" + }, + { + "name": "kidneysolo", + "url": "https://github.com/kidneysolo" + }, + { + "name": "Koki Nishihara", + "url": "https://github.com/Nishikoh" + }, + { + "name": "Konstantin Tarkus", + "url": "https://github.com/koistya" + }, + { + "name": "Kyle Farnung", + "url": "https://github.com/kfarnung" + }, + { + "name": "Kyle Kovacs", + "url": "https://github.com/nullromo" + }, + { + "name": "legendecas", + "url": "https://github.com/legendecas" + }, + { + "name": "LongYinan", + "url": "https://github.com/Brooooooklyn" + }, + { + "name": "Lovell Fuller", + "url": "https://github.com/lovell" + }, + { + "name": "Luciano Martorella", + "url": "https://github.com/lmartorella" + }, + { + "name": "mastergberry", + "url": "https://github.com/mastergberry" + }, + { + "name": "Mathias Küsel", + "url": "https://github.com/mathiask88" + }, + { + "name": "Matteo Collina", + "url": "https://github.com/mcollina" + }, + { + "name": "Michael Dawson", + "url": "https://github.com/mhdawson" + }, + { + "name": "Michael Price", + "url": "https://github.com/mikepricedev" + }, + { + "name": "Michele Campus", + "url": "https://github.com/kYroL01" + }, + { + "name": "Mikhail Cheshkov", + "url": "https://github.com/mcheshkov" + }, + { + "name": "nempoBu4", + "url": "https://github.com/nempoBu4" + }, + { + "name": "Nicola Del Gobbo", + "url": "https://github.com/NickNaso" + }, + { + "name": "Nick Soggin", + "url": "https://github.com/iSkore" + }, + { + "name": "Nikolai Vavilov", + "url": "https://github.com/seishun" + }, + { + "name": "Nurbol Alpysbayev", + "url": "https://github.com/anurbol" + }, + { + "name": "pacop", + "url": "https://github.com/pacop" + }, + { + "name": "Peter Šándor", + "url": "https://github.com/petersandor" + }, + { + "name": "Philipp Renoth", + "url": "https://github.com/DaAitch" + }, + { + "name": "rgerd", + "url": "https://github.com/rgerd" + }, + { + "name": "Richard Lau", + "url": "https://github.com/richardlau" + }, + { + "name": "Rolf Timmermans", + "url": "https://github.com/rolftimmermans" + }, + { + "name": "Ross Weir", + "url": "https://github.com/ross-weir" + }, + { + "name": "Ryuichi Okumura", + "url": "https://github.com/okuryu" + }, + { + "name": "Saint Gabriel", + "url": "https://github.com/chineduG" + }, + { + "name": "Sampson Gao", + "url": "https://github.com/sampsongao" + }, + { + "name": "Sam Roberts", + "url": "https://github.com/sam-github" + }, + { + "name": "strager", + "url": "https://github.com/strager" + }, + { + "name": "Taylor Woll", + "url": "https://github.com/boingoing" + }, + { + "name": "Thomas Gentilhomme", + "url": "https://github.com/fraxken" + }, + { + "name": "Tim Rach", + "url": "https://github.com/timrach" + }, + { + "name": "Tobias Nießen", + "url": "https://github.com/tniessen" + }, + { + "name": "todoroff", + "url": "https://github.com/todoroff" + }, + { + "name": "Tux3", + "url": "https://github.com/tux3" + }, + { + "name": "Vlad Velmisov", + "url": "https://github.com/Velmisov" + }, + { + "name": "Vladimir Morozov", + "url": "https://github.com/vmoroz" + + }, + { + "name": "WenheLI", + "url": "https://github.com/WenheLI" + }, + { + "name": "Xuguang Mei", + "url": "https://github.com/meixg" + }, + { + "name": "Yohei Kishimoto", + "url": "https://github.com/morokosi" + }, + { + "name": "Yulong Wang", + "url": "https://github.com/fs-eire" + }, + { + "name": "Ziqiu Zhao", + "url": "https://github.com/ZzqiZQute" + }, + { + "name": "Feng Yu", + "url": "https://github.com/F3n67u" + } + ], + "description": "Node.js API (Node-API)", + "devDependencies": { + "benchmark": "^2.1.4", + "bindings": "^1.5.0", + "clang-format": "^1.4.0", + "eslint": "^7.32.0", + "eslint-config-semistandard": "^16.0.0", + "eslint-config-standard": "^16.0.3", + "eslint-plugin-import": "^2.24.2", + "eslint-plugin-node": "^11.1.0", + "eslint-plugin-promise": "^5.1.0", + "fs-extra": "^9.0.1", + "path": "^0.12.7", + "pre-commit": "^1.2.2", + "safe-buffer": "^5.1.1" + }, + "directories": {}, + "gypfile": false, + "homepage": "https://github.com/nodejs/node-addon-api", + "keywords": [ + "n-api", + "napi", + "addon", + "native", + "bindings", + "c", + "c++", + "nan", + "node-addon-api" + ], + "license": "MIT", + "main": "index.js", + "name": "node-addon-api", + "readme": "README.md", + "repository": { + "type": "git", + "url": "git://github.com/nodejs/node-addon-api.git" + }, + "files": [ + "*.{c,h,gyp,gypi}", + "package-support.json", + "tools/" + ], + "scripts": { + "prebenchmark": "node-gyp rebuild -C benchmark", + "benchmark": "node benchmark", + "pretest": "node-gyp rebuild -C test", + "test": "node test", + "test:debug": "node-gyp rebuild -C test --debug && NODE_API_BUILD_CONFIG=Debug node ./test/index.js", + "predev": "node-gyp rebuild -C test --debug", + "dev": "node test", + "predev:incremental": "node-gyp configure build -C test --debug", + "dev:incremental": "node test", + "doc": "doxygen doc/Doxyfile", + "lint": "node tools/eslint-format && node tools/clang-format", + "lint:fix": "node tools/clang-format --fix && node tools/eslint-format --fix" + }, + "pre-commit": "lint", + "version": "5.1.0", + "support": true +} diff --git a/node_modules/node-addon-api/tools/README.md b/node_modules/node-addon-api/tools/README.md new file mode 100644 index 0000000..6b80e94 --- /dev/null +++ b/node_modules/node-addon-api/tools/README.md @@ -0,0 +1,73 @@ +# Tools + +## clang-format + +The clang-format checking tools is designed to check changed lines of code compared to given git-refs. + +## Migration Script + +The migration tool is designed to reduce repetitive work in the migration process. However, the script is not aiming to convert every thing for you. There are usually some small fixes and major reconstruction required. + +### How To Use + +To run the conversion script, first make sure you have the latest `node-addon-api` in your `node_modules` directory. +``` +npm install node-addon-api +``` + +Then run the script passing your project directory +``` +node ./node_modules/node-addon-api/tools/conversion.js ./ +``` + +After finish, recompile and debug things that are missed by the script. + + +### Quick Fixes +Here is the list of things that can be fixed easily. + 1. Change your methods' return value to void if it doesn't return value to JavaScript. + 2. Use `.` to access attribute or to invoke member function in Napi::Object instead of `->`. + 3. `Napi::New(env, value);` to `Napi::[Type]::New(env, value); + + +### Major Reconstructions +The implementation of `Napi::ObjectWrap` is significantly different from NAN's. `Napi::ObjectWrap` takes a pointer to the wrapped object and creates a reference to the wrapped object inside ObjectWrap constructor. `Napi::ObjectWrap` also associates wrapped object's instance methods to Javascript module instead of static methods like NAN. + +So if you use Nan::ObjectWrap in your module, you will need to execute the following steps. + + 1. Convert your [ClassName]::New function to a constructor function that takes a `Napi::CallbackInfo`. Declare it as +``` +[ClassName](const Napi::CallbackInfo& info); +``` +and define it as +``` +[ClassName]::[ClassName](const Napi::CallbackInfo& info) : Napi::ObjectWrap<[ClassName]>(info){ + ... +} +``` +This way, the `Napi::ObjectWrap` constructor will be invoked after the object has been instantiated and `Napi::ObjectWrap` can use the `this` pointer to create a reference to the wrapped object. + + 2. Move your original constructor code into the new constructor. Delete your original constructor. + 3. In your class initialization function, associate native methods in the following way. +``` +Napi::FunctionReference constructor; + +void [ClassName]::Init(Napi::Env env, Napi::Object exports, Napi::Object module) { + Napi::HandleScope scope(env); + Napi::Function ctor = DefineClass(env, "Canvas", { + InstanceMethod<&[ClassName]::Func1>("Func1"), + InstanceMethod<&[ClassName]::Func2>("Func2"), + InstanceAccessor<&[ClassName]::ValueGetter>("Value"), + StaticMethod<&[ClassName]::StaticMethod>("MethodName"), + InstanceValue("Value", Napi::[Type]::New(env, value)), + }); + + constructor = Napi::Persistent(ctor); + constructor .SuppressDestruct(); + exports.Set("[ClassName]", ctor); +} +``` + 4. In function where you need to Unwrap the ObjectWrap in NAN like `[ClassName]* native = Nan::ObjectWrap::Unwrap<[ClassName]>(info.This());`, use `this` pointer directly as the unwrapped object as each ObjectWrap instance is associated with a unique object instance. + + +If you still find issues after following this guide, please leave us an issue describing your problem and we will try to resolve it. diff --git a/node_modules/node-addon-api/tools/check-napi.js b/node_modules/node-addon-api/tools/check-napi.js new file mode 100644 index 0000000..9199af3 --- /dev/null +++ b/node_modules/node-addon-api/tools/check-napi.js @@ -0,0 +1,99 @@ +'use strict'; +// Descend into a directory structure and, for each file matching *.node, output +// based on the imports found in the file whether it's an N-API module or not. + +const fs = require('fs'); +const path = require('path'); + +// Read the output of the command, break it into lines, and use the reducer to +// decide whether the file is an N-API module or not. +function checkFile (file, command, argv, reducer) { + const child = require('child_process').spawn(command, argv, { + stdio: ['inherit', 'pipe', 'inherit'] + }); + let leftover = ''; + let isNapi; + child.stdout.on('data', (chunk) => { + if (isNapi === undefined) { + chunk = (leftover + chunk.toString()).split(/[\r\n]+/); + leftover = chunk.pop(); + isNapi = chunk.reduce(reducer, isNapi); + if (isNapi !== undefined) { + child.kill(); + } + } + }); + child.on('close', (code, signal) => { + if ((code === null && signal !== null) || (code !== 0)) { + console.log( + command + ' exited with code: ' + code + ' and signal: ' + signal); + } else { + // Green if it's a N-API module, red otherwise. + console.log( + '\x1b[' + (isNapi ? '42' : '41') + 'm' + + (isNapi ? ' N-API' : 'Not N-API') + + '\x1b[0m: ' + file); + } + }); +} + +// Use nm -a to list symbols. +function checkFileUNIX (file) { + checkFile(file, 'nm', ['-a', file], (soFar, line) => { + if (soFar === undefined) { + line = line.match(/([0-9a-f]*)? ([a-zA-Z]) (.*$)/); + if (line[2] === 'U') { + if (/^napi/.test(line[3])) { + soFar = true; + } + } + } + return soFar; + }); +} + +// Use dumpbin /imports to list symbols. +function checkFileWin32 (file) { + checkFile(file, 'dumpbin', ['/imports', file], (soFar, line) => { + if (soFar === undefined) { + line = line.match(/([0-9a-f]*)? +([a-zA-Z0-9]) (.*$)/); + if (line && /^napi/.test(line[line.length - 1])) { + soFar = true; + } + } + return soFar; + }); +} + +// Descend into a directory structure and pass each file ending in '.node' to +// one of the above checks, depending on the OS. +function recurse (top) { + fs.readdir(top, (error, items) => { + if (error) { + throw new Error('error reading directory ' + top + ': ' + error); + } + items.forEach((item) => { + item = path.join(top, item); + fs.stat(item, ((item) => (error, stats) => { + if (error) { + throw new Error('error about ' + item + ': ' + error); + } + if (stats.isDirectory()) { + recurse(item); + } else if (/[.]node$/.test(item) && + // Explicitly ignore files called 'nothing.node' because they are + // artefacts of node-addon-api having identified a version of + // Node.js that ships with a correct implementation of N-API. + path.basename(item) !== 'nothing.node') { + process.platform === 'win32' + ? checkFileWin32(item) + : checkFileUNIX(item); + } + })(item)); + }); + }); +} + +// Start with the directory given on the command line or the current directory +// if nothing was given. +recurse(process.argv.length > 3 ? process.argv[2] : '.'); diff --git a/node_modules/node-addon-api/tools/clang-format.js b/node_modules/node-addon-api/tools/clang-format.js new file mode 100644 index 0000000..e4bb4f5 --- /dev/null +++ b/node_modules/node-addon-api/tools/clang-format.js @@ -0,0 +1,71 @@ +#!/usr/bin/env node + +const spawn = require('child_process').spawnSync; +const path = require('path'); + +const filesToCheck = ['*.h', '*.cc']; +const FORMAT_START = process.env.FORMAT_START || 'main'; + +function main (args) { + let fix = false; + while (args.length > 0) { + switch (args[0]) { + case '-f': + case '--fix': + fix = true; + break; + default: + } + args.shift(); + } + + const clangFormatPath = path.dirname(require.resolve('clang-format')); + const binary = process.platform === 'win32' + ? 'node_modules\\.bin\\clang-format.cmd' + : 'node_modules/.bin/clang-format'; + const options = ['--binary=' + binary, '--style=file']; + if (fix) { + options.push(FORMAT_START); + } else { + options.push('--diff', FORMAT_START); + } + + const gitClangFormatPath = path.join(clangFormatPath, 'bin/git-clang-format'); + const result = spawn( + 'python', + [gitClangFormatPath, ...options, '--', ...filesToCheck], + { encoding: 'utf-8' } + ); + + if (result.stderr) { + console.error('Error running git-clang-format:', result.stderr); + return 2; + } + + const clangFormatOutput = result.stdout.trim(); + // Bail fast if in fix mode. + if (fix) { + console.log(clangFormatOutput); + return 0; + } + // Detect if there is any complains from clang-format + if ( + clangFormatOutput !== '' && + clangFormatOutput !== 'no modified files to format' && + clangFormatOutput !== 'clang-format did not modify any files' + ) { + console.error(clangFormatOutput); + const fixCmd = 'npm run lint:fix'; + console.error(` + ERROR: please run "${fixCmd}" to format changes in your commit + Note that when running the command locally, please keep your local + main branch and working branch up to date with nodejs/node-addon-api + to exclude un-related complains. + Or you can run "env FORMAT_START=upstream/main ${fixCmd}".`); + return 1; + } +} + +if (require.main === module) { + process.exitCode = main(process.argv.slice(2)); +} diff --git a/node_modules/node-addon-api/tools/conversion.js b/node_modules/node-addon-api/tools/conversion.js new file mode 100755 index 0000000..f89245a --- /dev/null +++ b/node_modules/node-addon-api/tools/conversion.js @@ -0,0 +1,301 @@ +#! /usr/bin/env node + +'use strict'; + +const fs = require('fs'); +const path = require('path'); + +const args = process.argv.slice(2); +const dir = args[0]; +if (!dir) { + console.log('Usage: node ' + path.basename(__filename) + ' '); + process.exit(1); +} + +const NodeApiVersion = require('../package.json').version; + +const disable = args[1]; +let ConfigFileOperations; +if (disable !== '--disable' && dir !== '--disable') { + ConfigFileOperations = { + 'package.json': [ + [/([ ]*)"dependencies": {/g, '$1"dependencies": {\n$1 "node-addon-api": "' + NodeApiVersion + '",'], + [/[ ]*"nan": *"[^"]+"(,|)[\n\r]/g, ''] + ], + 'binding.gyp': [ + [/([ ]*)'include_dirs': \[/g, '$1\'include_dirs\': [\n$1 \'\s+(\w+)\s*=\s*Nan::New\([\w\d:]+\);(?:\w+->Reset\(\1\))?\s+\1->SetClassName\(Nan::String::New\("(\w+)"\)\);/g, 'Napi::Function $1 = DefineClass(env, "$2", {'], + [/Local\s+(\w+)\s*=\s*Nan::New\([\w\d:]+\);\s+(\w+)\.Reset\((\1)\);\s+\1->SetClassName\((Nan::String::New|Nan::New<(v8::)*String>)\("(.+?)"\)\);/g, 'Napi::Function $1 = DefineClass(env, "$6", {'], + [/Local\s+(\w+)\s*=\s*Nan::New\([\w\d:]+\);(?:\w+->Reset\(\1\))?\s+\1->SetClassName\(Nan::String::New\("(\w+)"\)\);/g, 'Napi::Function $1 = DefineClass(env, "$2", {'], + [/Nan::New\(([\w\d:]+)\)->GetFunction\(\)/g, 'Napi::Function::New(env, $1)'], + [/Nan::New\(([\w\d:]+)\)->GetFunction()/g, 'Napi::Function::New(env, $1);'], + [/Nan::New\(([\w\d:]+)\)/g, 'Napi::Function::New(env, $1)'], + [/Nan::New\(([\w\d:]+)\)/g, 'Napi::Function::New(env, $1)'], + + // FunctionTemplate to FunctionReference + [/Nan::Persistent<(v8::)*FunctionTemplate>/g, 'Napi::FunctionReference'], + [/Nan::Persistent<(v8::)*Function>/g, 'Napi::FunctionReference'], + [/v8::Local/g, 'Napi::FunctionReference'], + [/Local/g, 'Napi::FunctionReference'], + [/v8::FunctionTemplate/g, 'Napi::FunctionReference'], + [/FunctionTemplate/g, 'Napi::FunctionReference'], + + [/([ ]*)Nan::SetPrototypeMethod\(\w+, "(\w+)", (\w+)\);/g, '$1InstanceMethod("$2", &$3),'], + [/([ ]*)(?:\w+\.Reset\(\w+\);\s+)?\(target\)\.Set\("(\w+)",\s*Nan::GetFunction\((\w+)\)\);/gm, + '});\n\n' + + '$1constructor = Napi::Persistent($3);\n' + + '$1constructor.SuppressDestruct();\n' + + '$1target.Set("$2", $3);'], + + // TODO: Other attribute combinations + [/static_cast\(ReadOnly\s*\|\s*DontDelete\)/gm, + 'static_cast(napi_enumerable | napi_configurable)'], + + [/([\w\d:<>]+?)::Cast\((.+?)\)/g, '$2.As<$1>()'], + + [/\*Nan::Utf8String\(([^)]+)\)/g, '$1->As().Utf8Value().c_str()'], + [/Nan::Utf8String +(\w+)\(([^)]+)\)/g, 'std::string $1 = $2.As()'], + [/Nan::Utf8String/g, 'std::string'], + + [/v8::String::Utf8Value (.+?)\((.+?)\)/g, 'Napi::String $1(env, $2)'], + [/String::Utf8Value (.+?)\((.+?)\)/g, 'Napi::String $1(env, $2)'], + [/\.length\(\)/g, '.Length()'], + + [/Nan::MakeCallback\(([^,]+),[\s\\]+([^,]+),/gm, '$2.MakeCallback($1,'], + + [/class\s+(\w+)\s*:\s*public\s+Nan::ObjectWrap/g, 'class $1 : public Napi::ObjectWrap<$1>'], + [/(\w+)\(([^)]*)\)\s*:\s*Nan::ObjectWrap\(\)\s*(,)?/gm, '$1($2) : Napi::ObjectWrap<$1>()$3'], + + // HandleOKCallback to OnOK + [/HandleOKCallback/g, 'OnOK'], + // HandleErrorCallback to OnError + [/HandleErrorCallback/g, 'OnError'], + + // ex. .As() to .As() + [/\.As\(\)/g, '.As()'], + [/\.As<(Value|Boolean|String|Number|Object|Array|Symbol|External|Function)>\(\)/g, '.As()'], + + // ex. Nan::New(info[0]) to Napi::Number::New(info[0]) + [/Nan::New<(v8::)*Integer>\((.+?)\)/g, 'Napi::Number::New(env, $2)'], + [/Nan::New\(([0-9.]+)\)/g, 'Napi::Number::New(env, $1)'], + [/Nan::New<(v8::)*String>\("(.+?)"\)/g, 'Napi::String::New(env, "$2")'], + [/Nan::New\("(.+?)"\)/g, 'Napi::String::New(env, "$1")'], + [/Nan::New<(v8::)*(.+?)>\(\)/g, 'Napi::$2::New(env)'], + [/Nan::New<(.+?)>\(\)/g, 'Napi::$1::New(env)'], + [/Nan::New<(v8::)*(.+?)>\(/g, 'Napi::$2::New(env, '], + [/Nan::New<(.+?)>\(/g, 'Napi::$1::New(env, '], + [/Nan::NewBuffer\(/g, 'Napi::Buffer::New(env, '], + // TODO: Properly handle this + [/Nan::New\(/g, 'Napi::New(env, '], + + [/\.IsInt32\(\)/g, '.IsNumber()'], + [/->IsInt32\(\)/g, '.IsNumber()'], + + [/(.+?)->BooleanValue\(\)/g, '$1.As().Value()'], + [/(.+?)->Int32Value\(\)/g, '$1.As().Int32Value()'], + [/(.+?)->Uint32Value\(\)/g, '$1.As().Uint32Value()'], + [/(.+?)->IntegerValue\(\)/g, '$1.As().Int64Value()'], + [/(.+?)->NumberValue\(\)/g, '$1.As().DoubleValue()'], + + // ex. Nan::To(info[0]) to info[0].Value() + [/Nan::To\((.+?)\)/g, '$2.To()'], + [/Nan::To<(Boolean|String|Number|Object|Array|Symbol|Function)>\((.+?)\)/g, '$2.To()'], + // ex. Nan::To(info[0]) to info[0].As().Value() + [/Nan::To\((.+?)\)/g, '$1.As().Value()'], + // ex. Nan::To(info[0]) to info[0].As().Int32Value() + [/Nan::To\((.+?)\)/g, '$1.As().Int32Value()'], + // ex. Nan::To(info[0]) to info[0].As().Int32Value() + [/Nan::To\((.+?)\)/g, '$1.As().Int32Value()'], + // ex. Nan::To(info[0]) to info[0].As().Uint32Value() + [/Nan::To\((.+?)\)/g, '$1.As().Uint32Value()'], + // ex. Nan::To(info[0]) to info[0].As().Int64Value() + [/Nan::To\((.+?)\)/g, '$1.As().Int64Value()'], + // ex. Nan::To(info[0]) to info[0].As().FloatValue() + [/Nan::To\((.+?)\)/g, '$1.As().FloatValue()'], + // ex. Nan::To(info[0]) to info[0].As().DoubleValue() + [/Nan::To\((.+?)\)/g, '$1.As().DoubleValue()'], + + [/Nan::New\((\w+)\)->HasInstance\((\w+)\)/g, '$2.InstanceOf($1.Value())'], + + [/Nan::Has\(([^,]+),\s*/gm, '($1).Has('], + [/\.Has\([\s|\\]*Nan::New<(v8::)*String>\(([^)]+)\)\)/gm, '.Has($1)'], + [/\.Has\([\s|\\]*Nan::New\(([^)]+)\)\)/gm, '.Has($1)'], + + [/Nan::Get\(([^,]+),\s*/gm, '($1).Get('], + [/\.Get\([\s|\\]*Nan::New<(v8::)*String>\(([^)]+)\)\)/gm, '.Get($1)'], + [/\.Get\([\s|\\]*Nan::New\(([^)]+)\)\)/gm, '.Get($1)'], + + [/Nan::Set\(([^,]+),\s*/gm, '($1).Set('], + [/\.Set\([\s|\\]*Nan::New<(v8::)*String>\(([^)]+)\)\s*,/gm, '.Set($1,'], + [/\.Set\([\s|\\]*Nan::New\(([^)]+)\)\s*,/gm, '.Set($1,'], + + // ex. node::Buffer::HasInstance(info[0]) to info[0].IsBuffer() + [/node::Buffer::HasInstance\((.+?)\)/g, '$1.IsBuffer()'], + // ex. node::Buffer::Length(info[0]) to info[0].Length() + [/node::Buffer::Length\((.+?)\)/g, '$1.As>().Length()'], + // ex. node::Buffer::Data(info[0]) to info[0].Data() + [/node::Buffer::Data\((.+?)\)/g, '$1.As>().Data()'], + [/Nan::CopyBuffer\(/g, 'Napi::Buffer::Copy(env, '], + + // Nan::AsyncQueueWorker(worker) + [/Nan::AsyncQueueWorker\((.+)\);/g, '$1.Queue();'], + [/Nan::(Undefined|Null|True|False)\(\)/g, 'env.$1()'], + + // Nan::ThrowError(error) to Napi::Error::New(env, error).ThrowAsJavaScriptException() + [/([ ]*)return Nan::Throw(\w*?)Error\((.+?)\);/g, '$1Napi::$2Error::New(env, $3).ThrowAsJavaScriptException();\n$1return env.Null();'], + [/Nan::Throw(\w*?)Error\((.+?)\);\n(\s*)return;/g, 'Napi::$1Error::New(env, $2).ThrowAsJavaScriptException();\n$3return env.Null();'], + [/Nan::Throw(\w*?)Error\((.+?)\);/g, 'Napi::$1Error::New(env, $2).ThrowAsJavaScriptException();\n'], + // Nan::RangeError(error) to Napi::RangeError::New(env, error) + [/Nan::(\w*?)Error\((.+)\)/g, 'Napi::$1Error::New(env, $2)'], + + [/Nan::Set\((.+?),\n* *(.+?),\n* *(.+?),\n* *(.+?)\)/g, '$1.Set($2, $3, $4)'], + + [/Nan::(Escapable)?HandleScope\s+(\w+)\s*;/g, 'Napi::$1HandleScope $2(env);'], + [/Nan::(Escapable)?HandleScope/g, 'Napi::$1HandleScope'], + [/Nan::ForceSet\(([^,]+), ?/g, '$1->DefineProperty('], + [/\.ForceSet\(Napi::String::New\(env, "(\w+)"\),\s*?/g, '.DefineProperty("$1", '], + // [ /Nan::GetPropertyNames\(([^,]+)\)/, '$1->GetPropertyNames()' ], + [/Nan::Equals\(([^,]+),/g, '$1.StrictEquals('], + + [/(.+)->Set\(/g, '$1.Set('], + + [/Nan::Callback/g, 'Napi::FunctionReference'], + + [/Nan::Persistent/g, 'Napi::ObjectReference'], + [/Nan::ADDON_REGISTER_FUNCTION_ARGS_TYPE target/g, 'Napi::Env& env, Napi::Object& target'], + + [/(\w+)\*\s+(\w+)\s*=\s*Nan::ObjectWrap::Unwrap<\w+>\(info\.This\(\)\);/g, '$1* $2 = this;'], + [/Nan::ObjectWrap::Unwrap<(\w+)>\((.*)\);/g, '$2.Unwrap<$1>();'], + + [/Nan::NAN_METHOD_RETURN_TYPE/g, 'void'], + [/NAN_INLINE/g, 'inline'], + + [/Nan::NAN_METHOD_ARGS_TYPE/g, 'const Napi::CallbackInfo&'], + [/NAN_METHOD\(([\w\d:]+?)\)/g, 'Napi::Value $1(const Napi::CallbackInfo& info)'], + [/static\s*NAN_GETTER\(([\w\d:]+?)\)/g, 'Napi::Value $1(const Napi::CallbackInfo& info)'], + [/NAN_GETTER\(([\w\d:]+?)\)/g, 'Napi::Value $1(const Napi::CallbackInfo& info)'], + [/static\s*NAN_SETTER\(([\w\d:]+?)\)/g, 'void $1(const Napi::CallbackInfo& info, const Napi::Value& value)'], + [/NAN_SETTER\(([\w\d:]+?)\)/g, 'void $1(const Napi::CallbackInfo& info, const Napi::Value& value)'], + [/void Init\((v8::)*Local<(v8::)*Object> exports\)/g, 'Napi::Object Init(Napi::Env env, Napi::Object exports)'], + [/NAN_MODULE_INIT\(([\w\d:]+?)\);/g, 'Napi::Object $1(Napi::Env env, Napi::Object exports);'], + [/NAN_MODULE_INIT\(([\w\d:]+?)\)/g, 'Napi::Object $1(Napi::Env env, Napi::Object exports)'], + + [/::(Init(?:ialize)?)\(target\)/g, '::$1(env, target, module)'], + [/constructor_template/g, 'constructor'], + + [/Nan::FunctionCallbackInfo<(v8::)?Value>[ ]*& [ ]*info\)[ ]*{\n*([ ]*)/gm, 'Napi::CallbackInfo& info) {\n$2Napi::Env env = info.Env();\n$2'], + [/Nan::FunctionCallbackInfo<(v8::)*Value>\s*&\s*info\);/g, 'Napi::CallbackInfo& info);'], + [/Nan::FunctionCallbackInfo<(v8::)*Value>\s*&/g, 'Napi::CallbackInfo&'], + + [/Buffer::HasInstance\(([^)]+)\)/g, '$1.IsBuffer()'], + + [/info\[(\d+)\]->/g, 'info[$1].'], + [/info\[([\w\d]+)\]->/g, 'info[$1].'], + [/info\.This\(\)->/g, 'info.This().'], + [/->Is(Object|String|Int32|Number)\(\)/g, '.Is$1()'], + [/info.GetReturnValue\(\).SetUndefined\(\)/g, 'return env.Undefined()'], + [/info\.GetReturnValue\(\)\.Set\(((\n|.)+?)\);/g, 'return $1;'], + + // ex. Local to Napi::Value + [/v8::Local/g, 'Napi::$1'], + [/Local<(Value|Boolean|String|Number|Object|Array|Symbol|External|Function)>/g, 'Napi::$1'], + + // Declare an env in helper functions that take a Napi::Value + [/(\w+)\(Napi::Value (\w+)(,\s*[^()]+)?\)\s*{\n*([ ]*)/gm, '$1(Napi::Value $2$3) {\n$4Napi::Env env = $2.Env();\n$4'], + + // delete #include and/or + [/#include +(<|")(?:node|nan).h("|>)/g, '#include $1napi.h$2\n#include $1uv.h$2'], + // NODE_MODULE to NODE_API_MODULE + [/NODE_MODULE/g, 'NODE_API_MODULE'], + [/Nan::/g, 'Napi::'], + [/nan.h/g, 'napi.h'], + + // delete .FromJust() + [/\.FromJust\(\)/g, ''], + // delete .ToLocalCheck() + [/\.ToLocalChecked\(\)/g, ''], + [/^.*->SetInternalFieldCount\(.*$/gm, ''], + + // replace using node; and/or using v8; to using Napi; + [/using (node|v8);/g, 'using Napi;'], + [/using namespace (node|Nan|v8);/g, 'using namespace Napi;'], + // delete using v8::Local; + [/using v8::Local;\n/g, ''], + // replace using v8::XXX; with using Napi::XXX + [/using v8::([A-Za-z]+);/g, 'using Napi::$1;'] + +]; + +const paths = listFiles(dir); +paths.forEach(function (dirEntry) { + const filename = dirEntry.split('\\').pop().split('/').pop(); + + // Check whether the file is a source file or a config file + // then execute function accordingly + const sourcePattern = /.+\.h|.+\.cc|.+\.cpp/; + if (sourcePattern.test(filename)) { + convertFile(dirEntry, SourceFileOperations); + } else if (ConfigFileOperations[filename] != null) { + convertFile(dirEntry, ConfigFileOperations[filename]); + } +}); + +function listFiles (dir, filelist) { + const files = fs.readdirSync(dir); + filelist = filelist || []; + files.forEach(function (file) { + if (file === 'node_modules') { + return; + } + + if (fs.statSync(path.join(dir, file)).isDirectory()) { + filelist = listFiles(path.join(dir, file), filelist); + } else { + filelist.push(path.join(dir, file)); + } + }); + return filelist; +} + +function convert (content, operations) { + for (let i = 0; i < operations.length; i++) { + const operation = operations[i]; + content = content.replace(operation[0], operation[1]); + } + return content; +} + +function convertFile (fileName, operations) { + fs.readFile(fileName, 'utf-8', function (err, file) { + if (err) throw err; + + file = convert(file, operations); + + fs.writeFile(fileName, file, function (err) { + if (err) throw err; + }); + }); +} diff --git a/node_modules/node-addon-api/tools/eslint-format.js b/node_modules/node-addon-api/tools/eslint-format.js new file mode 100644 index 0000000..1dda444 --- /dev/null +++ b/node_modules/node-addon-api/tools/eslint-format.js @@ -0,0 +1,79 @@ +#!/usr/bin/env node + +const spawn = require('child_process').spawnSync; + +const filesToCheck = '*.js'; +const FORMAT_START = process.env.FORMAT_START || 'main'; +const IS_WIN = process.platform === 'win32'; +const ESLINT_PATH = IS_WIN ? 'node_modules\\.bin\\eslint.cmd' : 'node_modules/.bin/eslint'; + +function main (args) { + let fix = false; + while (args.length > 0) { + switch (args[0]) { + case '-f': + case '--fix': + fix = true; + break; + default: + } + args.shift(); + } + + // Check js files that change on unstaged file + const fileUnStaged = spawn( + 'git', + ['diff', '--name-only', FORMAT_START, filesToCheck], + { + encoding: 'utf-8' + } + ); + + // Check js files that change on staged file + const fileStaged = spawn( + 'git', + ['diff', '--name-only', '--cached', FORMAT_START, filesToCheck], + { + encoding: 'utf-8' + } + ); + + const options = [ + ...fileStaged.stdout.split('\n').filter((f) => f !== ''), + ...fileUnStaged.stdout.split('\n').filter((f) => f !== '') + ]; + + if (fix) { + options.push('--fix'); + } + + const result = spawn(ESLINT_PATH, [...options], { + encoding: 'utf-8' + }); + + if (result.error && result.error.errno === 'ENOENT') { + console.error('Eslint not found! Eslint is supposed to be found at ', ESLINT_PATH); + return 2; + } + + if (result.status === 1) { + console.error('Eslint error:', result.stdout); + const fixCmd = 'npm run lint:fix'; + console.error(`ERROR: please run "${fixCmd}" to format changes in your commit + Note that when running the command locally, please keep your local + main branch and working branch up to date with nodejs/node-addon-api + to exclude un-related complains. + Or you can run "env FORMAT_START=upstream/main ${fixCmd}". + Also fix JS files by yourself if necessary.`); + return 1; + } + + if (result.stderr) { + console.error('Error running eslint:', result.stderr); + return 2; + } +} + +if (require.main === module) { + process.exitCode = main(process.argv.slice(2)); +} diff --git a/node_modules/node-gyp-build/LICENSE b/node_modules/node-gyp-build/LICENSE new file mode 100644 index 0000000..56fce08 --- /dev/null +++ b/node_modules/node-gyp-build/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2017 Mathias Buus + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/node_modules/node-gyp-build/README.md b/node_modules/node-gyp-build/README.md new file mode 100644 index 0000000..f712ca6 --- /dev/null +++ b/node_modules/node-gyp-build/README.md @@ -0,0 +1,58 @@ +# node-gyp-build + +> Build tool and bindings loader for [`node-gyp`][node-gyp] that supports prebuilds. + +``` +npm install node-gyp-build +``` + +[![Test](https://github.com/prebuild/node-gyp-build/actions/workflows/test.yml/badge.svg)](https://github.com/prebuild/node-gyp-build/actions/workflows/test.yml) + +Use together with [`prebuildify`][prebuildify] to easily support prebuilds for your native modules. + +## Usage + +> **Note.** Prebuild names have changed in [`prebuildify@3`][prebuildify] and `node-gyp-build@4`. Please see the documentation below. + +`node-gyp-build` works similar to [`node-gyp build`][node-gyp] except that it will check if a build or prebuild is present before rebuilding your project. + +It's main intended use is as an npm install script and bindings loader for native modules that bundle prebuilds using [`prebuildify`][prebuildify]. + +First add `node-gyp-build` as an install script to your native project + +``` js +{ + ... + "scripts": { + "install": "node-gyp-build" + } +} +``` + +Then in your `index.js`, instead of using the [`bindings`](https://www.npmjs.com/package/bindings) module use `node-gyp-build` to load your binding. + +``` js +var binding = require('node-gyp-build')(__dirname) +``` + +If you do these two things and bundle prebuilds with [`prebuildify`][prebuildify] your native module will work for most platforms +without having to compile on install time AND will work in both node and electron without the need to recompile between usage. + +Users can override `node-gyp-build` and force compiling by doing `npm install --build-from-source`. + +Prebuilds will be attempted loaded from `MODULE_PATH/prebuilds/...` and then next `EXEC_PATH/prebuilds/...` (the latter allowing use with `zeit/pkg`) + +## Supported prebuild names + +If so desired you can bundle more specific flavors, for example `musl` builds to support Alpine, or targeting a numbered ARM architecture version. + +These prebuilds can be bundled in addition to generic prebuilds; `node-gyp-build` will try to find the most specific flavor first. Prebuild filenames are composed of _tags_. The runtime tag takes precedence, as does an `abi` tag over `napi`. For more details on tags, please see [`prebuildify`][prebuildify]. + +Values for the `libc` and `armv` tags are auto-detected but can be overridden through the `LIBC` and `ARM_VERSION` environment variables, respectively. + +## License + +MIT + +[prebuildify]: https://github.com/prebuild/prebuildify +[node-gyp]: https://www.npmjs.com/package/node-gyp diff --git a/node_modules/node-gyp-build/bin.js b/node_modules/node-gyp-build/bin.js new file mode 100755 index 0000000..3fbcdf0 --- /dev/null +++ b/node_modules/node-gyp-build/bin.js @@ -0,0 +1,78 @@ +#!/usr/bin/env node + +var proc = require('child_process') +var os = require('os') +var path = require('path') + +if (!buildFromSource()) { + proc.exec('node-gyp-build-test', function (err, stdout, stderr) { + if (err) { + if (verbose()) console.error(stderr) + preinstall() + } + }) +} else { + preinstall() +} + +function build () { + var args = [os.platform() === 'win32' ? 'node-gyp.cmd' : 'node-gyp', 'rebuild'] + + try { + var pkg = require('node-gyp/package.json') + args = [ + process.execPath, + path.join(require.resolve('node-gyp/package.json'), '..', typeof pkg.bin === 'string' ? pkg.bin : pkg.bin['node-gyp']), + 'rebuild' + ] + } catch (_) {} + + proc.spawn(args[0], args.slice(1), { stdio: 'inherit' }).on('exit', function (code) { + if (code || !process.argv[3]) process.exit(code) + exec(process.argv[3]).on('exit', function (code) { + process.exit(code) + }) + }) +} + +function preinstall () { + if (!process.argv[2]) return build() + exec(process.argv[2]).on('exit', function (code) { + if (code) process.exit(code) + build() + }) +} + +function exec (cmd) { + if (process.platform !== 'win32') { + var shell = os.platform() === 'android' ? 'sh' : '/bin/sh' + return proc.spawn(shell, ['-c', '--', cmd], { + stdio: 'inherit' + }) + } + + return proc.spawn(process.env.comspec || 'cmd.exe', ['/s', '/c', '"' + cmd + '"'], { + windowsVerbatimArguments: true, + stdio: 'inherit' + }) +} + +function buildFromSource () { + return hasFlag('--build-from-source') || process.env.npm_config_build_from_source === 'true' +} + +function verbose () { + return hasFlag('--verbose') || process.env.npm_config_loglevel === 'verbose' +} + +// TODO (next major): remove in favor of env.npm_config_* which works since npm +// 0.1.8 while npm_config_argv will stop working in npm 7. See npm/rfcs#90 +function hasFlag (flag) { + if (!process.env.npm_config_argv) return false + + try { + return JSON.parse(process.env.npm_config_argv).original.indexOf(flag) !== -1 + } catch (_) { + return false + } +} diff --git a/node_modules/node-gyp-build/build-test.js b/node_modules/node-gyp-build/build-test.js new file mode 100755 index 0000000..b6622a5 --- /dev/null +++ b/node_modules/node-gyp-build/build-test.js @@ -0,0 +1,19 @@ +#!/usr/bin/env node + +process.env.NODE_ENV = 'test' + +var path = require('path') +var test = null + +try { + var pkg = require(path.join(process.cwd(), 'package.json')) + if (pkg.name && process.env[pkg.name.toUpperCase().replace(/-/g, '_')]) { + process.exit(0) + } + test = pkg.prebuild.test +} catch (err) { + // do nothing +} + +if (test) require(path.join(process.cwd(), test)) +else require('./')() diff --git a/node_modules/node-gyp-build/index.js b/node_modules/node-gyp-build/index.js new file mode 100644 index 0000000..07eb14f --- /dev/null +++ b/node_modules/node-gyp-build/index.js @@ -0,0 +1,6 @@ +const runtimeRequire = typeof __webpack_require__ === 'function' ? __non_webpack_require__ : require // eslint-disable-line +if (typeof runtimeRequire.addon === 'function') { // if the platform supports native resolving prefer that + module.exports = runtimeRequire.addon.bind(runtimeRequire) +} else { // else use the runtime version here + module.exports = require('./node-gyp-build.js') +} diff --git a/node_modules/node-gyp-build/node-gyp-build.js b/node_modules/node-gyp-build/node-gyp-build.js new file mode 100644 index 0000000..76b96e1 --- /dev/null +++ b/node_modules/node-gyp-build/node-gyp-build.js @@ -0,0 +1,207 @@ +var fs = require('fs') +var path = require('path') +var os = require('os') + +// Workaround to fix webpack's build warnings: 'the request of a dependency is an expression' +var runtimeRequire = typeof __webpack_require__ === 'function' ? __non_webpack_require__ : require // eslint-disable-line + +var vars = (process.config && process.config.variables) || {} +var prebuildsOnly = !!process.env.PREBUILDS_ONLY +var abi = process.versions.modules // TODO: support old node where this is undef +var runtime = isElectron() ? 'electron' : (isNwjs() ? 'node-webkit' : 'node') + +var arch = process.env.npm_config_arch || os.arch() +var platform = process.env.npm_config_platform || os.platform() +var libc = process.env.LIBC || (isAlpine(platform) ? 'musl' : 'glibc') +var armv = process.env.ARM_VERSION || (arch === 'arm64' ? '8' : vars.arm_version) || '' +var uv = (process.versions.uv || '').split('.')[0] + +module.exports = load + +function load (dir) { + return runtimeRequire(load.resolve(dir)) +} + +load.resolve = load.path = function (dir) { + dir = path.resolve(dir || '.') + + try { + var name = runtimeRequire(path.join(dir, 'package.json')).name.toUpperCase().replace(/-/g, '_') + if (process.env[name + '_PREBUILD']) dir = process.env[name + '_PREBUILD'] + } catch (err) {} + + if (!prebuildsOnly) { + var release = getFirst(path.join(dir, 'build/Release'), matchBuild) + if (release) return release + + var debug = getFirst(path.join(dir, 'build/Debug'), matchBuild) + if (debug) return debug + } + + var prebuild = resolve(dir) + if (prebuild) return prebuild + + var nearby = resolve(path.dirname(process.execPath)) + if (nearby) return nearby + + var target = [ + 'platform=' + platform, + 'arch=' + arch, + 'runtime=' + runtime, + 'abi=' + abi, + 'uv=' + uv, + armv ? 'armv=' + armv : '', + 'libc=' + libc, + 'node=' + process.versions.node, + process.versions.electron ? 'electron=' + process.versions.electron : '', + typeof __webpack_require__ === 'function' ? 'webpack=true' : '' // eslint-disable-line + ].filter(Boolean).join(' ') + + throw new Error('No native build was found for ' + target + '\n loaded from: ' + dir + '\n') + + function resolve (dir) { + // Find matching "prebuilds/-" directory + var tuples = readdirSync(path.join(dir, 'prebuilds')).map(parseTuple) + var tuple = tuples.filter(matchTuple(platform, arch)).sort(compareTuples)[0] + if (!tuple) return + + // Find most specific flavor first + var prebuilds = path.join(dir, 'prebuilds', tuple.name) + var parsed = readdirSync(prebuilds).map(parseTags) + var candidates = parsed.filter(matchTags(runtime, abi)) + var winner = candidates.sort(compareTags(runtime))[0] + if (winner) return path.join(prebuilds, winner.file) + } +} + +function readdirSync (dir) { + try { + return fs.readdirSync(dir) + } catch (err) { + return [] + } +} + +function getFirst (dir, filter) { + var files = readdirSync(dir).filter(filter) + return files[0] && path.join(dir, files[0]) +} + +function matchBuild (name) { + return /\.node$/.test(name) +} + +function parseTuple (name) { + // Example: darwin-x64+arm64 + var arr = name.split('-') + if (arr.length !== 2) return + + var platform = arr[0] + var architectures = arr[1].split('+') + + if (!platform) return + if (!architectures.length) return + if (!architectures.every(Boolean)) return + + return { name, platform, architectures } +} + +function matchTuple (platform, arch) { + return function (tuple) { + if (tuple == null) return false + if (tuple.platform !== platform) return false + return tuple.architectures.includes(arch) + } +} + +function compareTuples (a, b) { + // Prefer single-arch prebuilds over multi-arch + return a.architectures.length - b.architectures.length +} + +function parseTags (file) { + var arr = file.split('.') + var extension = arr.pop() + var tags = { file: file, specificity: 0 } + + if (extension !== 'node') return + + for (var i = 0; i < arr.length; i++) { + var tag = arr[i] + + if (tag === 'node' || tag === 'electron' || tag === 'node-webkit') { + tags.runtime = tag + } else if (tag === 'napi') { + tags.napi = true + } else if (tag.slice(0, 3) === 'abi') { + tags.abi = tag.slice(3) + } else if (tag.slice(0, 2) === 'uv') { + tags.uv = tag.slice(2) + } else if (tag.slice(0, 4) === 'armv') { + tags.armv = tag.slice(4) + } else if (tag === 'glibc' || tag === 'musl') { + tags.libc = tag + } else { + continue + } + + tags.specificity++ + } + + return tags +} + +function matchTags (runtime, abi) { + return function (tags) { + if (tags == null) return false + if (tags.runtime && tags.runtime !== runtime && !runtimeAgnostic(tags)) return false + if (tags.abi && tags.abi !== abi && !tags.napi) return false + if (tags.uv && tags.uv !== uv) return false + if (tags.armv && tags.armv !== armv) return false + if (tags.libc && tags.libc !== libc) return false + + return true + } +} + +function runtimeAgnostic (tags) { + return tags.runtime === 'node' && tags.napi +} + +function compareTags (runtime) { + // Precedence: non-agnostic runtime, abi over napi, then by specificity. + return function (a, b) { + if (a.runtime !== b.runtime) { + return a.runtime === runtime ? -1 : 1 + } else if (a.abi !== b.abi) { + return a.abi ? -1 : 1 + } else if (a.specificity !== b.specificity) { + return a.specificity > b.specificity ? -1 : 1 + } else { + return 0 + } + } +} + +function isNwjs () { + return !!(process.versions && process.versions.nw) +} + +function isElectron () { + if (process.versions && process.versions.electron) return true + if (process.env.ELECTRON_RUN_AS_NODE) return true + return typeof window !== 'undefined' && window.process && window.process.type === 'renderer' +} + +function isAlpine (platform) { + return platform === 'linux' && fs.existsSync('/etc/alpine-release') +} + +// Exposed for unit tests +// TODO: move to lib +load.parseTags = parseTags +load.matchTags = matchTags +load.compareTags = compareTags +load.parseTuple = parseTuple +load.matchTuple = matchTuple +load.compareTuples = compareTuples diff --git a/node_modules/node-gyp-build/optional.js b/node_modules/node-gyp-build/optional.js new file mode 100755 index 0000000..8daa04a --- /dev/null +++ b/node_modules/node-gyp-build/optional.js @@ -0,0 +1,7 @@ +#!/usr/bin/env node + +/* +I am only useful as an install script to make node-gyp not compile for purely optional native deps +*/ + +process.exit(0) diff --git a/node_modules/node-gyp-build/package.json b/node_modules/node-gyp-build/package.json new file mode 100644 index 0000000..86319f3 --- /dev/null +++ b/node_modules/node-gyp-build/package.json @@ -0,0 +1,29 @@ +{ + "name": "node-gyp-build", + "version": "4.8.0", + "description": "Build tool and bindings loader for node-gyp that supports prebuilds", + "main": "index.js", + "devDependencies": { + "array-shuffle": "^1.0.1", + "standard": "^14.0.0", + "tape": "^5.0.0" + }, + "scripts": { + "test": "standard && node test" + }, + "bin": { + "node-gyp-build": "./bin.js", + "node-gyp-build-optional": "./optional.js", + "node-gyp-build-test": "./build-test.js" + }, + "repository": { + "type": "git", + "url": "https://github.com/prebuild/node-gyp-build.git" + }, + "author": "Mathias Buus (@mafintosh)", + "license": "MIT", + "bugs": { + "url": "https://github.com/prebuild/node-gyp-build/issues" + }, + "homepage": "https://github.com/prebuild/node-gyp-build" +} diff --git a/node_modules/readable-stream/CONTRIBUTING.md b/node_modules/readable-stream/CONTRIBUTING.md new file mode 100644 index 0000000..f478d58 --- /dev/null +++ b/node_modules/readable-stream/CONTRIBUTING.md @@ -0,0 +1,38 @@ +# Developer's Certificate of Origin 1.1 + +By making a contribution to this project, I certify that: + +* (a) The contribution was created in whole or in part by me and I + have the right to submit it under the open source license + indicated in the file; or + +* (b) The contribution is based upon previous work that, to the best + of my knowledge, is covered under an appropriate open source + license and I have the right under that license to submit that + work with modifications, whether created in whole or in part + by me, under the same open source license (unless I am + permitted to submit under a different license), as indicated + in the file; or + +* (c) The contribution was provided directly to me by some other + person who certified (a), (b) or (c) and I have not modified + it. + +* (d) I understand and agree that this project and the contribution + are public and that a record of the contribution (including all + personal information I submit with it, including my sign-off) is + maintained indefinitely and may be redistributed consistent with + this project or the open source license(s) involved. + +## Moderation Policy + +The [Node.js Moderation Policy] applies to this WG. + +## Code of Conduct + +The [Node.js Code of Conduct][] applies to this WG. + +[Node.js Code of Conduct]: +https://github.com/nodejs/node/blob/master/CODE_OF_CONDUCT.md +[Node.js Moderation Policy]: +https://github.com/nodejs/TSC/blob/master/Moderation-Policy.md diff --git a/node_modules/readable-stream/GOVERNANCE.md b/node_modules/readable-stream/GOVERNANCE.md new file mode 100644 index 0000000..16ffb93 --- /dev/null +++ b/node_modules/readable-stream/GOVERNANCE.md @@ -0,0 +1,136 @@ +### Streams Working Group + +The Node.js Streams is jointly governed by a Working Group +(WG) +that is responsible for high-level guidance of the project. + +The WG has final authority over this project including: + +* Technical direction +* Project governance and process (including this policy) +* Contribution policy +* GitHub repository hosting +* Conduct guidelines +* Maintaining the list of additional Collaborators + +For the current list of WG members, see the project +[README.md](./README.md#current-project-team-members). + +### Collaborators + +The readable-stream GitHub repository is +maintained by the WG and additional Collaborators who are added by the +WG on an ongoing basis. + +Individuals making significant and valuable contributions are made +Collaborators and given commit-access to the project. These +individuals are identified by the WG and their addition as +Collaborators is discussed during the WG meeting. + +_Note:_ If you make a significant contribution and are not considered +for commit-access log an issue or contact a WG member directly and it +will be brought up in the next WG meeting. + +Modifications of the contents of the readable-stream repository are +made on +a collaborative basis. Anybody with a GitHub account may propose a +modification via pull request and it will be considered by the project +Collaborators. All pull requests must be reviewed and accepted by a +Collaborator with sufficient expertise who is able to take full +responsibility for the change. In the case of pull requests proposed +by an existing Collaborator, an additional Collaborator is required +for sign-off. Consensus should be sought if additional Collaborators +participate and there is disagreement around a particular +modification. See _Consensus Seeking Process_ below for further detail +on the consensus model used for governance. + +Collaborators may opt to elevate significant or controversial +modifications, or modifications that have not found consensus to the +WG for discussion by assigning the ***WG-agenda*** tag to a pull +request or issue. The WG should serve as the final arbiter where +required. + +For the current list of Collaborators, see the project +[README.md](./README.md#members). + +### WG Membership + +WG seats are not time-limited. There is no fixed size of the WG. +However, the expected target is between 6 and 12, to ensure adequate +coverage of important areas of expertise, balanced with the ability to +make decisions efficiently. + +There is no specific set of requirements or qualifications for WG +membership beyond these rules. + +The WG may add additional members to the WG by unanimous consensus. + +A WG member may be removed from the WG by voluntary resignation, or by +unanimous consensus of all other WG members. + +Changes to WG membership should be posted in the agenda, and may be +suggested as any other agenda item (see "WG Meetings" below). + +If an addition or removal is proposed during a meeting, and the full +WG is not in attendance to participate, then the addition or removal +is added to the agenda for the subsequent meeting. This is to ensure +that all members are given the opportunity to participate in all +membership decisions. If a WG member is unable to attend a meeting +where a planned membership decision is being made, then their consent +is assumed. + +No more than 1/3 of the WG members may be affiliated with the same +employer. If removal or resignation of a WG member, or a change of +employment by a WG member, creates a situation where more than 1/3 of +the WG membership shares an employer, then the situation must be +immediately remedied by the resignation or removal of one or more WG +members affiliated with the over-represented employer(s). + +### WG Meetings + +The WG meets occasionally on a Google Hangout On Air. A designated moderator +approved by the WG runs the meeting. Each meeting should be +published to YouTube. + +Items are added to the WG agenda that are considered contentious or +are modifications of governance, contribution policy, WG membership, +or release process. + +The intention of the agenda is not to approve or review all patches; +that should happen continuously on GitHub and be handled by the larger +group of Collaborators. + +Any community member or contributor can ask that something be added to +the next meeting's agenda by logging a GitHub Issue. Any Collaborator, +WG member or the moderator can add the item to the agenda by adding +the ***WG-agenda*** tag to the issue. + +Prior to each WG meeting the moderator will share the Agenda with +members of the WG. WG members can add any items they like to the +agenda at the beginning of each meeting. The moderator and the WG +cannot veto or remove items. + +The WG may invite persons or representatives from certain projects to +participate in a non-voting capacity. + +The moderator is responsible for summarizing the discussion of each +agenda item and sends it as a pull request after the meeting. + +### Consensus Seeking Process + +The WG follows a +[Consensus +Seeking](http://en.wikipedia.org/wiki/Consensus-seeking_decision-making) +decision-making model. + +When an agenda item has appeared to reach a consensus the moderator +will ask "Does anyone object?" as a final call for dissent from the +consensus. + +If an agenda item cannot reach a consensus a WG member can call for +either a closing vote or a vote to table the issue to the next +meeting. The call for a vote must be seconded by a majority of the WG +or else the discussion will continue. Simple majority wins. + +Note that changes to WG membership require a majority consensus. See +"WG Membership" above. diff --git a/node_modules/readable-stream/LICENSE b/node_modules/readable-stream/LICENSE new file mode 100644 index 0000000..2873b3b --- /dev/null +++ b/node_modules/readable-stream/LICENSE @@ -0,0 +1,47 @@ +Node.js is licensed for use as follows: + +""" +Copyright Node.js contributors. All rights reserved. + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to +deal in the Software without restriction, including without limitation the +rights to use, copy, modify, merge, publish, distribute, sublicense, and/or +sell copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS +IN THE SOFTWARE. +""" + +This license applies to parts of Node.js originating from the +https://github.com/joyent/node repository: + +""" +Copyright Joyent, Inc. and other Node contributors. All rights reserved. +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to +deal in the Software without restriction, including without limitation the +rights to use, copy, modify, merge, publish, distribute, sublicense, and/or +sell copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS +IN THE SOFTWARE. +""" diff --git a/node_modules/readable-stream/README.md b/node_modules/readable-stream/README.md new file mode 100644 index 0000000..19117c1 --- /dev/null +++ b/node_modules/readable-stream/README.md @@ -0,0 +1,106 @@ +# readable-stream + +***Node.js core streams for userland*** [![Build Status](https://travis-ci.com/nodejs/readable-stream.svg?branch=master)](https://travis-ci.com/nodejs/readable-stream) + + +[![NPM](https://nodei.co/npm/readable-stream.png?downloads=true&downloadRank=true)](https://nodei.co/npm/readable-stream/) +[![NPM](https://nodei.co/npm-dl/readable-stream.png?&months=6&height=3)](https://nodei.co/npm/readable-stream/) + + +[![Sauce Test Status](https://saucelabs.com/browser-matrix/readabe-stream.svg)](https://saucelabs.com/u/readabe-stream) + +```bash +npm install --save readable-stream +``` + +This package is a mirror of the streams implementations in Node.js. + +Full documentation may be found on the [Node.js website](https://nodejs.org/dist/v10.18.1/docs/api/stream.html). + +If you want to guarantee a stable streams base, regardless of what version of +Node you, or the users of your libraries are using, use **readable-stream** *only* and avoid the *"stream"* module in Node-core, for background see [this blogpost](http://r.va.gg/2014/06/why-i-dont-use-nodes-core-stream-module.html). + +As of version 2.0.0 **readable-stream** uses semantic versioning. + +## Version 3.x.x + +v3.x.x of `readable-stream` is a cut from Node 10. This version supports Node 6, 8, and 10, as well as evergreen browsers, IE 11 and latest Safari. The breaking changes introduced by v3 are composed by the combined breaking changes in [Node v9](https://nodejs.org/en/blog/release/v9.0.0/) and [Node v10](https://nodejs.org/en/blog/release/v10.0.0/), as follows: + +1. Error codes: https://github.com/nodejs/node/pull/13310, + https://github.com/nodejs/node/pull/13291, + https://github.com/nodejs/node/pull/16589, + https://github.com/nodejs/node/pull/15042, + https://github.com/nodejs/node/pull/15665, + https://github.com/nodejs/readable-stream/pull/344 +2. 'readable' have precedence over flowing + https://github.com/nodejs/node/pull/18994 +3. make virtual methods errors consistent + https://github.com/nodejs/node/pull/18813 +4. updated streams error handling + https://github.com/nodejs/node/pull/18438 +5. writable.end should return this. + https://github.com/nodejs/node/pull/18780 +6. readable continues to read when push('') + https://github.com/nodejs/node/pull/18211 +7. add custom inspect to BufferList + https://github.com/nodejs/node/pull/17907 +8. always defer 'readable' with nextTick + https://github.com/nodejs/node/pull/17979 + +## Version 2.x.x +v2.x.x of `readable-stream` is a cut of the stream module from Node 8 (there have been no semver-major changes from Node 4 to 8). This version supports all Node.js versions from 0.8, as well as evergreen browsers and IE 10 & 11. + +### Big Thanks + +Cross-browser Testing Platform and Open Source <3 Provided by [Sauce Labs][sauce] + +# Usage + +You can swap your `require('stream')` with `require('readable-stream')` +without any changes, if you are just using one of the main classes and +functions. + +```js +const { + Readable, + Writable, + Transform, + Duplex, + pipeline, + finished +} = require('readable-stream') +```` + +Note that `require('stream')` will return `Stream`, while +`require('readable-stream')` will return `Readable`. We discourage using +whatever is exported directly, but rather use one of the properties as +shown in the example above. + +# Streams Working Group + +`readable-stream` is maintained by the Streams Working Group, which +oversees the development and maintenance of the Streams API within +Node.js. The responsibilities of the Streams Working Group include: + +* Addressing stream issues on the Node.js issue tracker. +* Authoring and editing stream documentation within the Node.js project. +* Reviewing changes to stream subclasses within the Node.js project. +* Redirecting changes to streams from the Node.js project to this + project. +* Assisting in the implementation of stream providers within Node.js. +* Recommending versions of `readable-stream` to be included in Node.js. +* Messaging about the future of streams to give the community advance + notice of changes. + + +## Team Members + +* **Calvin Metcalf** ([@calvinmetcalf](https://github.com/calvinmetcalf)) <calvin.metcalf@gmail.com> + - Release GPG key: F3EF5F62A87FC27A22E643F714CE4FF5015AA242 +* **Mathias Buus** ([@mafintosh](https://github.com/mafintosh)) <mathiasbuus@gmail.com> +* **Matteo Collina** ([@mcollina](https://github.com/mcollina)) <matteo.collina@gmail.com> + - Release GPG key: 3ABC01543F22DD2239285CDD818674489FBC127E +* **Irina Shestak** ([@lrlna](https://github.com/lrlna)) <shestak.irina@gmail.com> +* **Yoshua Wyuts** ([@yoshuawuyts](https://github.com/yoshuawuyts)) <yoshuawuyts@gmail.com> + +[sauce]: https://saucelabs.com diff --git a/node_modules/readable-stream/errors-browser.js b/node_modules/readable-stream/errors-browser.js new file mode 100644 index 0000000..fb8e73e --- /dev/null +++ b/node_modules/readable-stream/errors-browser.js @@ -0,0 +1,127 @@ +'use strict'; + +function _inheritsLoose(subClass, superClass) { subClass.prototype = Object.create(superClass.prototype); subClass.prototype.constructor = subClass; subClass.__proto__ = superClass; } + +var codes = {}; + +function createErrorType(code, message, Base) { + if (!Base) { + Base = Error; + } + + function getMessage(arg1, arg2, arg3) { + if (typeof message === 'string') { + return message; + } else { + return message(arg1, arg2, arg3); + } + } + + var NodeError = + /*#__PURE__*/ + function (_Base) { + _inheritsLoose(NodeError, _Base); + + function NodeError(arg1, arg2, arg3) { + return _Base.call(this, getMessage(arg1, arg2, arg3)) || this; + } + + return NodeError; + }(Base); + + NodeError.prototype.name = Base.name; + NodeError.prototype.code = code; + codes[code] = NodeError; +} // https://github.com/nodejs/node/blob/v10.8.0/lib/internal/errors.js + + +function oneOf(expected, thing) { + if (Array.isArray(expected)) { + var len = expected.length; + expected = expected.map(function (i) { + return String(i); + }); + + if (len > 2) { + return "one of ".concat(thing, " ").concat(expected.slice(0, len - 1).join(', '), ", or ") + expected[len - 1]; + } else if (len === 2) { + return "one of ".concat(thing, " ").concat(expected[0], " or ").concat(expected[1]); + } else { + return "of ".concat(thing, " ").concat(expected[0]); + } + } else { + return "of ".concat(thing, " ").concat(String(expected)); + } +} // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/startsWith + + +function startsWith(str, search, pos) { + return str.substr(!pos || pos < 0 ? 0 : +pos, search.length) === search; +} // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/endsWith + + +function endsWith(str, search, this_len) { + if (this_len === undefined || this_len > str.length) { + this_len = str.length; + } + + return str.substring(this_len - search.length, this_len) === search; +} // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/includes + + +function includes(str, search, start) { + if (typeof start !== 'number') { + start = 0; + } + + if (start + search.length > str.length) { + return false; + } else { + return str.indexOf(search, start) !== -1; + } +} + +createErrorType('ERR_INVALID_OPT_VALUE', function (name, value) { + return 'The value "' + value + '" is invalid for option "' + name + '"'; +}, TypeError); +createErrorType('ERR_INVALID_ARG_TYPE', function (name, expected, actual) { + // determiner: 'must be' or 'must not be' + var determiner; + + if (typeof expected === 'string' && startsWith(expected, 'not ')) { + determiner = 'must not be'; + expected = expected.replace(/^not /, ''); + } else { + determiner = 'must be'; + } + + var msg; + + if (endsWith(name, ' argument')) { + // For cases like 'first argument' + msg = "The ".concat(name, " ").concat(determiner, " ").concat(oneOf(expected, 'type')); + } else { + var type = includes(name, '.') ? 'property' : 'argument'; + msg = "The \"".concat(name, "\" ").concat(type, " ").concat(determiner, " ").concat(oneOf(expected, 'type')); + } + + msg += ". Received type ".concat(typeof actual); + return msg; +}, TypeError); +createErrorType('ERR_STREAM_PUSH_AFTER_EOF', 'stream.push() after EOF'); +createErrorType('ERR_METHOD_NOT_IMPLEMENTED', function (name) { + return 'The ' + name + ' method is not implemented'; +}); +createErrorType('ERR_STREAM_PREMATURE_CLOSE', 'Premature close'); +createErrorType('ERR_STREAM_DESTROYED', function (name) { + return 'Cannot call ' + name + ' after a stream was destroyed'; +}); +createErrorType('ERR_MULTIPLE_CALLBACK', 'Callback called multiple times'); +createErrorType('ERR_STREAM_CANNOT_PIPE', 'Cannot pipe, not readable'); +createErrorType('ERR_STREAM_WRITE_AFTER_END', 'write after end'); +createErrorType('ERR_STREAM_NULL_VALUES', 'May not write null values to stream', TypeError); +createErrorType('ERR_UNKNOWN_ENCODING', function (arg) { + return 'Unknown encoding: ' + arg; +}, TypeError); +createErrorType('ERR_STREAM_UNSHIFT_AFTER_END_EVENT', 'stream.unshift() after end event'); +module.exports.codes = codes; diff --git a/node_modules/readable-stream/errors.js b/node_modules/readable-stream/errors.js new file mode 100644 index 0000000..8471526 --- /dev/null +++ b/node_modules/readable-stream/errors.js @@ -0,0 +1,116 @@ +'use strict'; + +const codes = {}; + +function createErrorType(code, message, Base) { + if (!Base) { + Base = Error + } + + function getMessage (arg1, arg2, arg3) { + if (typeof message === 'string') { + return message + } else { + return message(arg1, arg2, arg3) + } + } + + class NodeError extends Base { + constructor (arg1, arg2, arg3) { + super(getMessage(arg1, arg2, arg3)); + } + } + + NodeError.prototype.name = Base.name; + NodeError.prototype.code = code; + + codes[code] = NodeError; +} + +// https://github.com/nodejs/node/blob/v10.8.0/lib/internal/errors.js +function oneOf(expected, thing) { + if (Array.isArray(expected)) { + const len = expected.length; + expected = expected.map((i) => String(i)); + if (len > 2) { + return `one of ${thing} ${expected.slice(0, len - 1).join(', ')}, or ` + + expected[len - 1]; + } else if (len === 2) { + return `one of ${thing} ${expected[0]} or ${expected[1]}`; + } else { + return `of ${thing} ${expected[0]}`; + } + } else { + return `of ${thing} ${String(expected)}`; + } +} + +// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/startsWith +function startsWith(str, search, pos) { + return str.substr(!pos || pos < 0 ? 0 : +pos, search.length) === search; +} + +// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/endsWith +function endsWith(str, search, this_len) { + if (this_len === undefined || this_len > str.length) { + this_len = str.length; + } + return str.substring(this_len - search.length, this_len) === search; +} + +// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/includes +function includes(str, search, start) { + if (typeof start !== 'number') { + start = 0; + } + + if (start + search.length > str.length) { + return false; + } else { + return str.indexOf(search, start) !== -1; + } +} + +createErrorType('ERR_INVALID_OPT_VALUE', function (name, value) { + return 'The value "' + value + '" is invalid for option "' + name + '"' +}, TypeError); +createErrorType('ERR_INVALID_ARG_TYPE', function (name, expected, actual) { + // determiner: 'must be' or 'must not be' + let determiner; + if (typeof expected === 'string' && startsWith(expected, 'not ')) { + determiner = 'must not be'; + expected = expected.replace(/^not /, ''); + } else { + determiner = 'must be'; + } + + let msg; + if (endsWith(name, ' argument')) { + // For cases like 'first argument' + msg = `The ${name} ${determiner} ${oneOf(expected, 'type')}`; + } else { + const type = includes(name, '.') ? 'property' : 'argument'; + msg = `The "${name}" ${type} ${determiner} ${oneOf(expected, 'type')}`; + } + + msg += `. Received type ${typeof actual}`; + return msg; +}, TypeError); +createErrorType('ERR_STREAM_PUSH_AFTER_EOF', 'stream.push() after EOF'); +createErrorType('ERR_METHOD_NOT_IMPLEMENTED', function (name) { + return 'The ' + name + ' method is not implemented' +}); +createErrorType('ERR_STREAM_PREMATURE_CLOSE', 'Premature close'); +createErrorType('ERR_STREAM_DESTROYED', function (name) { + return 'Cannot call ' + name + ' after a stream was destroyed'; +}); +createErrorType('ERR_MULTIPLE_CALLBACK', 'Callback called multiple times'); +createErrorType('ERR_STREAM_CANNOT_PIPE', 'Cannot pipe, not readable'); +createErrorType('ERR_STREAM_WRITE_AFTER_END', 'write after end'); +createErrorType('ERR_STREAM_NULL_VALUES', 'May not write null values to stream', TypeError); +createErrorType('ERR_UNKNOWN_ENCODING', function (arg) { + return 'Unknown encoding: ' + arg +}, TypeError); +createErrorType('ERR_STREAM_UNSHIFT_AFTER_END_EVENT', 'stream.unshift() after end event'); + +module.exports.codes = codes; diff --git a/node_modules/readable-stream/experimentalWarning.js b/node_modules/readable-stream/experimentalWarning.js new file mode 100644 index 0000000..78e8414 --- /dev/null +++ b/node_modules/readable-stream/experimentalWarning.js @@ -0,0 +1,17 @@ +'use strict' + +var experimentalWarnings = new Set(); + +function emitExperimentalWarning(feature) { + if (experimentalWarnings.has(feature)) return; + var msg = feature + ' is an experimental feature. This feature could ' + + 'change at any time'; + experimentalWarnings.add(feature); + process.emitWarning(msg, 'ExperimentalWarning'); +} + +function noop() {} + +module.exports.emitExperimentalWarning = process.emitWarning + ? emitExperimentalWarning + : noop; diff --git a/node_modules/readable-stream/lib/_stream_duplex.js b/node_modules/readable-stream/lib/_stream_duplex.js new file mode 100644 index 0000000..19abfa6 --- /dev/null +++ b/node_modules/readable-stream/lib/_stream_duplex.js @@ -0,0 +1,126 @@ +// Copyright Joyent, Inc. and other Node contributors. +// +// Permission is hereby granted, free of charge, to any person obtaining a +// copy of this software and associated documentation files (the +// "Software"), to deal in the Software without restriction, including +// without limitation the rights to use, copy, modify, merge, publish, +// distribute, sublicense, and/or sell copies of the Software, and to permit +// persons to whom the Software is furnished to do so, subject to the +// following conditions: +// +// The above copyright notice and this permission notice shall be included +// in all copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN +// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, +// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR +// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE +// USE OR OTHER DEALINGS IN THE SOFTWARE. + +// a duplex stream is just a stream that is both readable and writable. +// Since JS doesn't have multiple prototypal inheritance, this class +// prototypally inherits from Readable, and then parasitically from +// Writable. + +'use strict'; + +/**/ +var objectKeys = Object.keys || function (obj) { + var keys = []; + for (var key in obj) keys.push(key); + return keys; +}; +/**/ + +module.exports = Duplex; +var Readable = require('./_stream_readable'); +var Writable = require('./_stream_writable'); +require('inherits')(Duplex, Readable); +{ + // Allow the keys array to be GC'ed. + var keys = objectKeys(Writable.prototype); + for (var v = 0; v < keys.length; v++) { + var method = keys[v]; + if (!Duplex.prototype[method]) Duplex.prototype[method] = Writable.prototype[method]; + } +} +function Duplex(options) { + if (!(this instanceof Duplex)) return new Duplex(options); + Readable.call(this, options); + Writable.call(this, options); + this.allowHalfOpen = true; + if (options) { + if (options.readable === false) this.readable = false; + if (options.writable === false) this.writable = false; + if (options.allowHalfOpen === false) { + this.allowHalfOpen = false; + this.once('end', onend); + } + } +} +Object.defineProperty(Duplex.prototype, 'writableHighWaterMark', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._writableState.highWaterMark; + } +}); +Object.defineProperty(Duplex.prototype, 'writableBuffer', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._writableState && this._writableState.getBuffer(); + } +}); +Object.defineProperty(Duplex.prototype, 'writableLength', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._writableState.length; + } +}); + +// the no-half-open enforcer +function onend() { + // If the writable side ended, then we're ok. + if (this._writableState.ended) return; + + // no more data can be written. + // But allow more writes to happen in this tick. + process.nextTick(onEndNT, this); +} +function onEndNT(self) { + self.end(); +} +Object.defineProperty(Duplex.prototype, 'destroyed', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + if (this._readableState === undefined || this._writableState === undefined) { + return false; + } + return this._readableState.destroyed && this._writableState.destroyed; + }, + set: function set(value) { + // we ignore the value if the stream + // has not been initialized yet + if (this._readableState === undefined || this._writableState === undefined) { + return; + } + + // backward compatibility, the user is explicitly + // managing destroyed + this._readableState.destroyed = value; + this._writableState.destroyed = value; + } +}); \ No newline at end of file diff --git a/node_modules/readable-stream/lib/_stream_passthrough.js b/node_modules/readable-stream/lib/_stream_passthrough.js new file mode 100644 index 0000000..24a6bdd --- /dev/null +++ b/node_modules/readable-stream/lib/_stream_passthrough.js @@ -0,0 +1,37 @@ +// Copyright Joyent, Inc. and other Node contributors. +// +// Permission is hereby granted, free of charge, to any person obtaining a +// copy of this software and associated documentation files (the +// "Software"), to deal in the Software without restriction, including +// without limitation the rights to use, copy, modify, merge, publish, +// distribute, sublicense, and/or sell copies of the Software, and to permit +// persons to whom the Software is furnished to do so, subject to the +// following conditions: +// +// The above copyright notice and this permission notice shall be included +// in all copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN +// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, +// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR +// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE +// USE OR OTHER DEALINGS IN THE SOFTWARE. + +// a passthrough stream. +// basically just the most minimal sort of Transform stream. +// Every written chunk gets output as-is. + +'use strict'; + +module.exports = PassThrough; +var Transform = require('./_stream_transform'); +require('inherits')(PassThrough, Transform); +function PassThrough(options) { + if (!(this instanceof PassThrough)) return new PassThrough(options); + Transform.call(this, options); +} +PassThrough.prototype._transform = function (chunk, encoding, cb) { + cb(null, chunk); +}; \ No newline at end of file diff --git a/node_modules/readable-stream/lib/_stream_readable.js b/node_modules/readable-stream/lib/_stream_readable.js new file mode 100644 index 0000000..df1f608 --- /dev/null +++ b/node_modules/readable-stream/lib/_stream_readable.js @@ -0,0 +1,1027 @@ +// Copyright Joyent, Inc. and other Node contributors. +// +// Permission is hereby granted, free of charge, to any person obtaining a +// copy of this software and associated documentation files (the +// "Software"), to deal in the Software without restriction, including +// without limitation the rights to use, copy, modify, merge, publish, +// distribute, sublicense, and/or sell copies of the Software, and to permit +// persons to whom the Software is furnished to do so, subject to the +// following conditions: +// +// The above copyright notice and this permission notice shall be included +// in all copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN +// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, +// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR +// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE +// USE OR OTHER DEALINGS IN THE SOFTWARE. + +'use strict'; + +module.exports = Readable; + +/**/ +var Duplex; +/**/ + +Readable.ReadableState = ReadableState; + +/**/ +var EE = require('events').EventEmitter; +var EElistenerCount = function EElistenerCount(emitter, type) { + return emitter.listeners(type).length; +}; +/**/ + +/**/ +var Stream = require('./internal/streams/stream'); +/**/ + +var Buffer = require('buffer').Buffer; +var OurUint8Array = (typeof global !== 'undefined' ? global : typeof window !== 'undefined' ? window : typeof self !== 'undefined' ? self : {}).Uint8Array || function () {}; +function _uint8ArrayToBuffer(chunk) { + return Buffer.from(chunk); +} +function _isUint8Array(obj) { + return Buffer.isBuffer(obj) || obj instanceof OurUint8Array; +} + +/**/ +var debugUtil = require('util'); +var debug; +if (debugUtil && debugUtil.debuglog) { + debug = debugUtil.debuglog('stream'); +} else { + debug = function debug() {}; +} +/**/ + +var BufferList = require('./internal/streams/buffer_list'); +var destroyImpl = require('./internal/streams/destroy'); +var _require = require('./internal/streams/state'), + getHighWaterMark = _require.getHighWaterMark; +var _require$codes = require('../errors').codes, + ERR_INVALID_ARG_TYPE = _require$codes.ERR_INVALID_ARG_TYPE, + ERR_STREAM_PUSH_AFTER_EOF = _require$codes.ERR_STREAM_PUSH_AFTER_EOF, + ERR_METHOD_NOT_IMPLEMENTED = _require$codes.ERR_METHOD_NOT_IMPLEMENTED, + ERR_STREAM_UNSHIFT_AFTER_END_EVENT = _require$codes.ERR_STREAM_UNSHIFT_AFTER_END_EVENT; + +// Lazy loaded to improve the startup performance. +var StringDecoder; +var createReadableStreamAsyncIterator; +var from; +require('inherits')(Readable, Stream); +var errorOrDestroy = destroyImpl.errorOrDestroy; +var kProxyEvents = ['error', 'close', 'destroy', 'pause', 'resume']; +function prependListener(emitter, event, fn) { + // Sadly this is not cacheable as some libraries bundle their own + // event emitter implementation with them. + if (typeof emitter.prependListener === 'function') return emitter.prependListener(event, fn); + + // This is a hack to make sure that our error handler is attached before any + // userland ones. NEVER DO THIS. This is here only because this code needs + // to continue to work with older versions of Node.js that do not include + // the prependListener() method. The goal is to eventually remove this hack. + if (!emitter._events || !emitter._events[event]) emitter.on(event, fn);else if (Array.isArray(emitter._events[event])) emitter._events[event].unshift(fn);else emitter._events[event] = [fn, emitter._events[event]]; +} +function ReadableState(options, stream, isDuplex) { + Duplex = Duplex || require('./_stream_duplex'); + options = options || {}; + + // Duplex streams are both readable and writable, but share + // the same options object. + // However, some cases require setting options to different + // values for the readable and the writable sides of the duplex stream. + // These options can be provided separately as readableXXX and writableXXX. + if (typeof isDuplex !== 'boolean') isDuplex = stream instanceof Duplex; + + // object stream flag. Used to make read(n) ignore n and to + // make all the buffer merging and length checks go away + this.objectMode = !!options.objectMode; + if (isDuplex) this.objectMode = this.objectMode || !!options.readableObjectMode; + + // the point at which it stops calling _read() to fill the buffer + // Note: 0 is a valid value, means "don't call _read preemptively ever" + this.highWaterMark = getHighWaterMark(this, options, 'readableHighWaterMark', isDuplex); + + // A linked list is used to store data chunks instead of an array because the + // linked list can remove elements from the beginning faster than + // array.shift() + this.buffer = new BufferList(); + this.length = 0; + this.pipes = null; + this.pipesCount = 0; + this.flowing = null; + this.ended = false; + this.endEmitted = false; + this.reading = false; + + // a flag to be able to tell if the event 'readable'/'data' is emitted + // immediately, or on a later tick. We set this to true at first, because + // any actions that shouldn't happen until "later" should generally also + // not happen before the first read call. + this.sync = true; + + // whenever we return null, then we set a flag to say + // that we're awaiting a 'readable' event emission. + this.needReadable = false; + this.emittedReadable = false; + this.readableListening = false; + this.resumeScheduled = false; + this.paused = true; + + // Should close be emitted on destroy. Defaults to true. + this.emitClose = options.emitClose !== false; + + // Should .destroy() be called after 'end' (and potentially 'finish') + this.autoDestroy = !!options.autoDestroy; + + // has it been destroyed + this.destroyed = false; + + // Crypto is kind of old and crusty. Historically, its default string + // encoding is 'binary' so we have to make this configurable. + // Everything else in the universe uses 'utf8', though. + this.defaultEncoding = options.defaultEncoding || 'utf8'; + + // the number of writers that are awaiting a drain event in .pipe()s + this.awaitDrain = 0; + + // if true, a maybeReadMore has been scheduled + this.readingMore = false; + this.decoder = null; + this.encoding = null; + if (options.encoding) { + if (!StringDecoder) StringDecoder = require('string_decoder/').StringDecoder; + this.decoder = new StringDecoder(options.encoding); + this.encoding = options.encoding; + } +} +function Readable(options) { + Duplex = Duplex || require('./_stream_duplex'); + if (!(this instanceof Readable)) return new Readable(options); + + // Checking for a Stream.Duplex instance is faster here instead of inside + // the ReadableState constructor, at least with V8 6.5 + var isDuplex = this instanceof Duplex; + this._readableState = new ReadableState(options, this, isDuplex); + + // legacy + this.readable = true; + if (options) { + if (typeof options.read === 'function') this._read = options.read; + if (typeof options.destroy === 'function') this._destroy = options.destroy; + } + Stream.call(this); +} +Object.defineProperty(Readable.prototype, 'destroyed', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + if (this._readableState === undefined) { + return false; + } + return this._readableState.destroyed; + }, + set: function set(value) { + // we ignore the value if the stream + // has not been initialized yet + if (!this._readableState) { + return; + } + + // backward compatibility, the user is explicitly + // managing destroyed + this._readableState.destroyed = value; + } +}); +Readable.prototype.destroy = destroyImpl.destroy; +Readable.prototype._undestroy = destroyImpl.undestroy; +Readable.prototype._destroy = function (err, cb) { + cb(err); +}; + +// Manually shove something into the read() buffer. +// This returns true if the highWaterMark has not been hit yet, +// similar to how Writable.write() returns true if you should +// write() some more. +Readable.prototype.push = function (chunk, encoding) { + var state = this._readableState; + var skipChunkCheck; + if (!state.objectMode) { + if (typeof chunk === 'string') { + encoding = encoding || state.defaultEncoding; + if (encoding !== state.encoding) { + chunk = Buffer.from(chunk, encoding); + encoding = ''; + } + skipChunkCheck = true; + } + } else { + skipChunkCheck = true; + } + return readableAddChunk(this, chunk, encoding, false, skipChunkCheck); +}; + +// Unshift should *always* be something directly out of read() +Readable.prototype.unshift = function (chunk) { + return readableAddChunk(this, chunk, null, true, false); +}; +function readableAddChunk(stream, chunk, encoding, addToFront, skipChunkCheck) { + debug('readableAddChunk', chunk); + var state = stream._readableState; + if (chunk === null) { + state.reading = false; + onEofChunk(stream, state); + } else { + var er; + if (!skipChunkCheck) er = chunkInvalid(state, chunk); + if (er) { + errorOrDestroy(stream, er); + } else if (state.objectMode || chunk && chunk.length > 0) { + if (typeof chunk !== 'string' && !state.objectMode && Object.getPrototypeOf(chunk) !== Buffer.prototype) { + chunk = _uint8ArrayToBuffer(chunk); + } + if (addToFront) { + if (state.endEmitted) errorOrDestroy(stream, new ERR_STREAM_UNSHIFT_AFTER_END_EVENT());else addChunk(stream, state, chunk, true); + } else if (state.ended) { + errorOrDestroy(stream, new ERR_STREAM_PUSH_AFTER_EOF()); + } else if (state.destroyed) { + return false; + } else { + state.reading = false; + if (state.decoder && !encoding) { + chunk = state.decoder.write(chunk); + if (state.objectMode || chunk.length !== 0) addChunk(stream, state, chunk, false);else maybeReadMore(stream, state); + } else { + addChunk(stream, state, chunk, false); + } + } + } else if (!addToFront) { + state.reading = false; + maybeReadMore(stream, state); + } + } + + // We can push more data if we are below the highWaterMark. + // Also, if we have no data yet, we can stand some more bytes. + // This is to work around cases where hwm=0, such as the repl. + return !state.ended && (state.length < state.highWaterMark || state.length === 0); +} +function addChunk(stream, state, chunk, addToFront) { + if (state.flowing && state.length === 0 && !state.sync) { + state.awaitDrain = 0; + stream.emit('data', chunk); + } else { + // update the buffer info. + state.length += state.objectMode ? 1 : chunk.length; + if (addToFront) state.buffer.unshift(chunk);else state.buffer.push(chunk); + if (state.needReadable) emitReadable(stream); + } + maybeReadMore(stream, state); +} +function chunkInvalid(state, chunk) { + var er; + if (!_isUint8Array(chunk) && typeof chunk !== 'string' && chunk !== undefined && !state.objectMode) { + er = new ERR_INVALID_ARG_TYPE('chunk', ['string', 'Buffer', 'Uint8Array'], chunk); + } + return er; +} +Readable.prototype.isPaused = function () { + return this._readableState.flowing === false; +}; + +// backwards compatibility. +Readable.prototype.setEncoding = function (enc) { + if (!StringDecoder) StringDecoder = require('string_decoder/').StringDecoder; + var decoder = new StringDecoder(enc); + this._readableState.decoder = decoder; + // If setEncoding(null), decoder.encoding equals utf8 + this._readableState.encoding = this._readableState.decoder.encoding; + + // Iterate over current buffer to convert already stored Buffers: + var p = this._readableState.buffer.head; + var content = ''; + while (p !== null) { + content += decoder.write(p.data); + p = p.next; + } + this._readableState.buffer.clear(); + if (content !== '') this._readableState.buffer.push(content); + this._readableState.length = content.length; + return this; +}; + +// Don't raise the hwm > 1GB +var MAX_HWM = 0x40000000; +function computeNewHighWaterMark(n) { + if (n >= MAX_HWM) { + // TODO(ronag): Throw ERR_VALUE_OUT_OF_RANGE. + n = MAX_HWM; + } else { + // Get the next highest power of 2 to prevent increasing hwm excessively in + // tiny amounts + n--; + n |= n >>> 1; + n |= n >>> 2; + n |= n >>> 4; + n |= n >>> 8; + n |= n >>> 16; + n++; + } + return n; +} + +// This function is designed to be inlinable, so please take care when making +// changes to the function body. +function howMuchToRead(n, state) { + if (n <= 0 || state.length === 0 && state.ended) return 0; + if (state.objectMode) return 1; + if (n !== n) { + // Only flow one buffer at a time + if (state.flowing && state.length) return state.buffer.head.data.length;else return state.length; + } + // If we're asking for more than the current hwm, then raise the hwm. + if (n > state.highWaterMark) state.highWaterMark = computeNewHighWaterMark(n); + if (n <= state.length) return n; + // Don't have enough + if (!state.ended) { + state.needReadable = true; + return 0; + } + return state.length; +} + +// you can override either this method, or the async _read(n) below. +Readable.prototype.read = function (n) { + debug('read', n); + n = parseInt(n, 10); + var state = this._readableState; + var nOrig = n; + if (n !== 0) state.emittedReadable = false; + + // if we're doing read(0) to trigger a readable event, but we + // already have a bunch of data in the buffer, then just trigger + // the 'readable' event and move on. + if (n === 0 && state.needReadable && ((state.highWaterMark !== 0 ? state.length >= state.highWaterMark : state.length > 0) || state.ended)) { + debug('read: emitReadable', state.length, state.ended); + if (state.length === 0 && state.ended) endReadable(this);else emitReadable(this); + return null; + } + n = howMuchToRead(n, state); + + // if we've ended, and we're now clear, then finish it up. + if (n === 0 && state.ended) { + if (state.length === 0) endReadable(this); + return null; + } + + // All the actual chunk generation logic needs to be + // *below* the call to _read. The reason is that in certain + // synthetic stream cases, such as passthrough streams, _read + // may be a completely synchronous operation which may change + // the state of the read buffer, providing enough data when + // before there was *not* enough. + // + // So, the steps are: + // 1. Figure out what the state of things will be after we do + // a read from the buffer. + // + // 2. If that resulting state will trigger a _read, then call _read. + // Note that this may be asynchronous, or synchronous. Yes, it is + // deeply ugly to write APIs this way, but that still doesn't mean + // that the Readable class should behave improperly, as streams are + // designed to be sync/async agnostic. + // Take note if the _read call is sync or async (ie, if the read call + // has returned yet), so that we know whether or not it's safe to emit + // 'readable' etc. + // + // 3. Actually pull the requested chunks out of the buffer and return. + + // if we need a readable event, then we need to do some reading. + var doRead = state.needReadable; + debug('need readable', doRead); + + // if we currently have less than the highWaterMark, then also read some + if (state.length === 0 || state.length - n < state.highWaterMark) { + doRead = true; + debug('length less than watermark', doRead); + } + + // however, if we've ended, then there's no point, and if we're already + // reading, then it's unnecessary. + if (state.ended || state.reading) { + doRead = false; + debug('reading or ended', doRead); + } else if (doRead) { + debug('do read'); + state.reading = true; + state.sync = true; + // if the length is currently zero, then we *need* a readable event. + if (state.length === 0) state.needReadable = true; + // call internal read method + this._read(state.highWaterMark); + state.sync = false; + // If _read pushed data synchronously, then `reading` will be false, + // and we need to re-evaluate how much data we can return to the user. + if (!state.reading) n = howMuchToRead(nOrig, state); + } + var ret; + if (n > 0) ret = fromList(n, state);else ret = null; + if (ret === null) { + state.needReadable = state.length <= state.highWaterMark; + n = 0; + } else { + state.length -= n; + state.awaitDrain = 0; + } + if (state.length === 0) { + // If we have nothing in the buffer, then we want to know + // as soon as we *do* get something into the buffer. + if (!state.ended) state.needReadable = true; + + // If we tried to read() past the EOF, then emit end on the next tick. + if (nOrig !== n && state.ended) endReadable(this); + } + if (ret !== null) this.emit('data', ret); + return ret; +}; +function onEofChunk(stream, state) { + debug('onEofChunk'); + if (state.ended) return; + if (state.decoder) { + var chunk = state.decoder.end(); + if (chunk && chunk.length) { + state.buffer.push(chunk); + state.length += state.objectMode ? 1 : chunk.length; + } + } + state.ended = true; + if (state.sync) { + // if we are sync, wait until next tick to emit the data. + // Otherwise we risk emitting data in the flow() + // the readable code triggers during a read() call + emitReadable(stream); + } else { + // emit 'readable' now to make sure it gets picked up. + state.needReadable = false; + if (!state.emittedReadable) { + state.emittedReadable = true; + emitReadable_(stream); + } + } +} + +// Don't emit readable right away in sync mode, because this can trigger +// another read() call => stack overflow. This way, it might trigger +// a nextTick recursion warning, but that's not so bad. +function emitReadable(stream) { + var state = stream._readableState; + debug('emitReadable', state.needReadable, state.emittedReadable); + state.needReadable = false; + if (!state.emittedReadable) { + debug('emitReadable', state.flowing); + state.emittedReadable = true; + process.nextTick(emitReadable_, stream); + } +} +function emitReadable_(stream) { + var state = stream._readableState; + debug('emitReadable_', state.destroyed, state.length, state.ended); + if (!state.destroyed && (state.length || state.ended)) { + stream.emit('readable'); + state.emittedReadable = false; + } + + // The stream needs another readable event if + // 1. It is not flowing, as the flow mechanism will take + // care of it. + // 2. It is not ended. + // 3. It is below the highWaterMark, so we can schedule + // another readable later. + state.needReadable = !state.flowing && !state.ended && state.length <= state.highWaterMark; + flow(stream); +} + +// at this point, the user has presumably seen the 'readable' event, +// and called read() to consume some data. that may have triggered +// in turn another _read(n) call, in which case reading = true if +// it's in progress. +// However, if we're not ended, or reading, and the length < hwm, +// then go ahead and try to read some more preemptively. +function maybeReadMore(stream, state) { + if (!state.readingMore) { + state.readingMore = true; + process.nextTick(maybeReadMore_, stream, state); + } +} +function maybeReadMore_(stream, state) { + // Attempt to read more data if we should. + // + // The conditions for reading more data are (one of): + // - Not enough data buffered (state.length < state.highWaterMark). The loop + // is responsible for filling the buffer with enough data if such data + // is available. If highWaterMark is 0 and we are not in the flowing mode + // we should _not_ attempt to buffer any extra data. We'll get more data + // when the stream consumer calls read() instead. + // - No data in the buffer, and the stream is in flowing mode. In this mode + // the loop below is responsible for ensuring read() is called. Failing to + // call read here would abort the flow and there's no other mechanism for + // continuing the flow if the stream consumer has just subscribed to the + // 'data' event. + // + // In addition to the above conditions to keep reading data, the following + // conditions prevent the data from being read: + // - The stream has ended (state.ended). + // - There is already a pending 'read' operation (state.reading). This is a + // case where the the stream has called the implementation defined _read() + // method, but they are processing the call asynchronously and have _not_ + // called push() with new data. In this case we skip performing more + // read()s. The execution ends in this method again after the _read() ends + // up calling push() with more data. + while (!state.reading && !state.ended && (state.length < state.highWaterMark || state.flowing && state.length === 0)) { + var len = state.length; + debug('maybeReadMore read 0'); + stream.read(0); + if (len === state.length) + // didn't get any data, stop spinning. + break; + } + state.readingMore = false; +} + +// abstract method. to be overridden in specific implementation classes. +// call cb(er, data) where data is <= n in length. +// for virtual (non-string, non-buffer) streams, "length" is somewhat +// arbitrary, and perhaps not very meaningful. +Readable.prototype._read = function (n) { + errorOrDestroy(this, new ERR_METHOD_NOT_IMPLEMENTED('_read()')); +}; +Readable.prototype.pipe = function (dest, pipeOpts) { + var src = this; + var state = this._readableState; + switch (state.pipesCount) { + case 0: + state.pipes = dest; + break; + case 1: + state.pipes = [state.pipes, dest]; + break; + default: + state.pipes.push(dest); + break; + } + state.pipesCount += 1; + debug('pipe count=%d opts=%j', state.pipesCount, pipeOpts); + var doEnd = (!pipeOpts || pipeOpts.end !== false) && dest !== process.stdout && dest !== process.stderr; + var endFn = doEnd ? onend : unpipe; + if (state.endEmitted) process.nextTick(endFn);else src.once('end', endFn); + dest.on('unpipe', onunpipe); + function onunpipe(readable, unpipeInfo) { + debug('onunpipe'); + if (readable === src) { + if (unpipeInfo && unpipeInfo.hasUnpiped === false) { + unpipeInfo.hasUnpiped = true; + cleanup(); + } + } + } + function onend() { + debug('onend'); + dest.end(); + } + + // when the dest drains, it reduces the awaitDrain counter + // on the source. This would be more elegant with a .once() + // handler in flow(), but adding and removing repeatedly is + // too slow. + var ondrain = pipeOnDrain(src); + dest.on('drain', ondrain); + var cleanedUp = false; + function cleanup() { + debug('cleanup'); + // cleanup event handlers once the pipe is broken + dest.removeListener('close', onclose); + dest.removeListener('finish', onfinish); + dest.removeListener('drain', ondrain); + dest.removeListener('error', onerror); + dest.removeListener('unpipe', onunpipe); + src.removeListener('end', onend); + src.removeListener('end', unpipe); + src.removeListener('data', ondata); + cleanedUp = true; + + // if the reader is waiting for a drain event from this + // specific writer, then it would cause it to never start + // flowing again. + // So, if this is awaiting a drain, then we just call it now. + // If we don't know, then assume that we are waiting for one. + if (state.awaitDrain && (!dest._writableState || dest._writableState.needDrain)) ondrain(); + } + src.on('data', ondata); + function ondata(chunk) { + debug('ondata'); + var ret = dest.write(chunk); + debug('dest.write', ret); + if (ret === false) { + // If the user unpiped during `dest.write()`, it is possible + // to get stuck in a permanently paused state if that write + // also returned false. + // => Check whether `dest` is still a piping destination. + if ((state.pipesCount === 1 && state.pipes === dest || state.pipesCount > 1 && indexOf(state.pipes, dest) !== -1) && !cleanedUp) { + debug('false write response, pause', state.awaitDrain); + state.awaitDrain++; + } + src.pause(); + } + } + + // if the dest has an error, then stop piping into it. + // however, don't suppress the throwing behavior for this. + function onerror(er) { + debug('onerror', er); + unpipe(); + dest.removeListener('error', onerror); + if (EElistenerCount(dest, 'error') === 0) errorOrDestroy(dest, er); + } + + // Make sure our error handler is attached before userland ones. + prependListener(dest, 'error', onerror); + + // Both close and finish should trigger unpipe, but only once. + function onclose() { + dest.removeListener('finish', onfinish); + unpipe(); + } + dest.once('close', onclose); + function onfinish() { + debug('onfinish'); + dest.removeListener('close', onclose); + unpipe(); + } + dest.once('finish', onfinish); + function unpipe() { + debug('unpipe'); + src.unpipe(dest); + } + + // tell the dest that it's being piped to + dest.emit('pipe', src); + + // start the flow if it hasn't been started already. + if (!state.flowing) { + debug('pipe resume'); + src.resume(); + } + return dest; +}; +function pipeOnDrain(src) { + return function pipeOnDrainFunctionResult() { + var state = src._readableState; + debug('pipeOnDrain', state.awaitDrain); + if (state.awaitDrain) state.awaitDrain--; + if (state.awaitDrain === 0 && EElistenerCount(src, 'data')) { + state.flowing = true; + flow(src); + } + }; +} +Readable.prototype.unpipe = function (dest) { + var state = this._readableState; + var unpipeInfo = { + hasUnpiped: false + }; + + // if we're not piping anywhere, then do nothing. + if (state.pipesCount === 0) return this; + + // just one destination. most common case. + if (state.pipesCount === 1) { + // passed in one, but it's not the right one. + if (dest && dest !== state.pipes) return this; + if (!dest) dest = state.pipes; + + // got a match. + state.pipes = null; + state.pipesCount = 0; + state.flowing = false; + if (dest) dest.emit('unpipe', this, unpipeInfo); + return this; + } + + // slow case. multiple pipe destinations. + + if (!dest) { + // remove all. + var dests = state.pipes; + var len = state.pipesCount; + state.pipes = null; + state.pipesCount = 0; + state.flowing = false; + for (var i = 0; i < len; i++) dests[i].emit('unpipe', this, { + hasUnpiped: false + }); + return this; + } + + // try to find the right one. + var index = indexOf(state.pipes, dest); + if (index === -1) return this; + state.pipes.splice(index, 1); + state.pipesCount -= 1; + if (state.pipesCount === 1) state.pipes = state.pipes[0]; + dest.emit('unpipe', this, unpipeInfo); + return this; +}; + +// set up data events if they are asked for +// Ensure readable listeners eventually get something +Readable.prototype.on = function (ev, fn) { + var res = Stream.prototype.on.call(this, ev, fn); + var state = this._readableState; + if (ev === 'data') { + // update readableListening so that resume() may be a no-op + // a few lines down. This is needed to support once('readable'). + state.readableListening = this.listenerCount('readable') > 0; + + // Try start flowing on next tick if stream isn't explicitly paused + if (state.flowing !== false) this.resume(); + } else if (ev === 'readable') { + if (!state.endEmitted && !state.readableListening) { + state.readableListening = state.needReadable = true; + state.flowing = false; + state.emittedReadable = false; + debug('on readable', state.length, state.reading); + if (state.length) { + emitReadable(this); + } else if (!state.reading) { + process.nextTick(nReadingNextTick, this); + } + } + } + return res; +}; +Readable.prototype.addListener = Readable.prototype.on; +Readable.prototype.removeListener = function (ev, fn) { + var res = Stream.prototype.removeListener.call(this, ev, fn); + if (ev === 'readable') { + // We need to check if there is someone still listening to + // readable and reset the state. However this needs to happen + // after readable has been emitted but before I/O (nextTick) to + // support once('readable', fn) cycles. This means that calling + // resume within the same tick will have no + // effect. + process.nextTick(updateReadableListening, this); + } + return res; +}; +Readable.prototype.removeAllListeners = function (ev) { + var res = Stream.prototype.removeAllListeners.apply(this, arguments); + if (ev === 'readable' || ev === undefined) { + // We need to check if there is someone still listening to + // readable and reset the state. However this needs to happen + // after readable has been emitted but before I/O (nextTick) to + // support once('readable', fn) cycles. This means that calling + // resume within the same tick will have no + // effect. + process.nextTick(updateReadableListening, this); + } + return res; +}; +function updateReadableListening(self) { + var state = self._readableState; + state.readableListening = self.listenerCount('readable') > 0; + if (state.resumeScheduled && !state.paused) { + // flowing needs to be set to true now, otherwise + // the upcoming resume will not flow. + state.flowing = true; + + // crude way to check if we should resume + } else if (self.listenerCount('data') > 0) { + self.resume(); + } +} +function nReadingNextTick(self) { + debug('readable nexttick read 0'); + self.read(0); +} + +// pause() and resume() are remnants of the legacy readable stream API +// If the user uses them, then switch into old mode. +Readable.prototype.resume = function () { + var state = this._readableState; + if (!state.flowing) { + debug('resume'); + // we flow only if there is no one listening + // for readable, but we still have to call + // resume() + state.flowing = !state.readableListening; + resume(this, state); + } + state.paused = false; + return this; +}; +function resume(stream, state) { + if (!state.resumeScheduled) { + state.resumeScheduled = true; + process.nextTick(resume_, stream, state); + } +} +function resume_(stream, state) { + debug('resume', state.reading); + if (!state.reading) { + stream.read(0); + } + state.resumeScheduled = false; + stream.emit('resume'); + flow(stream); + if (state.flowing && !state.reading) stream.read(0); +} +Readable.prototype.pause = function () { + debug('call pause flowing=%j', this._readableState.flowing); + if (this._readableState.flowing !== false) { + debug('pause'); + this._readableState.flowing = false; + this.emit('pause'); + } + this._readableState.paused = true; + return this; +}; +function flow(stream) { + var state = stream._readableState; + debug('flow', state.flowing); + while (state.flowing && stream.read() !== null); +} + +// wrap an old-style stream as the async data source. +// This is *not* part of the readable stream interface. +// It is an ugly unfortunate mess of history. +Readable.prototype.wrap = function (stream) { + var _this = this; + var state = this._readableState; + var paused = false; + stream.on('end', function () { + debug('wrapped end'); + if (state.decoder && !state.ended) { + var chunk = state.decoder.end(); + if (chunk && chunk.length) _this.push(chunk); + } + _this.push(null); + }); + stream.on('data', function (chunk) { + debug('wrapped data'); + if (state.decoder) chunk = state.decoder.write(chunk); + + // don't skip over falsy values in objectMode + if (state.objectMode && (chunk === null || chunk === undefined)) return;else if (!state.objectMode && (!chunk || !chunk.length)) return; + var ret = _this.push(chunk); + if (!ret) { + paused = true; + stream.pause(); + } + }); + + // proxy all the other methods. + // important when wrapping filters and duplexes. + for (var i in stream) { + if (this[i] === undefined && typeof stream[i] === 'function') { + this[i] = function methodWrap(method) { + return function methodWrapReturnFunction() { + return stream[method].apply(stream, arguments); + }; + }(i); + } + } + + // proxy certain important events. + for (var n = 0; n < kProxyEvents.length; n++) { + stream.on(kProxyEvents[n], this.emit.bind(this, kProxyEvents[n])); + } + + // when we try to consume some more bytes, simply unpause the + // underlying stream. + this._read = function (n) { + debug('wrapped _read', n); + if (paused) { + paused = false; + stream.resume(); + } + }; + return this; +}; +if (typeof Symbol === 'function') { + Readable.prototype[Symbol.asyncIterator] = function () { + if (createReadableStreamAsyncIterator === undefined) { + createReadableStreamAsyncIterator = require('./internal/streams/async_iterator'); + } + return createReadableStreamAsyncIterator(this); + }; +} +Object.defineProperty(Readable.prototype, 'readableHighWaterMark', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._readableState.highWaterMark; + } +}); +Object.defineProperty(Readable.prototype, 'readableBuffer', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._readableState && this._readableState.buffer; + } +}); +Object.defineProperty(Readable.prototype, 'readableFlowing', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._readableState.flowing; + }, + set: function set(state) { + if (this._readableState) { + this._readableState.flowing = state; + } + } +}); + +// exposed for testing purposes only. +Readable._fromList = fromList; +Object.defineProperty(Readable.prototype, 'readableLength', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._readableState.length; + } +}); + +// Pluck off n bytes from an array of buffers. +// Length is the combined lengths of all the buffers in the list. +// This function is designed to be inlinable, so please take care when making +// changes to the function body. +function fromList(n, state) { + // nothing buffered + if (state.length === 0) return null; + var ret; + if (state.objectMode) ret = state.buffer.shift();else if (!n || n >= state.length) { + // read it all, truncate the list + if (state.decoder) ret = state.buffer.join('');else if (state.buffer.length === 1) ret = state.buffer.first();else ret = state.buffer.concat(state.length); + state.buffer.clear(); + } else { + // read part of list + ret = state.buffer.consume(n, state.decoder); + } + return ret; +} +function endReadable(stream) { + var state = stream._readableState; + debug('endReadable', state.endEmitted); + if (!state.endEmitted) { + state.ended = true; + process.nextTick(endReadableNT, state, stream); + } +} +function endReadableNT(state, stream) { + debug('endReadableNT', state.endEmitted, state.length); + + // Check that we didn't get one last unshift. + if (!state.endEmitted && state.length === 0) { + state.endEmitted = true; + stream.readable = false; + stream.emit('end'); + if (state.autoDestroy) { + // In case of duplex streams we need a way to detect + // if the writable side is ready for autoDestroy as well + var wState = stream._writableState; + if (!wState || wState.autoDestroy && wState.finished) { + stream.destroy(); + } + } + } +} +if (typeof Symbol === 'function') { + Readable.from = function (iterable, opts) { + if (from === undefined) { + from = require('./internal/streams/from'); + } + return from(Readable, iterable, opts); + }; +} +function indexOf(xs, x) { + for (var i = 0, l = xs.length; i < l; i++) { + if (xs[i] === x) return i; + } + return -1; +} \ No newline at end of file diff --git a/node_modules/readable-stream/lib/_stream_transform.js b/node_modules/readable-stream/lib/_stream_transform.js new file mode 100644 index 0000000..1ccb715 --- /dev/null +++ b/node_modules/readable-stream/lib/_stream_transform.js @@ -0,0 +1,190 @@ +// Copyright Joyent, Inc. and other Node contributors. +// +// Permission is hereby granted, free of charge, to any person obtaining a +// copy of this software and associated documentation files (the +// "Software"), to deal in the Software without restriction, including +// without limitation the rights to use, copy, modify, merge, publish, +// distribute, sublicense, and/or sell copies of the Software, and to permit +// persons to whom the Software is furnished to do so, subject to the +// following conditions: +// +// The above copyright notice and this permission notice shall be included +// in all copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN +// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, +// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR +// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE +// USE OR OTHER DEALINGS IN THE SOFTWARE. + +// a transform stream is a readable/writable stream where you do +// something with the data. Sometimes it's called a "filter", +// but that's not a great name for it, since that implies a thing where +// some bits pass through, and others are simply ignored. (That would +// be a valid example of a transform, of course.) +// +// While the output is causally related to the input, it's not a +// necessarily symmetric or synchronous transformation. For example, +// a zlib stream might take multiple plain-text writes(), and then +// emit a single compressed chunk some time in the future. +// +// Here's how this works: +// +// The Transform stream has all the aspects of the readable and writable +// stream classes. When you write(chunk), that calls _write(chunk,cb) +// internally, and returns false if there's a lot of pending writes +// buffered up. When you call read(), that calls _read(n) until +// there's enough pending readable data buffered up. +// +// In a transform stream, the written data is placed in a buffer. When +// _read(n) is called, it transforms the queued up data, calling the +// buffered _write cb's as it consumes chunks. If consuming a single +// written chunk would result in multiple output chunks, then the first +// outputted bit calls the readcb, and subsequent chunks just go into +// the read buffer, and will cause it to emit 'readable' if necessary. +// +// This way, back-pressure is actually determined by the reading side, +// since _read has to be called to start processing a new chunk. However, +// a pathological inflate type of transform can cause excessive buffering +// here. For example, imagine a stream where every byte of input is +// interpreted as an integer from 0-255, and then results in that many +// bytes of output. Writing the 4 bytes {ff,ff,ff,ff} would result in +// 1kb of data being output. In this case, you could write a very small +// amount of input, and end up with a very large amount of output. In +// such a pathological inflating mechanism, there'd be no way to tell +// the system to stop doing the transform. A single 4MB write could +// cause the system to run out of memory. +// +// However, even in such a pathological case, only a single written chunk +// would be consumed, and then the rest would wait (un-transformed) until +// the results of the previous transformed chunk were consumed. + +'use strict'; + +module.exports = Transform; +var _require$codes = require('../errors').codes, + ERR_METHOD_NOT_IMPLEMENTED = _require$codes.ERR_METHOD_NOT_IMPLEMENTED, + ERR_MULTIPLE_CALLBACK = _require$codes.ERR_MULTIPLE_CALLBACK, + ERR_TRANSFORM_ALREADY_TRANSFORMING = _require$codes.ERR_TRANSFORM_ALREADY_TRANSFORMING, + ERR_TRANSFORM_WITH_LENGTH_0 = _require$codes.ERR_TRANSFORM_WITH_LENGTH_0; +var Duplex = require('./_stream_duplex'); +require('inherits')(Transform, Duplex); +function afterTransform(er, data) { + var ts = this._transformState; + ts.transforming = false; + var cb = ts.writecb; + if (cb === null) { + return this.emit('error', new ERR_MULTIPLE_CALLBACK()); + } + ts.writechunk = null; + ts.writecb = null; + if (data != null) + // single equals check for both `null` and `undefined` + this.push(data); + cb(er); + var rs = this._readableState; + rs.reading = false; + if (rs.needReadable || rs.length < rs.highWaterMark) { + this._read(rs.highWaterMark); + } +} +function Transform(options) { + if (!(this instanceof Transform)) return new Transform(options); + Duplex.call(this, options); + this._transformState = { + afterTransform: afterTransform.bind(this), + needTransform: false, + transforming: false, + writecb: null, + writechunk: null, + writeencoding: null + }; + + // start out asking for a readable event once data is transformed. + this._readableState.needReadable = true; + + // we have implemented the _read method, and done the other things + // that Readable wants before the first _read call, so unset the + // sync guard flag. + this._readableState.sync = false; + if (options) { + if (typeof options.transform === 'function') this._transform = options.transform; + if (typeof options.flush === 'function') this._flush = options.flush; + } + + // When the writable side finishes, then flush out anything remaining. + this.on('prefinish', prefinish); +} +function prefinish() { + var _this = this; + if (typeof this._flush === 'function' && !this._readableState.destroyed) { + this._flush(function (er, data) { + done(_this, er, data); + }); + } else { + done(this, null, null); + } +} +Transform.prototype.push = function (chunk, encoding) { + this._transformState.needTransform = false; + return Duplex.prototype.push.call(this, chunk, encoding); +}; + +// This is the part where you do stuff! +// override this function in implementation classes. +// 'chunk' is an input chunk. +// +// Call `push(newChunk)` to pass along transformed output +// to the readable side. You may call 'push' zero or more times. +// +// Call `cb(err)` when you are done with this chunk. If you pass +// an error, then that'll put the hurt on the whole operation. If you +// never call cb(), then you'll never get another chunk. +Transform.prototype._transform = function (chunk, encoding, cb) { + cb(new ERR_METHOD_NOT_IMPLEMENTED('_transform()')); +}; +Transform.prototype._write = function (chunk, encoding, cb) { + var ts = this._transformState; + ts.writecb = cb; + ts.writechunk = chunk; + ts.writeencoding = encoding; + if (!ts.transforming) { + var rs = this._readableState; + if (ts.needTransform || rs.needReadable || rs.length < rs.highWaterMark) this._read(rs.highWaterMark); + } +}; + +// Doesn't matter what the args are here. +// _transform does all the work. +// That we got here means that the readable side wants more data. +Transform.prototype._read = function (n) { + var ts = this._transformState; + if (ts.writechunk !== null && !ts.transforming) { + ts.transforming = true; + this._transform(ts.writechunk, ts.writeencoding, ts.afterTransform); + } else { + // mark that we need a transform, so that any data that comes in + // will get processed, now that we've asked for it. + ts.needTransform = true; + } +}; +Transform.prototype._destroy = function (err, cb) { + Duplex.prototype._destroy.call(this, err, function (err2) { + cb(err2); + }); +}; +function done(stream, er, data) { + if (er) return stream.emit('error', er); + if (data != null) + // single equals check for both `null` and `undefined` + stream.push(data); + + // TODO(BridgeAR): Write a test for these two error cases + // if there's nothing in the write buffer, then that means + // that nothing more will ever be provided + if (stream._writableState.length) throw new ERR_TRANSFORM_WITH_LENGTH_0(); + if (stream._transformState.transforming) throw new ERR_TRANSFORM_ALREADY_TRANSFORMING(); + return stream.push(null); +} \ No newline at end of file diff --git a/node_modules/readable-stream/lib/_stream_writable.js b/node_modules/readable-stream/lib/_stream_writable.js new file mode 100644 index 0000000..292415e --- /dev/null +++ b/node_modules/readable-stream/lib/_stream_writable.js @@ -0,0 +1,641 @@ +// Copyright Joyent, Inc. and other Node contributors. +// +// Permission is hereby granted, free of charge, to any person obtaining a +// copy of this software and associated documentation files (the +// "Software"), to deal in the Software without restriction, including +// without limitation the rights to use, copy, modify, merge, publish, +// distribute, sublicense, and/or sell copies of the Software, and to permit +// persons to whom the Software is furnished to do so, subject to the +// following conditions: +// +// The above copyright notice and this permission notice shall be included +// in all copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN +// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, +// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR +// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE +// USE OR OTHER DEALINGS IN THE SOFTWARE. + +// A bit simpler than readable streams. +// Implement an async ._write(chunk, encoding, cb), and it'll handle all +// the drain event emission and buffering. + +'use strict'; + +module.exports = Writable; + +/* */ +function WriteReq(chunk, encoding, cb) { + this.chunk = chunk; + this.encoding = encoding; + this.callback = cb; + this.next = null; +} + +// It seems a linked list but it is not +// there will be only 2 of these for each stream +function CorkedRequest(state) { + var _this = this; + this.next = null; + this.entry = null; + this.finish = function () { + onCorkedFinish(_this, state); + }; +} +/* */ + +/**/ +var Duplex; +/**/ + +Writable.WritableState = WritableState; + +/**/ +var internalUtil = { + deprecate: require('util-deprecate') +}; +/**/ + +/**/ +var Stream = require('./internal/streams/stream'); +/**/ + +var Buffer = require('buffer').Buffer; +var OurUint8Array = (typeof global !== 'undefined' ? global : typeof window !== 'undefined' ? window : typeof self !== 'undefined' ? self : {}).Uint8Array || function () {}; +function _uint8ArrayToBuffer(chunk) { + return Buffer.from(chunk); +} +function _isUint8Array(obj) { + return Buffer.isBuffer(obj) || obj instanceof OurUint8Array; +} +var destroyImpl = require('./internal/streams/destroy'); +var _require = require('./internal/streams/state'), + getHighWaterMark = _require.getHighWaterMark; +var _require$codes = require('../errors').codes, + ERR_INVALID_ARG_TYPE = _require$codes.ERR_INVALID_ARG_TYPE, + ERR_METHOD_NOT_IMPLEMENTED = _require$codes.ERR_METHOD_NOT_IMPLEMENTED, + ERR_MULTIPLE_CALLBACK = _require$codes.ERR_MULTIPLE_CALLBACK, + ERR_STREAM_CANNOT_PIPE = _require$codes.ERR_STREAM_CANNOT_PIPE, + ERR_STREAM_DESTROYED = _require$codes.ERR_STREAM_DESTROYED, + ERR_STREAM_NULL_VALUES = _require$codes.ERR_STREAM_NULL_VALUES, + ERR_STREAM_WRITE_AFTER_END = _require$codes.ERR_STREAM_WRITE_AFTER_END, + ERR_UNKNOWN_ENCODING = _require$codes.ERR_UNKNOWN_ENCODING; +var errorOrDestroy = destroyImpl.errorOrDestroy; +require('inherits')(Writable, Stream); +function nop() {} +function WritableState(options, stream, isDuplex) { + Duplex = Duplex || require('./_stream_duplex'); + options = options || {}; + + // Duplex streams are both readable and writable, but share + // the same options object. + // However, some cases require setting options to different + // values for the readable and the writable sides of the duplex stream, + // e.g. options.readableObjectMode vs. options.writableObjectMode, etc. + if (typeof isDuplex !== 'boolean') isDuplex = stream instanceof Duplex; + + // object stream flag to indicate whether or not this stream + // contains buffers or objects. + this.objectMode = !!options.objectMode; + if (isDuplex) this.objectMode = this.objectMode || !!options.writableObjectMode; + + // the point at which write() starts returning false + // Note: 0 is a valid value, means that we always return false if + // the entire buffer is not flushed immediately on write() + this.highWaterMark = getHighWaterMark(this, options, 'writableHighWaterMark', isDuplex); + + // if _final has been called + this.finalCalled = false; + + // drain event flag. + this.needDrain = false; + // at the start of calling end() + this.ending = false; + // when end() has been called, and returned + this.ended = false; + // when 'finish' is emitted + this.finished = false; + + // has it been destroyed + this.destroyed = false; + + // should we decode strings into buffers before passing to _write? + // this is here so that some node-core streams can optimize string + // handling at a lower level. + var noDecode = options.decodeStrings === false; + this.decodeStrings = !noDecode; + + // Crypto is kind of old and crusty. Historically, its default string + // encoding is 'binary' so we have to make this configurable. + // Everything else in the universe uses 'utf8', though. + this.defaultEncoding = options.defaultEncoding || 'utf8'; + + // not an actual buffer we keep track of, but a measurement + // of how much we're waiting to get pushed to some underlying + // socket or file. + this.length = 0; + + // a flag to see when we're in the middle of a write. + this.writing = false; + + // when true all writes will be buffered until .uncork() call + this.corked = 0; + + // a flag to be able to tell if the onwrite cb is called immediately, + // or on a later tick. We set this to true at first, because any + // actions that shouldn't happen until "later" should generally also + // not happen before the first write call. + this.sync = true; + + // a flag to know if we're processing previously buffered items, which + // may call the _write() callback in the same tick, so that we don't + // end up in an overlapped onwrite situation. + this.bufferProcessing = false; + + // the callback that's passed to _write(chunk,cb) + this.onwrite = function (er) { + onwrite(stream, er); + }; + + // the callback that the user supplies to write(chunk,encoding,cb) + this.writecb = null; + + // the amount that is being written when _write is called. + this.writelen = 0; + this.bufferedRequest = null; + this.lastBufferedRequest = null; + + // number of pending user-supplied write callbacks + // this must be 0 before 'finish' can be emitted + this.pendingcb = 0; + + // emit prefinish if the only thing we're waiting for is _write cbs + // This is relevant for synchronous Transform streams + this.prefinished = false; + + // True if the error was already emitted and should not be thrown again + this.errorEmitted = false; + + // Should close be emitted on destroy. Defaults to true. + this.emitClose = options.emitClose !== false; + + // Should .destroy() be called after 'finish' (and potentially 'end') + this.autoDestroy = !!options.autoDestroy; + + // count buffered requests + this.bufferedRequestCount = 0; + + // allocate the first CorkedRequest, there is always + // one allocated and free to use, and we maintain at most two + this.corkedRequestsFree = new CorkedRequest(this); +} +WritableState.prototype.getBuffer = function getBuffer() { + var current = this.bufferedRequest; + var out = []; + while (current) { + out.push(current); + current = current.next; + } + return out; +}; +(function () { + try { + Object.defineProperty(WritableState.prototype, 'buffer', { + get: internalUtil.deprecate(function writableStateBufferGetter() { + return this.getBuffer(); + }, '_writableState.buffer is deprecated. Use _writableState.getBuffer ' + 'instead.', 'DEP0003') + }); + } catch (_) {} +})(); + +// Test _writableState for inheritance to account for Duplex streams, +// whose prototype chain only points to Readable. +var realHasInstance; +if (typeof Symbol === 'function' && Symbol.hasInstance && typeof Function.prototype[Symbol.hasInstance] === 'function') { + realHasInstance = Function.prototype[Symbol.hasInstance]; + Object.defineProperty(Writable, Symbol.hasInstance, { + value: function value(object) { + if (realHasInstance.call(this, object)) return true; + if (this !== Writable) return false; + return object && object._writableState instanceof WritableState; + } + }); +} else { + realHasInstance = function realHasInstance(object) { + return object instanceof this; + }; +} +function Writable(options) { + Duplex = Duplex || require('./_stream_duplex'); + + // Writable ctor is applied to Duplexes, too. + // `realHasInstance` is necessary because using plain `instanceof` + // would return false, as no `_writableState` property is attached. + + // Trying to use the custom `instanceof` for Writable here will also break the + // Node.js LazyTransform implementation, which has a non-trivial getter for + // `_writableState` that would lead to infinite recursion. + + // Checking for a Stream.Duplex instance is faster here instead of inside + // the WritableState constructor, at least with V8 6.5 + var isDuplex = this instanceof Duplex; + if (!isDuplex && !realHasInstance.call(Writable, this)) return new Writable(options); + this._writableState = new WritableState(options, this, isDuplex); + + // legacy. + this.writable = true; + if (options) { + if (typeof options.write === 'function') this._write = options.write; + if (typeof options.writev === 'function') this._writev = options.writev; + if (typeof options.destroy === 'function') this._destroy = options.destroy; + if (typeof options.final === 'function') this._final = options.final; + } + Stream.call(this); +} + +// Otherwise people can pipe Writable streams, which is just wrong. +Writable.prototype.pipe = function () { + errorOrDestroy(this, new ERR_STREAM_CANNOT_PIPE()); +}; +function writeAfterEnd(stream, cb) { + var er = new ERR_STREAM_WRITE_AFTER_END(); + // TODO: defer error events consistently everywhere, not just the cb + errorOrDestroy(stream, er); + process.nextTick(cb, er); +} + +// Checks that a user-supplied chunk is valid, especially for the particular +// mode the stream is in. Currently this means that `null` is never accepted +// and undefined/non-string values are only allowed in object mode. +function validChunk(stream, state, chunk, cb) { + var er; + if (chunk === null) { + er = new ERR_STREAM_NULL_VALUES(); + } else if (typeof chunk !== 'string' && !state.objectMode) { + er = new ERR_INVALID_ARG_TYPE('chunk', ['string', 'Buffer'], chunk); + } + if (er) { + errorOrDestroy(stream, er); + process.nextTick(cb, er); + return false; + } + return true; +} +Writable.prototype.write = function (chunk, encoding, cb) { + var state = this._writableState; + var ret = false; + var isBuf = !state.objectMode && _isUint8Array(chunk); + if (isBuf && !Buffer.isBuffer(chunk)) { + chunk = _uint8ArrayToBuffer(chunk); + } + if (typeof encoding === 'function') { + cb = encoding; + encoding = null; + } + if (isBuf) encoding = 'buffer';else if (!encoding) encoding = state.defaultEncoding; + if (typeof cb !== 'function') cb = nop; + if (state.ending) writeAfterEnd(this, cb);else if (isBuf || validChunk(this, state, chunk, cb)) { + state.pendingcb++; + ret = writeOrBuffer(this, state, isBuf, chunk, encoding, cb); + } + return ret; +}; +Writable.prototype.cork = function () { + this._writableState.corked++; +}; +Writable.prototype.uncork = function () { + var state = this._writableState; + if (state.corked) { + state.corked--; + if (!state.writing && !state.corked && !state.bufferProcessing && state.bufferedRequest) clearBuffer(this, state); + } +}; +Writable.prototype.setDefaultEncoding = function setDefaultEncoding(encoding) { + // node::ParseEncoding() requires lower case. + if (typeof encoding === 'string') encoding = encoding.toLowerCase(); + if (!(['hex', 'utf8', 'utf-8', 'ascii', 'binary', 'base64', 'ucs2', 'ucs-2', 'utf16le', 'utf-16le', 'raw'].indexOf((encoding + '').toLowerCase()) > -1)) throw new ERR_UNKNOWN_ENCODING(encoding); + this._writableState.defaultEncoding = encoding; + return this; +}; +Object.defineProperty(Writable.prototype, 'writableBuffer', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._writableState && this._writableState.getBuffer(); + } +}); +function decodeChunk(state, chunk, encoding) { + if (!state.objectMode && state.decodeStrings !== false && typeof chunk === 'string') { + chunk = Buffer.from(chunk, encoding); + } + return chunk; +} +Object.defineProperty(Writable.prototype, 'writableHighWaterMark', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._writableState.highWaterMark; + } +}); + +// if we're already writing something, then just put this +// in the queue, and wait our turn. Otherwise, call _write +// If we return false, then we need a drain event, so set that flag. +function writeOrBuffer(stream, state, isBuf, chunk, encoding, cb) { + if (!isBuf) { + var newChunk = decodeChunk(state, chunk, encoding); + if (chunk !== newChunk) { + isBuf = true; + encoding = 'buffer'; + chunk = newChunk; + } + } + var len = state.objectMode ? 1 : chunk.length; + state.length += len; + var ret = state.length < state.highWaterMark; + // we must ensure that previous needDrain will not be reset to false. + if (!ret) state.needDrain = true; + if (state.writing || state.corked) { + var last = state.lastBufferedRequest; + state.lastBufferedRequest = { + chunk: chunk, + encoding: encoding, + isBuf: isBuf, + callback: cb, + next: null + }; + if (last) { + last.next = state.lastBufferedRequest; + } else { + state.bufferedRequest = state.lastBufferedRequest; + } + state.bufferedRequestCount += 1; + } else { + doWrite(stream, state, false, len, chunk, encoding, cb); + } + return ret; +} +function doWrite(stream, state, writev, len, chunk, encoding, cb) { + state.writelen = len; + state.writecb = cb; + state.writing = true; + state.sync = true; + if (state.destroyed) state.onwrite(new ERR_STREAM_DESTROYED('write'));else if (writev) stream._writev(chunk, state.onwrite);else stream._write(chunk, encoding, state.onwrite); + state.sync = false; +} +function onwriteError(stream, state, sync, er, cb) { + --state.pendingcb; + if (sync) { + // defer the callback if we are being called synchronously + // to avoid piling up things on the stack + process.nextTick(cb, er); + // this can emit finish, and it will always happen + // after error + process.nextTick(finishMaybe, stream, state); + stream._writableState.errorEmitted = true; + errorOrDestroy(stream, er); + } else { + // the caller expect this to happen before if + // it is async + cb(er); + stream._writableState.errorEmitted = true; + errorOrDestroy(stream, er); + // this can emit finish, but finish must + // always follow error + finishMaybe(stream, state); + } +} +function onwriteStateUpdate(state) { + state.writing = false; + state.writecb = null; + state.length -= state.writelen; + state.writelen = 0; +} +function onwrite(stream, er) { + var state = stream._writableState; + var sync = state.sync; + var cb = state.writecb; + if (typeof cb !== 'function') throw new ERR_MULTIPLE_CALLBACK(); + onwriteStateUpdate(state); + if (er) onwriteError(stream, state, sync, er, cb);else { + // Check if we're actually ready to finish, but don't emit yet + var finished = needFinish(state) || stream.destroyed; + if (!finished && !state.corked && !state.bufferProcessing && state.bufferedRequest) { + clearBuffer(stream, state); + } + if (sync) { + process.nextTick(afterWrite, stream, state, finished, cb); + } else { + afterWrite(stream, state, finished, cb); + } + } +} +function afterWrite(stream, state, finished, cb) { + if (!finished) onwriteDrain(stream, state); + state.pendingcb--; + cb(); + finishMaybe(stream, state); +} + +// Must force callback to be called on nextTick, so that we don't +// emit 'drain' before the write() consumer gets the 'false' return +// value, and has a chance to attach a 'drain' listener. +function onwriteDrain(stream, state) { + if (state.length === 0 && state.needDrain) { + state.needDrain = false; + stream.emit('drain'); + } +} + +// if there's something in the buffer waiting, then process it +function clearBuffer(stream, state) { + state.bufferProcessing = true; + var entry = state.bufferedRequest; + if (stream._writev && entry && entry.next) { + // Fast case, write everything using _writev() + var l = state.bufferedRequestCount; + var buffer = new Array(l); + var holder = state.corkedRequestsFree; + holder.entry = entry; + var count = 0; + var allBuffers = true; + while (entry) { + buffer[count] = entry; + if (!entry.isBuf) allBuffers = false; + entry = entry.next; + count += 1; + } + buffer.allBuffers = allBuffers; + doWrite(stream, state, true, state.length, buffer, '', holder.finish); + + // doWrite is almost always async, defer these to save a bit of time + // as the hot path ends with doWrite + state.pendingcb++; + state.lastBufferedRequest = null; + if (holder.next) { + state.corkedRequestsFree = holder.next; + holder.next = null; + } else { + state.corkedRequestsFree = new CorkedRequest(state); + } + state.bufferedRequestCount = 0; + } else { + // Slow case, write chunks one-by-one + while (entry) { + var chunk = entry.chunk; + var encoding = entry.encoding; + var cb = entry.callback; + var len = state.objectMode ? 1 : chunk.length; + doWrite(stream, state, false, len, chunk, encoding, cb); + entry = entry.next; + state.bufferedRequestCount--; + // if we didn't call the onwrite immediately, then + // it means that we need to wait until it does. + // also, that means that the chunk and cb are currently + // being processed, so move the buffer counter past them. + if (state.writing) { + break; + } + } + if (entry === null) state.lastBufferedRequest = null; + } + state.bufferedRequest = entry; + state.bufferProcessing = false; +} +Writable.prototype._write = function (chunk, encoding, cb) { + cb(new ERR_METHOD_NOT_IMPLEMENTED('_write()')); +}; +Writable.prototype._writev = null; +Writable.prototype.end = function (chunk, encoding, cb) { + var state = this._writableState; + if (typeof chunk === 'function') { + cb = chunk; + chunk = null; + encoding = null; + } else if (typeof encoding === 'function') { + cb = encoding; + encoding = null; + } + if (chunk !== null && chunk !== undefined) this.write(chunk, encoding); + + // .end() fully uncorks + if (state.corked) { + state.corked = 1; + this.uncork(); + } + + // ignore unnecessary end() calls. + if (!state.ending) endWritable(this, state, cb); + return this; +}; +Object.defineProperty(Writable.prototype, 'writableLength', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._writableState.length; + } +}); +function needFinish(state) { + return state.ending && state.length === 0 && state.bufferedRequest === null && !state.finished && !state.writing; +} +function callFinal(stream, state) { + stream._final(function (err) { + state.pendingcb--; + if (err) { + errorOrDestroy(stream, err); + } + state.prefinished = true; + stream.emit('prefinish'); + finishMaybe(stream, state); + }); +} +function prefinish(stream, state) { + if (!state.prefinished && !state.finalCalled) { + if (typeof stream._final === 'function' && !state.destroyed) { + state.pendingcb++; + state.finalCalled = true; + process.nextTick(callFinal, stream, state); + } else { + state.prefinished = true; + stream.emit('prefinish'); + } + } +} +function finishMaybe(stream, state) { + var need = needFinish(state); + if (need) { + prefinish(stream, state); + if (state.pendingcb === 0) { + state.finished = true; + stream.emit('finish'); + if (state.autoDestroy) { + // In case of duplex streams we need a way to detect + // if the readable side is ready for autoDestroy as well + var rState = stream._readableState; + if (!rState || rState.autoDestroy && rState.endEmitted) { + stream.destroy(); + } + } + } + } + return need; +} +function endWritable(stream, state, cb) { + state.ending = true; + finishMaybe(stream, state); + if (cb) { + if (state.finished) process.nextTick(cb);else stream.once('finish', cb); + } + state.ended = true; + stream.writable = false; +} +function onCorkedFinish(corkReq, state, err) { + var entry = corkReq.entry; + corkReq.entry = null; + while (entry) { + var cb = entry.callback; + state.pendingcb--; + cb(err); + entry = entry.next; + } + + // reuse the free corkReq. + state.corkedRequestsFree.next = corkReq; +} +Object.defineProperty(Writable.prototype, 'destroyed', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + if (this._writableState === undefined) { + return false; + } + return this._writableState.destroyed; + }, + set: function set(value) { + // we ignore the value if the stream + // has not been initialized yet + if (!this._writableState) { + return; + } + + // backward compatibility, the user is explicitly + // managing destroyed + this._writableState.destroyed = value; + } +}); +Writable.prototype.destroy = destroyImpl.destroy; +Writable.prototype._undestroy = destroyImpl.undestroy; +Writable.prototype._destroy = function (err, cb) { + cb(err); +}; \ No newline at end of file diff --git a/node_modules/readable-stream/lib/internal/streams/async_iterator.js b/node_modules/readable-stream/lib/internal/streams/async_iterator.js new file mode 100644 index 0000000..742c5a4 --- /dev/null +++ b/node_modules/readable-stream/lib/internal/streams/async_iterator.js @@ -0,0 +1,180 @@ +'use strict'; + +var _Object$setPrototypeO; +function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } +function _toPropertyKey(arg) { var key = _toPrimitive(arg, "string"); return typeof key === "symbol" ? key : String(key); } +function _toPrimitive(input, hint) { if (typeof input !== "object" || input === null) return input; var prim = input[Symbol.toPrimitive]; if (prim !== undefined) { var res = prim.call(input, hint || "default"); if (typeof res !== "object") return res; throw new TypeError("@@toPrimitive must return a primitive value."); } return (hint === "string" ? String : Number)(input); } +var finished = require('./end-of-stream'); +var kLastResolve = Symbol('lastResolve'); +var kLastReject = Symbol('lastReject'); +var kError = Symbol('error'); +var kEnded = Symbol('ended'); +var kLastPromise = Symbol('lastPromise'); +var kHandlePromise = Symbol('handlePromise'); +var kStream = Symbol('stream'); +function createIterResult(value, done) { + return { + value: value, + done: done + }; +} +function readAndResolve(iter) { + var resolve = iter[kLastResolve]; + if (resolve !== null) { + var data = iter[kStream].read(); + // we defer if data is null + // we can be expecting either 'end' or + // 'error' + if (data !== null) { + iter[kLastPromise] = null; + iter[kLastResolve] = null; + iter[kLastReject] = null; + resolve(createIterResult(data, false)); + } + } +} +function onReadable(iter) { + // we wait for the next tick, because it might + // emit an error with process.nextTick + process.nextTick(readAndResolve, iter); +} +function wrapForNext(lastPromise, iter) { + return function (resolve, reject) { + lastPromise.then(function () { + if (iter[kEnded]) { + resolve(createIterResult(undefined, true)); + return; + } + iter[kHandlePromise](resolve, reject); + }, reject); + }; +} +var AsyncIteratorPrototype = Object.getPrototypeOf(function () {}); +var ReadableStreamAsyncIteratorPrototype = Object.setPrototypeOf((_Object$setPrototypeO = { + get stream() { + return this[kStream]; + }, + next: function next() { + var _this = this; + // if we have detected an error in the meanwhile + // reject straight away + var error = this[kError]; + if (error !== null) { + return Promise.reject(error); + } + if (this[kEnded]) { + return Promise.resolve(createIterResult(undefined, true)); + } + if (this[kStream].destroyed) { + // We need to defer via nextTick because if .destroy(err) is + // called, the error will be emitted via nextTick, and + // we cannot guarantee that there is no error lingering around + // waiting to be emitted. + return new Promise(function (resolve, reject) { + process.nextTick(function () { + if (_this[kError]) { + reject(_this[kError]); + } else { + resolve(createIterResult(undefined, true)); + } + }); + }); + } + + // if we have multiple next() calls + // we will wait for the previous Promise to finish + // this logic is optimized to support for await loops, + // where next() is only called once at a time + var lastPromise = this[kLastPromise]; + var promise; + if (lastPromise) { + promise = new Promise(wrapForNext(lastPromise, this)); + } else { + // fast path needed to support multiple this.push() + // without triggering the next() queue + var data = this[kStream].read(); + if (data !== null) { + return Promise.resolve(createIterResult(data, false)); + } + promise = new Promise(this[kHandlePromise]); + } + this[kLastPromise] = promise; + return promise; + } +}, _defineProperty(_Object$setPrototypeO, Symbol.asyncIterator, function () { + return this; +}), _defineProperty(_Object$setPrototypeO, "return", function _return() { + var _this2 = this; + // destroy(err, cb) is a private API + // we can guarantee we have that here, because we control the + // Readable class this is attached to + return new Promise(function (resolve, reject) { + _this2[kStream].destroy(null, function (err) { + if (err) { + reject(err); + return; + } + resolve(createIterResult(undefined, true)); + }); + }); +}), _Object$setPrototypeO), AsyncIteratorPrototype); +var createReadableStreamAsyncIterator = function createReadableStreamAsyncIterator(stream) { + var _Object$create; + var iterator = Object.create(ReadableStreamAsyncIteratorPrototype, (_Object$create = {}, _defineProperty(_Object$create, kStream, { + value: stream, + writable: true + }), _defineProperty(_Object$create, kLastResolve, { + value: null, + writable: true + }), _defineProperty(_Object$create, kLastReject, { + value: null, + writable: true + }), _defineProperty(_Object$create, kError, { + value: null, + writable: true + }), _defineProperty(_Object$create, kEnded, { + value: stream._readableState.endEmitted, + writable: true + }), _defineProperty(_Object$create, kHandlePromise, { + value: function value(resolve, reject) { + var data = iterator[kStream].read(); + if (data) { + iterator[kLastPromise] = null; + iterator[kLastResolve] = null; + iterator[kLastReject] = null; + resolve(createIterResult(data, false)); + } else { + iterator[kLastResolve] = resolve; + iterator[kLastReject] = reject; + } + }, + writable: true + }), _Object$create)); + iterator[kLastPromise] = null; + finished(stream, function (err) { + if (err && err.code !== 'ERR_STREAM_PREMATURE_CLOSE') { + var reject = iterator[kLastReject]; + // reject if we are waiting for data in the Promise + // returned by next() and store the error + if (reject !== null) { + iterator[kLastPromise] = null; + iterator[kLastResolve] = null; + iterator[kLastReject] = null; + reject(err); + } + iterator[kError] = err; + return; + } + var resolve = iterator[kLastResolve]; + if (resolve !== null) { + iterator[kLastPromise] = null; + iterator[kLastResolve] = null; + iterator[kLastReject] = null; + resolve(createIterResult(undefined, true)); + } + iterator[kEnded] = true; + }); + stream.on('readable', onReadable.bind(null, iterator)); + return iterator; +}; +module.exports = createReadableStreamAsyncIterator; \ No newline at end of file diff --git a/node_modules/readable-stream/lib/internal/streams/buffer_list.js b/node_modules/readable-stream/lib/internal/streams/buffer_list.js new file mode 100644 index 0000000..69bda49 --- /dev/null +++ b/node_modules/readable-stream/lib/internal/streams/buffer_list.js @@ -0,0 +1,183 @@ +'use strict'; + +function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; } +function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { _defineProperty(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; } +function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } +function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } +function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, _toPropertyKey(descriptor.key), descriptor); } } +function _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); Object.defineProperty(Constructor, "prototype", { writable: false }); return Constructor; } +function _toPropertyKey(arg) { var key = _toPrimitive(arg, "string"); return typeof key === "symbol" ? key : String(key); } +function _toPrimitive(input, hint) { if (typeof input !== "object" || input === null) return input; var prim = input[Symbol.toPrimitive]; if (prim !== undefined) { var res = prim.call(input, hint || "default"); if (typeof res !== "object") return res; throw new TypeError("@@toPrimitive must return a primitive value."); } return (hint === "string" ? String : Number)(input); } +var _require = require('buffer'), + Buffer = _require.Buffer; +var _require2 = require('util'), + inspect = _require2.inspect; +var custom = inspect && inspect.custom || 'inspect'; +function copyBuffer(src, target, offset) { + Buffer.prototype.copy.call(src, target, offset); +} +module.exports = /*#__PURE__*/function () { + function BufferList() { + _classCallCheck(this, BufferList); + this.head = null; + this.tail = null; + this.length = 0; + } + _createClass(BufferList, [{ + key: "push", + value: function push(v) { + var entry = { + data: v, + next: null + }; + if (this.length > 0) this.tail.next = entry;else this.head = entry; + this.tail = entry; + ++this.length; + } + }, { + key: "unshift", + value: function unshift(v) { + var entry = { + data: v, + next: this.head + }; + if (this.length === 0) this.tail = entry; + this.head = entry; + ++this.length; + } + }, { + key: "shift", + value: function shift() { + if (this.length === 0) return; + var ret = this.head.data; + if (this.length === 1) this.head = this.tail = null;else this.head = this.head.next; + --this.length; + return ret; + } + }, { + key: "clear", + value: function clear() { + this.head = this.tail = null; + this.length = 0; + } + }, { + key: "join", + value: function join(s) { + if (this.length === 0) return ''; + var p = this.head; + var ret = '' + p.data; + while (p = p.next) ret += s + p.data; + return ret; + } + }, { + key: "concat", + value: function concat(n) { + if (this.length === 0) return Buffer.alloc(0); + var ret = Buffer.allocUnsafe(n >>> 0); + var p = this.head; + var i = 0; + while (p) { + copyBuffer(p.data, ret, i); + i += p.data.length; + p = p.next; + } + return ret; + } + + // Consumes a specified amount of bytes or characters from the buffered data. + }, { + key: "consume", + value: function consume(n, hasStrings) { + var ret; + if (n < this.head.data.length) { + // `slice` is the same for buffers and strings. + ret = this.head.data.slice(0, n); + this.head.data = this.head.data.slice(n); + } else if (n === this.head.data.length) { + // First chunk is a perfect match. + ret = this.shift(); + } else { + // Result spans more than one buffer. + ret = hasStrings ? this._getString(n) : this._getBuffer(n); + } + return ret; + } + }, { + key: "first", + value: function first() { + return this.head.data; + } + + // Consumes a specified amount of characters from the buffered data. + }, { + key: "_getString", + value: function _getString(n) { + var p = this.head; + var c = 1; + var ret = p.data; + n -= ret.length; + while (p = p.next) { + var str = p.data; + var nb = n > str.length ? str.length : n; + if (nb === str.length) ret += str;else ret += str.slice(0, n); + n -= nb; + if (n === 0) { + if (nb === str.length) { + ++c; + if (p.next) this.head = p.next;else this.head = this.tail = null; + } else { + this.head = p; + p.data = str.slice(nb); + } + break; + } + ++c; + } + this.length -= c; + return ret; + } + + // Consumes a specified amount of bytes from the buffered data. + }, { + key: "_getBuffer", + value: function _getBuffer(n) { + var ret = Buffer.allocUnsafe(n); + var p = this.head; + var c = 1; + p.data.copy(ret); + n -= p.data.length; + while (p = p.next) { + var buf = p.data; + var nb = n > buf.length ? buf.length : n; + buf.copy(ret, ret.length - n, 0, nb); + n -= nb; + if (n === 0) { + if (nb === buf.length) { + ++c; + if (p.next) this.head = p.next;else this.head = this.tail = null; + } else { + this.head = p; + p.data = buf.slice(nb); + } + break; + } + ++c; + } + this.length -= c; + return ret; + } + + // Make sure the linked list only shows the minimal necessary information. + }, { + key: custom, + value: function value(_, options) { + return inspect(this, _objectSpread(_objectSpread({}, options), {}, { + // Only inspect one level. + depth: 0, + // It should not recurse. + customInspect: false + })); + } + }]); + return BufferList; +}(); \ No newline at end of file diff --git a/node_modules/readable-stream/lib/internal/streams/destroy.js b/node_modules/readable-stream/lib/internal/streams/destroy.js new file mode 100644 index 0000000..31a17c4 --- /dev/null +++ b/node_modules/readable-stream/lib/internal/streams/destroy.js @@ -0,0 +1,96 @@ +'use strict'; + +// undocumented cb() API, needed for core, not for public API +function destroy(err, cb) { + var _this = this; + var readableDestroyed = this._readableState && this._readableState.destroyed; + var writableDestroyed = this._writableState && this._writableState.destroyed; + if (readableDestroyed || writableDestroyed) { + if (cb) { + cb(err); + } else if (err) { + if (!this._writableState) { + process.nextTick(emitErrorNT, this, err); + } else if (!this._writableState.errorEmitted) { + this._writableState.errorEmitted = true; + process.nextTick(emitErrorNT, this, err); + } + } + return this; + } + + // we set destroyed to true before firing error callbacks in order + // to make it re-entrance safe in case destroy() is called within callbacks + + if (this._readableState) { + this._readableState.destroyed = true; + } + + // if this is a duplex stream mark the writable part as destroyed as well + if (this._writableState) { + this._writableState.destroyed = true; + } + this._destroy(err || null, function (err) { + if (!cb && err) { + if (!_this._writableState) { + process.nextTick(emitErrorAndCloseNT, _this, err); + } else if (!_this._writableState.errorEmitted) { + _this._writableState.errorEmitted = true; + process.nextTick(emitErrorAndCloseNT, _this, err); + } else { + process.nextTick(emitCloseNT, _this); + } + } else if (cb) { + process.nextTick(emitCloseNT, _this); + cb(err); + } else { + process.nextTick(emitCloseNT, _this); + } + }); + return this; +} +function emitErrorAndCloseNT(self, err) { + emitErrorNT(self, err); + emitCloseNT(self); +} +function emitCloseNT(self) { + if (self._writableState && !self._writableState.emitClose) return; + if (self._readableState && !self._readableState.emitClose) return; + self.emit('close'); +} +function undestroy() { + if (this._readableState) { + this._readableState.destroyed = false; + this._readableState.reading = false; + this._readableState.ended = false; + this._readableState.endEmitted = false; + } + if (this._writableState) { + this._writableState.destroyed = false; + this._writableState.ended = false; + this._writableState.ending = false; + this._writableState.finalCalled = false; + this._writableState.prefinished = false; + this._writableState.finished = false; + this._writableState.errorEmitted = false; + } +} +function emitErrorNT(self, err) { + self.emit('error', err); +} +function errorOrDestroy(stream, err) { + // We have tests that rely on errors being emitted + // in the same tick, so changing this is semver major. + // For now when you opt-in to autoDestroy we allow + // the error to be emitted nextTick. In a future + // semver major update we should change the default to this. + + var rState = stream._readableState; + var wState = stream._writableState; + if (rState && rState.autoDestroy || wState && wState.autoDestroy) stream.destroy(err);else stream.emit('error', err); +} +module.exports = { + destroy: destroy, + undestroy: undestroy, + errorOrDestroy: errorOrDestroy +}; \ No newline at end of file diff --git a/node_modules/readable-stream/lib/internal/streams/end-of-stream.js b/node_modules/readable-stream/lib/internal/streams/end-of-stream.js new file mode 100644 index 0000000..59c671b --- /dev/null +++ b/node_modules/readable-stream/lib/internal/streams/end-of-stream.js @@ -0,0 +1,86 @@ +// Ported from https://github.com/mafintosh/end-of-stream with +// permission from the author, Mathias Buus (@mafintosh). + +'use strict'; + +var ERR_STREAM_PREMATURE_CLOSE = require('../../../errors').codes.ERR_STREAM_PREMATURE_CLOSE; +function once(callback) { + var called = false; + return function () { + if (called) return; + called = true; + for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { + args[_key] = arguments[_key]; + } + callback.apply(this, args); + }; +} +function noop() {} +function isRequest(stream) { + return stream.setHeader && typeof stream.abort === 'function'; +} +function eos(stream, opts, callback) { + if (typeof opts === 'function') return eos(stream, null, opts); + if (!opts) opts = {}; + callback = once(callback || noop); + var readable = opts.readable || opts.readable !== false && stream.readable; + var writable = opts.writable || opts.writable !== false && stream.writable; + var onlegacyfinish = function onlegacyfinish() { + if (!stream.writable) onfinish(); + }; + var writableEnded = stream._writableState && stream._writableState.finished; + var onfinish = function onfinish() { + writable = false; + writableEnded = true; + if (!readable) callback.call(stream); + }; + var readableEnded = stream._readableState && stream._readableState.endEmitted; + var onend = function onend() { + readable = false; + readableEnded = true; + if (!writable) callback.call(stream); + }; + var onerror = function onerror(err) { + callback.call(stream, err); + }; + var onclose = function onclose() { + var err; + if (readable && !readableEnded) { + if (!stream._readableState || !stream._readableState.ended) err = new ERR_STREAM_PREMATURE_CLOSE(); + return callback.call(stream, err); + } + if (writable && !writableEnded) { + if (!stream._writableState || !stream._writableState.ended) err = new ERR_STREAM_PREMATURE_CLOSE(); + return callback.call(stream, err); + } + }; + var onrequest = function onrequest() { + stream.req.on('finish', onfinish); + }; + if (isRequest(stream)) { + stream.on('complete', onfinish); + stream.on('abort', onclose); + if (stream.req) onrequest();else stream.on('request', onrequest); + } else if (writable && !stream._writableState) { + // legacy streams + stream.on('end', onlegacyfinish); + stream.on('close', onlegacyfinish); + } + stream.on('end', onend); + stream.on('finish', onfinish); + if (opts.error !== false) stream.on('error', onerror); + stream.on('close', onclose); + return function () { + stream.removeListener('complete', onfinish); + stream.removeListener('abort', onclose); + stream.removeListener('request', onrequest); + if (stream.req) stream.req.removeListener('finish', onfinish); + stream.removeListener('end', onlegacyfinish); + stream.removeListener('close', onlegacyfinish); + stream.removeListener('finish', onfinish); + stream.removeListener('end', onend); + stream.removeListener('error', onerror); + stream.removeListener('close', onclose); + }; +} +module.exports = eos; \ No newline at end of file diff --git a/node_modules/readable-stream/lib/internal/streams/from-browser.js b/node_modules/readable-stream/lib/internal/streams/from-browser.js new file mode 100644 index 0000000..a4ce56f --- /dev/null +++ b/node_modules/readable-stream/lib/internal/streams/from-browser.js @@ -0,0 +1,3 @@ +module.exports = function () { + throw new Error('Readable.from is not available in the browser') +}; diff --git a/node_modules/readable-stream/lib/internal/streams/from.js b/node_modules/readable-stream/lib/internal/streams/from.js new file mode 100644 index 0000000..0a34ee9 --- /dev/null +++ b/node_modules/readable-stream/lib/internal/streams/from.js @@ -0,0 +1,52 @@ +'use strict'; + +function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { Promise.resolve(value).then(_next, _throw); } } +function _asyncToGenerator(fn) { return function () { var self = this, args = arguments; return new Promise(function (resolve, reject) { var gen = fn.apply(self, args); function _next(value) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value); } function _throw(err) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err); } _next(undefined); }); }; } +function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; } +function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { _defineProperty(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; } +function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } +function _toPropertyKey(arg) { var key = _toPrimitive(arg, "string"); return typeof key === "symbol" ? key : String(key); } +function _toPrimitive(input, hint) { if (typeof input !== "object" || input === null) return input; var prim = input[Symbol.toPrimitive]; if (prim !== undefined) { var res = prim.call(input, hint || "default"); if (typeof res !== "object") return res; throw new TypeError("@@toPrimitive must return a primitive value."); } return (hint === "string" ? String : Number)(input); } +var ERR_INVALID_ARG_TYPE = require('../../../errors').codes.ERR_INVALID_ARG_TYPE; +function from(Readable, iterable, opts) { + var iterator; + if (iterable && typeof iterable.next === 'function') { + iterator = iterable; + } else if (iterable && iterable[Symbol.asyncIterator]) iterator = iterable[Symbol.asyncIterator]();else if (iterable && iterable[Symbol.iterator]) iterator = iterable[Symbol.iterator]();else throw new ERR_INVALID_ARG_TYPE('iterable', ['Iterable'], iterable); + var readable = new Readable(_objectSpread({ + objectMode: true + }, opts)); + // Reading boolean to protect against _read + // being called before last iteration completion. + var reading = false; + readable._read = function () { + if (!reading) { + reading = true; + next(); + } + }; + function next() { + return _next2.apply(this, arguments); + } + function _next2() { + _next2 = _asyncToGenerator(function* () { + try { + var _yield$iterator$next = yield iterator.next(), + value = _yield$iterator$next.value, + done = _yield$iterator$next.done; + if (done) { + readable.push(null); + } else if (readable.push(yield value)) { + next(); + } else { + reading = false; + } + } catch (err) { + readable.destroy(err); + } + }); + return _next2.apply(this, arguments); + } + return readable; +} +module.exports = from; diff --git a/node_modules/readable-stream/lib/internal/streams/pipeline.js b/node_modules/readable-stream/lib/internal/streams/pipeline.js new file mode 100644 index 0000000..e6f3924 --- /dev/null +++ b/node_modules/readable-stream/lib/internal/streams/pipeline.js @@ -0,0 +1,86 @@ +// Ported from https://github.com/mafintosh/pump with +// permission from the author, Mathias Buus (@mafintosh). + +'use strict'; + +var eos; +function once(callback) { + var called = false; + return function () { + if (called) return; + called = true; + callback.apply(void 0, arguments); + }; +} +var _require$codes = require('../../../errors').codes, + ERR_MISSING_ARGS = _require$codes.ERR_MISSING_ARGS, + ERR_STREAM_DESTROYED = _require$codes.ERR_STREAM_DESTROYED; +function noop(err) { + // Rethrow the error if it exists to avoid swallowing it + if (err) throw err; +} +function isRequest(stream) { + return stream.setHeader && typeof stream.abort === 'function'; +} +function destroyer(stream, reading, writing, callback) { + callback = once(callback); + var closed = false; + stream.on('close', function () { + closed = true; + }); + if (eos === undefined) eos = require('./end-of-stream'); + eos(stream, { + readable: reading, + writable: writing + }, function (err) { + if (err) return callback(err); + closed = true; + callback(); + }); + var destroyed = false; + return function (err) { + if (closed) return; + if (destroyed) return; + destroyed = true; + + // request.destroy just do .end - .abort is what we want + if (isRequest(stream)) return stream.abort(); + if (typeof stream.destroy === 'function') return stream.destroy(); + callback(err || new ERR_STREAM_DESTROYED('pipe')); + }; +} +function call(fn) { + fn(); +} +function pipe(from, to) { + return from.pipe(to); +} +function popCallback(streams) { + if (!streams.length) return noop; + if (typeof streams[streams.length - 1] !== 'function') return noop; + return streams.pop(); +} +function pipeline() { + for (var _len = arguments.length, streams = new Array(_len), _key = 0; _key < _len; _key++) { + streams[_key] = arguments[_key]; + } + var callback = popCallback(streams); + if (Array.isArray(streams[0])) streams = streams[0]; + if (streams.length < 2) { + throw new ERR_MISSING_ARGS('streams'); + } + var error; + var destroys = streams.map(function (stream, i) { + var reading = i < streams.length - 1; + var writing = i > 0; + return destroyer(stream, reading, writing, function (err) { + if (!error) error = err; + if (err) destroys.forEach(call); + if (reading) return; + destroys.forEach(call); + callback(error); + }); + }); + return streams.reduce(pipe); +} +module.exports = pipeline; \ No newline at end of file diff --git a/node_modules/readable-stream/lib/internal/streams/state.js b/node_modules/readable-stream/lib/internal/streams/state.js new file mode 100644 index 0000000..3fbf892 --- /dev/null +++ b/node_modules/readable-stream/lib/internal/streams/state.js @@ -0,0 +1,22 @@ +'use strict'; + +var ERR_INVALID_OPT_VALUE = require('../../../errors').codes.ERR_INVALID_OPT_VALUE; +function highWaterMarkFrom(options, isDuplex, duplexKey) { + return options.highWaterMark != null ? options.highWaterMark : isDuplex ? options[duplexKey] : null; +} +function getHighWaterMark(state, options, duplexKey, isDuplex) { + var hwm = highWaterMarkFrom(options, isDuplex, duplexKey); + if (hwm != null) { + if (!(isFinite(hwm) && Math.floor(hwm) === hwm) || hwm < 0) { + var name = isDuplex ? duplexKey : 'highWaterMark'; + throw new ERR_INVALID_OPT_VALUE(name, hwm); + } + return Math.floor(hwm); + } + + // Default value + return state.objectMode ? 16 : 16 * 1024; +} +module.exports = { + getHighWaterMark: getHighWaterMark +}; \ No newline at end of file diff --git a/node_modules/readable-stream/lib/internal/streams/stream-browser.js b/node_modules/readable-stream/lib/internal/streams/stream-browser.js new file mode 100644 index 0000000..9332a3f --- /dev/null +++ b/node_modules/readable-stream/lib/internal/streams/stream-browser.js @@ -0,0 +1 @@ +module.exports = require('events').EventEmitter; diff --git a/node_modules/readable-stream/lib/internal/streams/stream.js b/node_modules/readable-stream/lib/internal/streams/stream.js new file mode 100644 index 0000000..ce2ad5b --- /dev/null +++ b/node_modules/readable-stream/lib/internal/streams/stream.js @@ -0,0 +1 @@ +module.exports = require('stream'); diff --git a/node_modules/readable-stream/package.json b/node_modules/readable-stream/package.json new file mode 100644 index 0000000..ade59e7 --- /dev/null +++ b/node_modules/readable-stream/package.json @@ -0,0 +1,68 @@ +{ + "name": "readable-stream", + "version": "3.6.2", + "description": "Streams3, a user-land copy of the stream library from Node.js", + "main": "readable.js", + "engines": { + "node": ">= 6" + }, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "devDependencies": { + "@babel/cli": "^7.2.0", + "@babel/core": "^7.2.0", + "@babel/polyfill": "^7.0.0", + "@babel/preset-env": "^7.2.0", + "airtap": "0.0.9", + "assert": "^1.4.0", + "bl": "^2.0.0", + "deep-strict-equal": "^0.2.0", + "events.once": "^2.0.2", + "glob": "^7.1.2", + "gunzip-maybe": "^1.4.1", + "hyperquest": "^2.1.3", + "lolex": "^2.6.0", + "nyc": "^11.0.0", + "pump": "^3.0.0", + "rimraf": "^2.6.2", + "tap": "^12.0.0", + "tape": "^4.9.0", + "tar-fs": "^1.16.2", + "util-promisify": "^2.1.0" + }, + "scripts": { + "test": "tap -J --no-esm test/parallel/*.js test/ours/*.js", + "ci": "TAP=1 tap --no-esm test/parallel/*.js test/ours/*.js | tee test.tap", + "test-browsers": "airtap --sauce-connect --loopback airtap.local -- test/browser.js", + "test-browser-local": "airtap --open --local -- test/browser.js", + "cover": "nyc npm test", + "report": "nyc report --reporter=lcov", + "update-browser-errors": "babel -o errors-browser.js errors.js" + }, + "repository": { + "type": "git", + "url": "git://github.com/nodejs/readable-stream" + }, + "keywords": [ + "readable", + "stream", + "pipe" + ], + "browser": { + "util": false, + "worker_threads": false, + "./errors": "./errors-browser.js", + "./readable.js": "./readable-browser.js", + "./lib/internal/streams/from.js": "./lib/internal/streams/from-browser.js", + "./lib/internal/streams/stream.js": "./lib/internal/streams/stream-browser.js" + }, + "nyc": { + "include": [ + "lib/**.js" + ] + }, + "license": "MIT" +} diff --git a/node_modules/readable-stream/readable-browser.js b/node_modules/readable-stream/readable-browser.js new file mode 100644 index 0000000..adbf60d --- /dev/null +++ b/node_modules/readable-stream/readable-browser.js @@ -0,0 +1,9 @@ +exports = module.exports = require('./lib/_stream_readable.js'); +exports.Stream = exports; +exports.Readable = exports; +exports.Writable = require('./lib/_stream_writable.js'); +exports.Duplex = require('./lib/_stream_duplex.js'); +exports.Transform = require('./lib/_stream_transform.js'); +exports.PassThrough = require('./lib/_stream_passthrough.js'); +exports.finished = require('./lib/internal/streams/end-of-stream.js'); +exports.pipeline = require('./lib/internal/streams/pipeline.js'); diff --git a/node_modules/readable-stream/readable.js b/node_modules/readable-stream/readable.js new file mode 100644 index 0000000..9e0ca12 --- /dev/null +++ b/node_modules/readable-stream/readable.js @@ -0,0 +1,16 @@ +var Stream = require('stream'); +if (process.env.READABLE_STREAM === 'disable' && Stream) { + module.exports = Stream.Readable; + Object.assign(module.exports, Stream); + module.exports.Stream = Stream; +} else { + exports = module.exports = require('./lib/_stream_readable.js'); + exports.Stream = Stream || exports; + exports.Readable = exports; + exports.Writable = require('./lib/_stream_writable.js'); + exports.Duplex = require('./lib/_stream_duplex.js'); + exports.Transform = require('./lib/_stream_transform.js'); + exports.PassThrough = require('./lib/_stream_passthrough.js'); + exports.finished = require('./lib/internal/streams/end-of-stream.js'); + exports.pipeline = require('./lib/internal/streams/pipeline.js'); +} diff --git a/node_modules/ripemd160/CHANGELOG.md b/node_modules/ripemd160/CHANGELOG.md new file mode 100644 index 0000000..91ba912 --- /dev/null +++ b/node_modules/ripemd160/CHANGELOG.md @@ -0,0 +1,72 @@ +2.0.1 / 2016-06-22 +------------------ +- added LICENSE file. + +2.0.0 / 2016-04-11 +------------------ +- rewritten, license change BSD-3 to MIT. [#13][#13] +- stream support [#13][#13] + +1.0.1 / 2015-05-05 +------------------ +- standard formatting + +1.0.0 / 2015-01-14 +------------------ +- updated dev deps +- added more test fixtures +- updated readme with usage, testing, etc +- moved from https://github.com/cryptocoinjs/ripemd160 to https://github.com/crypto-browserify/ripemd160 + +0.2.1 / 2014-12-31 +------------------ +- made license clear in `package.json` +- deleted `Makefile`, moved targets to `package.json` +- removed `terst` for `assert` + +0.2.0 / 2014-03-09 +------------------ +* removed bower.json and component.json +* changed 4 spacing to 2 +* returns `Buffer` type now, input must be Array, Uint8Array, Buffer, or string +* remove deps: `convert-hex` and `convert-string` + +0.1.0 / 2013-11-20 +------------------ +* changed package name +* removed AMD support + +0.0.2 / 2013-11-06 +------------------ +* fixed component.json file + +0.0.1 / 2013-11-03 +------------------ +* initial release + + +[#13]: https://github.com/crypto-browserify/ripemd160/pull/13 + +[#12]: https://github.com/crypto-browserify/ripemd160/pull/12 + +[#11]: https://github.com/crypto-browserify/ripemd160/pull/11 + +[#10]: https://github.com/crypto-browserify/ripemd160/pull/10 + +[#9]: https://github.com/crypto-browserify/ripemd160/pull/9 + +[#8]: https://github.com/crypto-browserify/ripemd160/issues/8 + +[#7]: https://github.com/crypto-browserify/ripemd160/pull/7 + +[#6]: https://github.com/crypto-browserify/ripemd160/pull/6 + +[#5]: https://github.com/crypto-browserify/ripemd160/issues/5 + +[#4]: https://github.com/crypto-browserify/ripemd160/pull/4 + +[#3]: https://github.com/crypto-browserify/ripemd160/pull/3 + +[#2]: https://github.com/crypto-browserify/ripemd160/pull/2 + +[#1]: https://github.com/crypto-browserify/ripemd160/pull/1 diff --git a/node_modules/ripemd160/LICENSE b/node_modules/ripemd160/LICENSE new file mode 100644 index 0000000..cbe3544 --- /dev/null +++ b/node_modules/ripemd160/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2016 crypto-browserify + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/node_modules/ripemd160/README.md b/node_modules/ripemd160/README.md new file mode 100644 index 0000000..056c6a2 --- /dev/null +++ b/node_modules/ripemd160/README.md @@ -0,0 +1,27 @@ +# ripemd160 + +[![NPM Package](https://img.shields.io/npm/v/ripemd160.svg?style=flat-square)](https://www.npmjs.org/package/ripemd160) +[![Build Status](https://img.shields.io/travis/crypto-browserify/ripemd160.svg?branch=master&style=flat-square)](https://travis-ci.org/crypto-browserify/ripemd160) +[![Dependency status](https://img.shields.io/david/crypto-browserify/ripemd160.svg?style=flat-square)](https://david-dm.org/crypto-browserify/ripemd160#info=dependencies) + +[![js-standard-style](https://cdn.rawgit.com/feross/standard/master/badge.svg)](https://github.com/feross/standard) + +Node style `ripemd160` on pure JavaScript. + +## Example + +```js +var RIPEMD160 = require('ripemd160') + +console.log(new RIPEMD160().update('42').digest('hex')) +// => 0df020ba32aa9b8b904471ff582ce6b579bf8bc8 + +var ripemd160stream = new RIPEMD160() +ripemd160stream.end('42') +console.log(ripemd160stream.read().toString('hex')) +// => 0df020ba32aa9b8b904471ff582ce6b579bf8bc8 +``` + +## LICENSE + +MIT diff --git a/node_modules/ripemd160/index.js b/node_modules/ripemd160/index.js new file mode 100644 index 0000000..870de9c --- /dev/null +++ b/node_modules/ripemd160/index.js @@ -0,0 +1,163 @@ +'use strict' +var Buffer = require('buffer').Buffer +var inherits = require('inherits') +var HashBase = require('hash-base') + +var ARRAY16 = new Array(16) + +var zl = [ + 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, + 7, 4, 13, 1, 10, 6, 15, 3, 12, 0, 9, 5, 2, 14, 11, 8, + 3, 10, 14, 4, 9, 15, 8, 1, 2, 7, 0, 6, 13, 11, 5, 12, + 1, 9, 11, 10, 0, 8, 12, 4, 13, 3, 7, 15, 14, 5, 6, 2, + 4, 0, 5, 9, 7, 12, 2, 10, 14, 1, 3, 8, 11, 6, 15, 13 +] + +var zr = [ + 5, 14, 7, 0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12, + 6, 11, 3, 7, 0, 13, 5, 10, 14, 15, 8, 12, 4, 9, 1, 2, + 15, 5, 1, 3, 7, 14, 6, 9, 11, 8, 12, 2, 10, 0, 4, 13, + 8, 6, 4, 1, 3, 11, 15, 0, 5, 12, 2, 13, 9, 7, 10, 14, + 12, 15, 10, 4, 1, 5, 8, 7, 6, 2, 13, 14, 0, 3, 9, 11 +] + +var sl = [ + 11, 14, 15, 12, 5, 8, 7, 9, 11, 13, 14, 15, 6, 7, 9, 8, + 7, 6, 8, 13, 11, 9, 7, 15, 7, 12, 15, 9, 11, 7, 13, 12, + 11, 13, 6, 7, 14, 9, 13, 15, 14, 8, 13, 6, 5, 12, 7, 5, + 11, 12, 14, 15, 14, 15, 9, 8, 9, 14, 5, 6, 8, 6, 5, 12, + 9, 15, 5, 11, 6, 8, 13, 12, 5, 12, 13, 14, 11, 8, 5, 6 +] + +var sr = [ + 8, 9, 9, 11, 13, 15, 15, 5, 7, 7, 8, 11, 14, 14, 12, 6, + 9, 13, 15, 7, 12, 8, 9, 11, 7, 7, 12, 7, 6, 15, 13, 11, + 9, 7, 15, 11, 8, 6, 6, 14, 12, 13, 5, 14, 13, 13, 7, 5, + 15, 5, 8, 11, 14, 14, 6, 14, 6, 9, 12, 9, 12, 5, 15, 8, + 8, 5, 12, 9, 12, 5, 14, 6, 8, 13, 6, 5, 15, 13, 11, 11 +] + +var hl = [0x00000000, 0x5a827999, 0x6ed9eba1, 0x8f1bbcdc, 0xa953fd4e] +var hr = [0x50a28be6, 0x5c4dd124, 0x6d703ef3, 0x7a6d76e9, 0x00000000] + +function RIPEMD160 () { + HashBase.call(this, 64) + + // state + this._a = 0x67452301 + this._b = 0xefcdab89 + this._c = 0x98badcfe + this._d = 0x10325476 + this._e = 0xc3d2e1f0 +} + +inherits(RIPEMD160, HashBase) + +RIPEMD160.prototype._update = function () { + var words = ARRAY16 + for (var j = 0; j < 16; ++j) words[j] = this._block.readInt32LE(j * 4) + + var al = this._a | 0 + var bl = this._b | 0 + var cl = this._c | 0 + var dl = this._d | 0 + var el = this._e | 0 + + var ar = this._a | 0 + var br = this._b | 0 + var cr = this._c | 0 + var dr = this._d | 0 + var er = this._e | 0 + + // computation + for (var i = 0; i < 80; i += 1) { + var tl + var tr + if (i < 16) { + tl = fn1(al, bl, cl, dl, el, words[zl[i]], hl[0], sl[i]) + tr = fn5(ar, br, cr, dr, er, words[zr[i]], hr[0], sr[i]) + } else if (i < 32) { + tl = fn2(al, bl, cl, dl, el, words[zl[i]], hl[1], sl[i]) + tr = fn4(ar, br, cr, dr, er, words[zr[i]], hr[1], sr[i]) + } else if (i < 48) { + tl = fn3(al, bl, cl, dl, el, words[zl[i]], hl[2], sl[i]) + tr = fn3(ar, br, cr, dr, er, words[zr[i]], hr[2], sr[i]) + } else if (i < 64) { + tl = fn4(al, bl, cl, dl, el, words[zl[i]], hl[3], sl[i]) + tr = fn2(ar, br, cr, dr, er, words[zr[i]], hr[3], sr[i]) + } else { // if (i<80) { + tl = fn5(al, bl, cl, dl, el, words[zl[i]], hl[4], sl[i]) + tr = fn1(ar, br, cr, dr, er, words[zr[i]], hr[4], sr[i]) + } + + al = el + el = dl + dl = rotl(cl, 10) + cl = bl + bl = tl + + ar = er + er = dr + dr = rotl(cr, 10) + cr = br + br = tr + } + + // update state + var t = (this._b + cl + dr) | 0 + this._b = (this._c + dl + er) | 0 + this._c = (this._d + el + ar) | 0 + this._d = (this._e + al + br) | 0 + this._e = (this._a + bl + cr) | 0 + this._a = t +} + +RIPEMD160.prototype._digest = function () { + // create padding and handle blocks + this._block[this._blockOffset++] = 0x80 + if (this._blockOffset > 56) { + this._block.fill(0, this._blockOffset, 64) + this._update() + this._blockOffset = 0 + } + + this._block.fill(0, this._blockOffset, 56) + this._block.writeUInt32LE(this._length[0], 56) + this._block.writeUInt32LE(this._length[1], 60) + this._update() + + // produce result + var buffer = Buffer.alloc ? Buffer.alloc(20) : new Buffer(20) + buffer.writeInt32LE(this._a, 0) + buffer.writeInt32LE(this._b, 4) + buffer.writeInt32LE(this._c, 8) + buffer.writeInt32LE(this._d, 12) + buffer.writeInt32LE(this._e, 16) + return buffer +} + +function rotl (x, n) { + return (x << n) | (x >>> (32 - n)) +} + +function fn1 (a, b, c, d, e, m, k, s) { + return (rotl((a + (b ^ c ^ d) + m + k) | 0, s) + e) | 0 +} + +function fn2 (a, b, c, d, e, m, k, s) { + return (rotl((a + ((b & c) | ((~b) & d)) + m + k) | 0, s) + e) | 0 +} + +function fn3 (a, b, c, d, e, m, k, s) { + return (rotl((a + ((b | (~c)) ^ d) + m + k) | 0, s) + e) | 0 +} + +function fn4 (a, b, c, d, e, m, k, s) { + return (rotl((a + ((b & d) | (c & (~d))) + m + k) | 0, s) + e) | 0 +} + +function fn5 (a, b, c, d, e, m, k, s) { + return (rotl((a + (b ^ (c | (~d))) + m + k) | 0, s) + e) | 0 +} + +module.exports = RIPEMD160 diff --git a/node_modules/ripemd160/package.json b/node_modules/ripemd160/package.json new file mode 100644 index 0000000..70d3c60 --- /dev/null +++ b/node_modules/ripemd160/package.json @@ -0,0 +1,37 @@ +{ + "name": "ripemd160", + "version": "2.0.2", + "description": "Compute ripemd160 of bytes or strings.", + "keywords": [ + "string", + "strings", + "ripemd160", + "ripe160", + "bitcoin", + "bytes", + "cryptography" + ], + "license": "MIT", + "files": [ + "index.js" + ], + "main": "./index", + "repository": { + "url": "https://github.com/crypto-browserify/ripemd160", + "type": "git" + }, + "scripts": { + "lint": "standard", + "test": "npm run lint && npm run unit", + "unit": "node test/*.js" + }, + "dependencies": { + "hash-base": "^3.0.0", + "inherits": "^2.0.1" + }, + "devDependencies": { + "hash-test-vectors": "^1.3.2", + "standard": "^6.0.7", + "tape": "^4.5.1" + } +} diff --git a/node_modules/safe-buffer/LICENSE b/node_modules/safe-buffer/LICENSE new file mode 100644 index 0000000..0c068ce --- /dev/null +++ b/node_modules/safe-buffer/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) Feross Aboukhadijeh + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/node_modules/safe-buffer/README.md b/node_modules/safe-buffer/README.md new file mode 100644 index 0000000..e9a81af --- /dev/null +++ b/node_modules/safe-buffer/README.md @@ -0,0 +1,584 @@ +# safe-buffer [![travis][travis-image]][travis-url] [![npm][npm-image]][npm-url] [![downloads][downloads-image]][downloads-url] [![javascript style guide][standard-image]][standard-url] + +[travis-image]: https://img.shields.io/travis/feross/safe-buffer/master.svg +[travis-url]: https://travis-ci.org/feross/safe-buffer +[npm-image]: https://img.shields.io/npm/v/safe-buffer.svg +[npm-url]: https://npmjs.org/package/safe-buffer +[downloads-image]: https://img.shields.io/npm/dm/safe-buffer.svg +[downloads-url]: https://npmjs.org/package/safe-buffer +[standard-image]: https://img.shields.io/badge/code_style-standard-brightgreen.svg +[standard-url]: https://standardjs.com + +#### Safer Node.js Buffer API + +**Use the new Node.js Buffer APIs (`Buffer.from`, `Buffer.alloc`, +`Buffer.allocUnsafe`, `Buffer.allocUnsafeSlow`) in all versions of Node.js.** + +**Uses the built-in implementation when available.** + +## install + +``` +npm install safe-buffer +``` + +## usage + +The goal of this package is to provide a safe replacement for the node.js `Buffer`. + +It's a drop-in replacement for `Buffer`. You can use it by adding one `require` line to +the top of your node.js modules: + +```js +var Buffer = require('safe-buffer').Buffer + +// Existing buffer code will continue to work without issues: + +new Buffer('hey', 'utf8') +new Buffer([1, 2, 3], 'utf8') +new Buffer(obj) +new Buffer(16) // create an uninitialized buffer (potentially unsafe) + +// But you can use these new explicit APIs to make clear what you want: + +Buffer.from('hey', 'utf8') // convert from many types to a Buffer +Buffer.alloc(16) // create a zero-filled buffer (safe) +Buffer.allocUnsafe(16) // create an uninitialized buffer (potentially unsafe) +``` + +## api + +### Class Method: Buffer.from(array) + + +* `array` {Array} + +Allocates a new `Buffer` using an `array` of octets. + +```js +const buf = Buffer.from([0x62,0x75,0x66,0x66,0x65,0x72]); + // creates a new Buffer containing ASCII bytes + // ['b','u','f','f','e','r'] +``` + +A `TypeError` will be thrown if `array` is not an `Array`. + +### Class Method: Buffer.from(arrayBuffer[, byteOffset[, length]]) + + +* `arrayBuffer` {ArrayBuffer} The `.buffer` property of a `TypedArray` or + a `new ArrayBuffer()` +* `byteOffset` {Number} Default: `0` +* `length` {Number} Default: `arrayBuffer.length - byteOffset` + +When passed a reference to the `.buffer` property of a `TypedArray` instance, +the newly created `Buffer` will share the same allocated memory as the +TypedArray. + +```js +const arr = new Uint16Array(2); +arr[0] = 5000; +arr[1] = 4000; + +const buf = Buffer.from(arr.buffer); // shares the memory with arr; + +console.log(buf); + // Prints: + +// changing the TypedArray changes the Buffer also +arr[1] = 6000; + +console.log(buf); + // Prints: +``` + +The optional `byteOffset` and `length` arguments specify a memory range within +the `arrayBuffer` that will be shared by the `Buffer`. + +```js +const ab = new ArrayBuffer(10); +const buf = Buffer.from(ab, 0, 2); +console.log(buf.length); + // Prints: 2 +``` + +A `TypeError` will be thrown if `arrayBuffer` is not an `ArrayBuffer`. + +### Class Method: Buffer.from(buffer) + + +* `buffer` {Buffer} + +Copies the passed `buffer` data onto a new `Buffer` instance. + +```js +const buf1 = Buffer.from('buffer'); +const buf2 = Buffer.from(buf1); + +buf1[0] = 0x61; +console.log(buf1.toString()); + // 'auffer' +console.log(buf2.toString()); + // 'buffer' (copy is not changed) +``` + +A `TypeError` will be thrown if `buffer` is not a `Buffer`. + +### Class Method: Buffer.from(str[, encoding]) + + +* `str` {String} String to encode. +* `encoding` {String} Encoding to use, Default: `'utf8'` + +Creates a new `Buffer` containing the given JavaScript string `str`. If +provided, the `encoding` parameter identifies the character encoding. +If not provided, `encoding` defaults to `'utf8'`. + +```js +const buf1 = Buffer.from('this is a tést'); +console.log(buf1.toString()); + // prints: this is a tést +console.log(buf1.toString('ascii')); + // prints: this is a tC)st + +const buf2 = Buffer.from('7468697320697320612074c3a97374', 'hex'); +console.log(buf2.toString()); + // prints: this is a tést +``` + +A `TypeError` will be thrown if `str` is not a string. + +### Class Method: Buffer.alloc(size[, fill[, encoding]]) + + +* `size` {Number} +* `fill` {Value} Default: `undefined` +* `encoding` {String} Default: `utf8` + +Allocates a new `Buffer` of `size` bytes. If `fill` is `undefined`, the +`Buffer` will be *zero-filled*. + +```js +const buf = Buffer.alloc(5); +console.log(buf); + // +``` + +The `size` must be less than or equal to the value of +`require('buffer').kMaxLength` (on 64-bit architectures, `kMaxLength` is +`(2^31)-1`). Otherwise, a [`RangeError`][] is thrown. A zero-length Buffer will +be created if a `size` less than or equal to 0 is specified. + +If `fill` is specified, the allocated `Buffer` will be initialized by calling +`buf.fill(fill)`. See [`buf.fill()`][] for more information. + +```js +const buf = Buffer.alloc(5, 'a'); +console.log(buf); + // +``` + +If both `fill` and `encoding` are specified, the allocated `Buffer` will be +initialized by calling `buf.fill(fill, encoding)`. For example: + +```js +const buf = Buffer.alloc(11, 'aGVsbG8gd29ybGQ=', 'base64'); +console.log(buf); + // +``` + +Calling `Buffer.alloc(size)` can be significantly slower than the alternative +`Buffer.allocUnsafe(size)` but ensures that the newly created `Buffer` instance +contents will *never contain sensitive data*. + +A `TypeError` will be thrown if `size` is not a number. + +### Class Method: Buffer.allocUnsafe(size) + + +* `size` {Number} + +Allocates a new *non-zero-filled* `Buffer` of `size` bytes. The `size` must +be less than or equal to the value of `require('buffer').kMaxLength` (on 64-bit +architectures, `kMaxLength` is `(2^31)-1`). Otherwise, a [`RangeError`][] is +thrown. A zero-length Buffer will be created if a `size` less than or equal to +0 is specified. + +The underlying memory for `Buffer` instances created in this way is *not +initialized*. The contents of the newly created `Buffer` are unknown and +*may contain sensitive data*. Use [`buf.fill(0)`][] to initialize such +`Buffer` instances to zeroes. + +```js +const buf = Buffer.allocUnsafe(5); +console.log(buf); + // + // (octets will be different, every time) +buf.fill(0); +console.log(buf); + // +``` + +A `TypeError` will be thrown if `size` is not a number. + +Note that the `Buffer` module pre-allocates an internal `Buffer` instance of +size `Buffer.poolSize` that is used as a pool for the fast allocation of new +`Buffer` instances created using `Buffer.allocUnsafe(size)` (and the deprecated +`new Buffer(size)` constructor) only when `size` is less than or equal to +`Buffer.poolSize >> 1` (floor of `Buffer.poolSize` divided by two). The default +value of `Buffer.poolSize` is `8192` but can be modified. + +Use of this pre-allocated internal memory pool is a key difference between +calling `Buffer.alloc(size, fill)` vs. `Buffer.allocUnsafe(size).fill(fill)`. +Specifically, `Buffer.alloc(size, fill)` will *never* use the internal Buffer +pool, while `Buffer.allocUnsafe(size).fill(fill)` *will* use the internal +Buffer pool if `size` is less than or equal to half `Buffer.poolSize`. The +difference is subtle but can be important when an application requires the +additional performance that `Buffer.allocUnsafe(size)` provides. + +### Class Method: Buffer.allocUnsafeSlow(size) + + +* `size` {Number} + +Allocates a new *non-zero-filled* and non-pooled `Buffer` of `size` bytes. The +`size` must be less than or equal to the value of +`require('buffer').kMaxLength` (on 64-bit architectures, `kMaxLength` is +`(2^31)-1`). Otherwise, a [`RangeError`][] is thrown. A zero-length Buffer will +be created if a `size` less than or equal to 0 is specified. + +The underlying memory for `Buffer` instances created in this way is *not +initialized*. The contents of the newly created `Buffer` are unknown and +*may contain sensitive data*. Use [`buf.fill(0)`][] to initialize such +`Buffer` instances to zeroes. + +When using `Buffer.allocUnsafe()` to allocate new `Buffer` instances, +allocations under 4KB are, by default, sliced from a single pre-allocated +`Buffer`. This allows applications to avoid the garbage collection overhead of +creating many individually allocated Buffers. This approach improves both +performance and memory usage by eliminating the need to track and cleanup as +many `Persistent` objects. + +However, in the case where a developer may need to retain a small chunk of +memory from a pool for an indeterminate amount of time, it may be appropriate +to create an un-pooled Buffer instance using `Buffer.allocUnsafeSlow()` then +copy out the relevant bits. + +```js +// need to keep around a few small chunks of memory +const store = []; + +socket.on('readable', () => { + const data = socket.read(); + // allocate for retained data + const sb = Buffer.allocUnsafeSlow(10); + // copy the data into the new allocation + data.copy(sb, 0, 0, 10); + store.push(sb); +}); +``` + +Use of `Buffer.allocUnsafeSlow()` should be used only as a last resort *after* +a developer has observed undue memory retention in their applications. + +A `TypeError` will be thrown if `size` is not a number. + +### All the Rest + +The rest of the `Buffer` API is exactly the same as in node.js. +[See the docs](https://nodejs.org/api/buffer.html). + + +## Related links + +- [Node.js issue: Buffer(number) is unsafe](https://github.com/nodejs/node/issues/4660) +- [Node.js Enhancement Proposal: Buffer.from/Buffer.alloc/Buffer.zalloc/Buffer() soft-deprecate](https://github.com/nodejs/node-eps/pull/4) + +## Why is `Buffer` unsafe? + +Today, the node.js `Buffer` constructor is overloaded to handle many different argument +types like `String`, `Array`, `Object`, `TypedArrayView` (`Uint8Array`, etc.), +`ArrayBuffer`, and also `Number`. + +The API is optimized for convenience: you can throw any type at it, and it will try to do +what you want. + +Because the Buffer constructor is so powerful, you often see code like this: + +```js +// Convert UTF-8 strings to hex +function toHex (str) { + return new Buffer(str).toString('hex') +} +``` + +***But what happens if `toHex` is called with a `Number` argument?*** + +### Remote Memory Disclosure + +If an attacker can make your program call the `Buffer` constructor with a `Number` +argument, then they can make it allocate uninitialized memory from the node.js process. +This could potentially disclose TLS private keys, user data, or database passwords. + +When the `Buffer` constructor is passed a `Number` argument, it returns an +**UNINITIALIZED** block of memory of the specified `size`. When you create a `Buffer` like +this, you **MUST** overwrite the contents before returning it to the user. + +From the [node.js docs](https://nodejs.org/api/buffer.html#buffer_new_buffer_size): + +> `new Buffer(size)` +> +> - `size` Number +> +> The underlying memory for `Buffer` instances created in this way is not initialized. +> **The contents of a newly created `Buffer` are unknown and could contain sensitive +> data.** Use `buf.fill(0)` to initialize a Buffer to zeroes. + +(Emphasis our own.) + +Whenever the programmer intended to create an uninitialized `Buffer` you often see code +like this: + +```js +var buf = new Buffer(16) + +// Immediately overwrite the uninitialized buffer with data from another buffer +for (var i = 0; i < buf.length; i++) { + buf[i] = otherBuf[i] +} +``` + + +### Would this ever be a problem in real code? + +Yes. It's surprisingly common to forget to check the type of your variables in a +dynamically-typed language like JavaScript. + +Usually the consequences of assuming the wrong type is that your program crashes with an +uncaught exception. But the failure mode for forgetting to check the type of arguments to +the `Buffer` constructor is more catastrophic. + +Here's an example of a vulnerable service that takes a JSON payload and converts it to +hex: + +```js +// Take a JSON payload {str: "some string"} and convert it to hex +var server = http.createServer(function (req, res) { + var data = '' + req.setEncoding('utf8') + req.on('data', function (chunk) { + data += chunk + }) + req.on('end', function () { + var body = JSON.parse(data) + res.end(new Buffer(body.str).toString('hex')) + }) +}) + +server.listen(8080) +``` + +In this example, an http client just has to send: + +```json +{ + "str": 1000 +} +``` + +and it will get back 1,000 bytes of uninitialized memory from the server. + +This is a very serious bug. It's similar in severity to the +[the Heartbleed bug](http://heartbleed.com/) that allowed disclosure of OpenSSL process +memory by remote attackers. + + +### Which real-world packages were vulnerable? + +#### [`bittorrent-dht`](https://www.npmjs.com/package/bittorrent-dht) + +[Mathias Buus](https://github.com/mafintosh) and I +([Feross Aboukhadijeh](http://feross.org/)) found this issue in one of our own packages, +[`bittorrent-dht`](https://www.npmjs.com/package/bittorrent-dht). The bug would allow +anyone on the internet to send a series of messages to a user of `bittorrent-dht` and get +them to reveal 20 bytes at a time of uninitialized memory from the node.js process. + +Here's +[the commit](https://github.com/feross/bittorrent-dht/commit/6c7da04025d5633699800a99ec3fbadf70ad35b8) +that fixed it. We released a new fixed version, created a +[Node Security Project disclosure](https://nodesecurity.io/advisories/68), and deprecated all +vulnerable versions on npm so users will get a warning to upgrade to a newer version. + +#### [`ws`](https://www.npmjs.com/package/ws) + +That got us wondering if there were other vulnerable packages. Sure enough, within a short +period of time, we found the same issue in [`ws`](https://www.npmjs.com/package/ws), the +most popular WebSocket implementation in node.js. + +If certain APIs were called with `Number` parameters instead of `String` or `Buffer` as +expected, then uninitialized server memory would be disclosed to the remote peer. + +These were the vulnerable methods: + +```js +socket.send(number) +socket.ping(number) +socket.pong(number) +``` + +Here's a vulnerable socket server with some echo functionality: + +```js +server.on('connection', function (socket) { + socket.on('message', function (message) { + message = JSON.parse(message) + if (message.type === 'echo') { + socket.send(message.data) // send back the user's message + } + }) +}) +``` + +`socket.send(number)` called on the server, will disclose server memory. + +Here's [the release](https://github.com/websockets/ws/releases/tag/1.0.1) where the issue +was fixed, with a more detailed explanation. Props to +[Arnout Kazemier](https://github.com/3rd-Eden) for the quick fix. Here's the +[Node Security Project disclosure](https://nodesecurity.io/advisories/67). + + +### What's the solution? + +It's important that node.js offers a fast way to get memory otherwise performance-critical +applications would needlessly get a lot slower. + +But we need a better way to *signal our intent* as programmers. **When we want +uninitialized memory, we should request it explicitly.** + +Sensitive functionality should not be packed into a developer-friendly API that loosely +accepts many different types. This type of API encourages the lazy practice of passing +variables in without checking the type very carefully. + +#### A new API: `Buffer.allocUnsafe(number)` + +The functionality of creating buffers with uninitialized memory should be part of another +API. We propose `Buffer.allocUnsafe(number)`. This way, it's not part of an API that +frequently gets user input of all sorts of different types passed into it. + +```js +var buf = Buffer.allocUnsafe(16) // careful, uninitialized memory! + +// Immediately overwrite the uninitialized buffer with data from another buffer +for (var i = 0; i < buf.length; i++) { + buf[i] = otherBuf[i] +} +``` + + +### How do we fix node.js core? + +We sent [a PR to node.js core](https://github.com/nodejs/node/pull/4514) (merged as +`semver-major`) which defends against one case: + +```js +var str = 16 +new Buffer(str, 'utf8') +``` + +In this situation, it's implied that the programmer intended the first argument to be a +string, since they passed an encoding as a second argument. Today, node.js will allocate +uninitialized memory in the case of `new Buffer(number, encoding)`, which is probably not +what the programmer intended. + +But this is only a partial solution, since if the programmer does `new Buffer(variable)` +(without an `encoding` parameter) there's no way to know what they intended. If `variable` +is sometimes a number, then uninitialized memory will sometimes be returned. + +### What's the real long-term fix? + +We could deprecate and remove `new Buffer(number)` and use `Buffer.allocUnsafe(number)` when +we need uninitialized memory. But that would break 1000s of packages. + +~~We believe the best solution is to:~~ + +~~1. Change `new Buffer(number)` to return safe, zeroed-out memory~~ + +~~2. Create a new API for creating uninitialized Buffers. We propose: `Buffer.allocUnsafe(number)`~~ + +#### Update + +We now support adding three new APIs: + +- `Buffer.from(value)` - convert from any type to a buffer +- `Buffer.alloc(size)` - create a zero-filled buffer +- `Buffer.allocUnsafe(size)` - create an uninitialized buffer with given size + +This solves the core problem that affected `ws` and `bittorrent-dht` which is +`Buffer(variable)` getting tricked into taking a number argument. + +This way, existing code continues working and the impact on the npm ecosystem will be +minimal. Over time, npm maintainers can migrate performance-critical code to use +`Buffer.allocUnsafe(number)` instead of `new Buffer(number)`. + + +### Conclusion + +We think there's a serious design issue with the `Buffer` API as it exists today. It +promotes insecure software by putting high-risk functionality into a convenient API +with friendly "developer ergonomics". + +This wasn't merely a theoretical exercise because we found the issue in some of the +most popular npm packages. + +Fortunately, there's an easy fix that can be applied today. Use `safe-buffer` in place of +`buffer`. + +```js +var Buffer = require('safe-buffer').Buffer +``` + +Eventually, we hope that node.js core can switch to this new, safer behavior. We believe +the impact on the ecosystem would be minimal since it's not a breaking change. +Well-maintained, popular packages would be updated to use `Buffer.alloc` quickly, while +older, insecure packages would magically become safe from this attack vector. + + +## links + +- [Node.js PR: buffer: throw if both length and enc are passed](https://github.com/nodejs/node/pull/4514) +- [Node Security Project disclosure for `ws`](https://nodesecurity.io/advisories/67) +- [Node Security Project disclosure for`bittorrent-dht`](https://nodesecurity.io/advisories/68) + + +## credit + +The original issues in `bittorrent-dht` +([disclosure](https://nodesecurity.io/advisories/68)) and +`ws` ([disclosure](https://nodesecurity.io/advisories/67)) were discovered by +[Mathias Buus](https://github.com/mafintosh) and +[Feross Aboukhadijeh](http://feross.org/). + +Thanks to [Adam Baldwin](https://github.com/evilpacket) for helping disclose these issues +and for his work running the [Node Security Project](https://nodesecurity.io/). + +Thanks to [John Hiesey](https://github.com/jhiesey) for proofreading this README and +auditing the code. + + +## license + +MIT. Copyright (C) [Feross Aboukhadijeh](http://feross.org) diff --git a/node_modules/safe-buffer/index.d.ts b/node_modules/safe-buffer/index.d.ts new file mode 100644 index 0000000..e9fed80 --- /dev/null +++ b/node_modules/safe-buffer/index.d.ts @@ -0,0 +1,187 @@ +declare module "safe-buffer" { + export class Buffer { + length: number + write(string: string, offset?: number, length?: number, encoding?: string): number; + toString(encoding?: string, start?: number, end?: number): string; + toJSON(): { type: 'Buffer', data: any[] }; + equals(otherBuffer: Buffer): boolean; + compare(otherBuffer: Buffer, targetStart?: number, targetEnd?: number, sourceStart?: number, sourceEnd?: number): number; + copy(targetBuffer: Buffer, targetStart?: number, sourceStart?: number, sourceEnd?: number): number; + slice(start?: number, end?: number): Buffer; + writeUIntLE(value: number, offset: number, byteLength: number, noAssert?: boolean): number; + writeUIntBE(value: number, offset: number, byteLength: number, noAssert?: boolean): number; + writeIntLE(value: number, offset: number, byteLength: number, noAssert?: boolean): number; + writeIntBE(value: number, offset: number, byteLength: number, noAssert?: boolean): number; + readUIntLE(offset: number, byteLength: number, noAssert?: boolean): number; + readUIntBE(offset: number, byteLength: number, noAssert?: boolean): number; + readIntLE(offset: number, byteLength: number, noAssert?: boolean): number; + readIntBE(offset: number, byteLength: number, noAssert?: boolean): number; + readUInt8(offset: number, noAssert?: boolean): number; + readUInt16LE(offset: number, noAssert?: boolean): number; + readUInt16BE(offset: number, noAssert?: boolean): number; + readUInt32LE(offset: number, noAssert?: boolean): number; + readUInt32BE(offset: number, noAssert?: boolean): number; + readInt8(offset: number, noAssert?: boolean): number; + readInt16LE(offset: number, noAssert?: boolean): number; + readInt16BE(offset: number, noAssert?: boolean): number; + readInt32LE(offset: number, noAssert?: boolean): number; + readInt32BE(offset: number, noAssert?: boolean): number; + readFloatLE(offset: number, noAssert?: boolean): number; + readFloatBE(offset: number, noAssert?: boolean): number; + readDoubleLE(offset: number, noAssert?: boolean): number; + readDoubleBE(offset: number, noAssert?: boolean): number; + swap16(): Buffer; + swap32(): Buffer; + swap64(): Buffer; + writeUInt8(value: number, offset: number, noAssert?: boolean): number; + writeUInt16LE(value: number, offset: number, noAssert?: boolean): number; + writeUInt16BE(value: number, offset: number, noAssert?: boolean): number; + writeUInt32LE(value: number, offset: number, noAssert?: boolean): number; + writeUInt32BE(value: number, offset: number, noAssert?: boolean): number; + writeInt8(value: number, offset: number, noAssert?: boolean): number; + writeInt16LE(value: number, offset: number, noAssert?: boolean): number; + writeInt16BE(value: number, offset: number, noAssert?: boolean): number; + writeInt32LE(value: number, offset: number, noAssert?: boolean): number; + writeInt32BE(value: number, offset: number, noAssert?: boolean): number; + writeFloatLE(value: number, offset: number, noAssert?: boolean): number; + writeFloatBE(value: number, offset: number, noAssert?: boolean): number; + writeDoubleLE(value: number, offset: number, noAssert?: boolean): number; + writeDoubleBE(value: number, offset: number, noAssert?: boolean): number; + fill(value: any, offset?: number, end?: number): this; + indexOf(value: string | number | Buffer, byteOffset?: number, encoding?: string): number; + lastIndexOf(value: string | number | Buffer, byteOffset?: number, encoding?: string): number; + includes(value: string | number | Buffer, byteOffset?: number, encoding?: string): boolean; + + /** + * Allocates a new buffer containing the given {str}. + * + * @param str String to store in buffer. + * @param encoding encoding to use, optional. Default is 'utf8' + */ + constructor (str: string, encoding?: string); + /** + * Allocates a new buffer of {size} octets. + * + * @param size count of octets to allocate. + */ + constructor (size: number); + /** + * Allocates a new buffer containing the given {array} of octets. + * + * @param array The octets to store. + */ + constructor (array: Uint8Array); + /** + * Produces a Buffer backed by the same allocated memory as + * the given {ArrayBuffer}. + * + * + * @param arrayBuffer The ArrayBuffer with which to share memory. + */ + constructor (arrayBuffer: ArrayBuffer); + /** + * Allocates a new buffer containing the given {array} of octets. + * + * @param array The octets to store. + */ + constructor (array: any[]); + /** + * Copies the passed {buffer} data onto a new {Buffer} instance. + * + * @param buffer The buffer to copy. + */ + constructor (buffer: Buffer); + prototype: Buffer; + /** + * Allocates a new Buffer using an {array} of octets. + * + * @param array + */ + static from(array: any[]): Buffer; + /** + * When passed a reference to the .buffer property of a TypedArray instance, + * the newly created Buffer will share the same allocated memory as the TypedArray. + * The optional {byteOffset} and {length} arguments specify a memory range + * within the {arrayBuffer} that will be shared by the Buffer. + * + * @param arrayBuffer The .buffer property of a TypedArray or a new ArrayBuffer() + * @param byteOffset + * @param length + */ + static from(arrayBuffer: ArrayBuffer, byteOffset?: number, length?: number): Buffer; + /** + * Copies the passed {buffer} data onto a new Buffer instance. + * + * @param buffer + */ + static from(buffer: Buffer): Buffer; + /** + * Creates a new Buffer containing the given JavaScript string {str}. + * If provided, the {encoding} parameter identifies the character encoding. + * If not provided, {encoding} defaults to 'utf8'. + * + * @param str + */ + static from(str: string, encoding?: string): Buffer; + /** + * Returns true if {obj} is a Buffer + * + * @param obj object to test. + */ + static isBuffer(obj: any): obj is Buffer; + /** + * Returns true if {encoding} is a valid encoding argument. + * Valid string encodings in Node 0.12: 'ascii'|'utf8'|'utf16le'|'ucs2'(alias of 'utf16le')|'base64'|'binary'(deprecated)|'hex' + * + * @param encoding string to test. + */ + static isEncoding(encoding: string): boolean; + /** + * Gives the actual byte length of a string. encoding defaults to 'utf8'. + * This is not the same as String.prototype.length since that returns the number of characters in a string. + * + * @param string string to test. + * @param encoding encoding used to evaluate (defaults to 'utf8') + */ + static byteLength(string: string, encoding?: string): number; + /** + * Returns a buffer which is the result of concatenating all the buffers in the list together. + * + * If the list has no items, or if the totalLength is 0, then it returns a zero-length buffer. + * If the list has exactly one item, then the first item of the list is returned. + * If the list has more than one item, then a new Buffer is created. + * + * @param list An array of Buffer objects to concatenate + * @param totalLength Total length of the buffers when concatenated. + * If totalLength is not provided, it is read from the buffers in the list. However, this adds an additional loop to the function, so it is faster to provide the length explicitly. + */ + static concat(list: Buffer[], totalLength?: number): Buffer; + /** + * The same as buf1.compare(buf2). + */ + static compare(buf1: Buffer, buf2: Buffer): number; + /** + * Allocates a new buffer of {size} octets. + * + * @param size count of octets to allocate. + * @param fill if specified, buffer will be initialized by calling buf.fill(fill). + * If parameter is omitted, buffer will be filled with zeros. + * @param encoding encoding used for call to buf.fill while initalizing + */ + static alloc(size: number, fill?: string | Buffer | number, encoding?: string): Buffer; + /** + * Allocates a new buffer of {size} octets, leaving memory not initialized, so the contents + * of the newly created Buffer are unknown and may contain sensitive data. + * + * @param size count of octets to allocate + */ + static allocUnsafe(size: number): Buffer; + /** + * Allocates a new non-pooled buffer of {size} octets, leaving memory not initialized, so the contents + * of the newly created Buffer are unknown and may contain sensitive data. + * + * @param size count of octets to allocate + */ + static allocUnsafeSlow(size: number): Buffer; + } +} \ No newline at end of file diff --git a/node_modules/safe-buffer/index.js b/node_modules/safe-buffer/index.js new file mode 100644 index 0000000..f8d3ec9 --- /dev/null +++ b/node_modules/safe-buffer/index.js @@ -0,0 +1,65 @@ +/*! safe-buffer. MIT License. Feross Aboukhadijeh */ +/* eslint-disable node/no-deprecated-api */ +var buffer = require('buffer') +var Buffer = buffer.Buffer + +// alternative to using Object.keys for old browsers +function copyProps (src, dst) { + for (var key in src) { + dst[key] = src[key] + } +} +if (Buffer.from && Buffer.alloc && Buffer.allocUnsafe && Buffer.allocUnsafeSlow) { + module.exports = buffer +} else { + // Copy properties from require('buffer') + copyProps(buffer, exports) + exports.Buffer = SafeBuffer +} + +function SafeBuffer (arg, encodingOrOffset, length) { + return Buffer(arg, encodingOrOffset, length) +} + +SafeBuffer.prototype = Object.create(Buffer.prototype) + +// Copy static methods from Buffer +copyProps(Buffer, SafeBuffer) + +SafeBuffer.from = function (arg, encodingOrOffset, length) { + if (typeof arg === 'number') { + throw new TypeError('Argument must not be a number') + } + return Buffer(arg, encodingOrOffset, length) +} + +SafeBuffer.alloc = function (size, fill, encoding) { + if (typeof size !== 'number') { + throw new TypeError('Argument must be a number') + } + var buf = Buffer(size) + if (fill !== undefined) { + if (typeof encoding === 'string') { + buf.fill(fill, encoding) + } else { + buf.fill(fill) + } + } else { + buf.fill(0) + } + return buf +} + +SafeBuffer.allocUnsafe = function (size) { + if (typeof size !== 'number') { + throw new TypeError('Argument must be a number') + } + return Buffer(size) +} + +SafeBuffer.allocUnsafeSlow = function (size) { + if (typeof size !== 'number') { + throw new TypeError('Argument must be a number') + } + return buffer.SlowBuffer(size) +} diff --git a/node_modules/safe-buffer/package.json b/node_modules/safe-buffer/package.json new file mode 100644 index 0000000..f2869e2 --- /dev/null +++ b/node_modules/safe-buffer/package.json @@ -0,0 +1,51 @@ +{ + "name": "safe-buffer", + "description": "Safer Node.js Buffer API", + "version": "5.2.1", + "author": { + "name": "Feross Aboukhadijeh", + "email": "feross@feross.org", + "url": "https://feross.org" + }, + "bugs": { + "url": "https://github.com/feross/safe-buffer/issues" + }, + "devDependencies": { + "standard": "*", + "tape": "^5.0.0" + }, + "homepage": "https://github.com/feross/safe-buffer", + "keywords": [ + "buffer", + "buffer allocate", + "node security", + "safe", + "safe-buffer", + "security", + "uninitialized" + ], + "license": "MIT", + "main": "index.js", + "types": "index.d.ts", + "repository": { + "type": "git", + "url": "git://github.com/feross/safe-buffer.git" + }, + "scripts": { + "test": "standard && tape test/*.js" + }, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] +} diff --git a/node_modules/secp256k1/API.md b/node_modules/secp256k1/API.md new file mode 100644 index 0000000..982fbb5 --- /dev/null +++ b/node_modules/secp256k1/API.md @@ -0,0 +1,116 @@ +## API Reference (v4.x) + +- Functions work with [Uint8Array](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Uint8Array). While [Buffer](https://nodejs.org/api/buffer.html) is awesome, current version for browsers ([feross/buffer](https://github.com/feross/buffer/)) is out of date (compare to Node.js Buffer) and in future difference probably will be only bigger. But because Buffer extends Uint8Array, you can pass and receive Buffers easily. Also, work with native Uint8Array reduce final build size, if you do not use Buffer in your browser application. + +- Custom type for data output. It's possible pass Buffer or Object which inherits Uint8Array to function for data output. Of course length should match, or you can pass function which accept number of bytes and return instance with specified length. + +- In place operations (follow [bitcoin-core/secp256k1](https://github.com/bitcoin-core/secp256k1) API): + + - `privateKeyNegate` + - `privateKeyTweakAdd` + - `privateKeyTweakMul` + - `signatureNormalize` + +If you need new instance this can be easily done with creating it before pass to functions. For example: + +```js +const newPrivateKey = secp256k1.privateKeyNegate(Buffer.from(privateKey)) +``` + +
+ +- [`.contextRandomize(seed: Uint8Array): void`](#contextrandomizeseed-uint8array-void) +- [`.privateKeyVerify(privateKey: Uint8Array): boolean`](#privatekeyverifyprivatekey-uint8array-boolean) +- [`.privateKeyNegate(privateKey: Uint8Array): Uint8Array`](#privatekeynegateprivatekey-uint8array-uint8array) +- [`.privateKeyTweakAdd(privateKey: Uint8Array, tweak: Uint8Array): Uint8Array`](#privatekeytweakaddprivatekey-uint8array-tweak-uint8array-uint8array) +- [`.privateKeyTweakMul(privateKey: Uint8Array, tweak: Uint8Array): Uint8Array`](#privatekeytweakmulprivatekey-uint8array-tweak-uint8array-uint8array) +- [`.publicKeyVerify(publicKey: Uint8Array): boolean`](#publickeyverifypublickey-uint8array-boolean) +- [`.publicKeyCreate(privateKey: Uint8Array, compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array`](#publickeycreateprivatekey-uint8array-compressed-boolean--true-output-uint8array--len-number--uint8array--len--new-uint8arraylen-uint8array) +- [`.publicKeyConvert(publicKey: Uint8Array, compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array`](#publickeyconvertpublickey-uint8array-compressed-boolean--true-output-uint8array--len-number--uint8array--len--new-uint8arraylen-uint8array) +- [`.publicKeyNegate(publicKey: Uint8Array, compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array`](#publickeynegatepublickey-uint8array-compressed-boolean--true-output-uint8array--len-number--uint8array--len--new-uint8arraylen-uint8array) +- [`.publicKeyCombine(publicKeys: Uint8Array[], compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array`](#publickeycombinepublickeys-uint8array-compressed-boolean--true-output-uint8array--len-number--uint8array--len--new-uint8arraylen-uint8array) +- [`.publicKeyTweakAdd(publicKey: Uint8Array, tweak: Uint8Array, compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array`](#publickeytweakaddpublickey-uint8array-tweak-uint8array-compressed-boolean--true-output-uint8array--len-number--uint8array--len--new-uint8arraylen-uint8array) +- [`.publicKeyTweakMul(publicKey: Uint8Array, tweak: Uint8Array, compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array`](#publickeytweakmulpublickey-uint8array-tweak-uint8array-compressed-boolean--true-output-uint8array--len-number--uint8array--len--new-uint8arraylen-uint8array) +- [`.signatureNormalize(signature: Uint8Array): Uint8Array`](#signaturenormalizesignature-uint8array-uint8array) +- [`.signatureExport(signature, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array`](#signatureexportsignature-output-uint8array--len-number--uint8array--len--new-uint8arraylen-uint8array) +- [`.signatureImport(signature, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array`](#signatureimportsignature-output-uint8array--len-number--uint8array--len--new-uint8arraylen-uint8array) +- [`.ecdsaSign(message: Uint8Array, privateKey: Uint8Array, { data, noncefn }: { data?: Uint8Array, noncefn?: ((message: Uint8Array, privateKey: Uint8Array, algo: null, data: Uint8Array, counter: number) => Uint8Array) } = {}, output: Uint8Array | ((len: number) => Uint8Array)): { signature: Uint8Array, recid: number }`](#ecdsasignmessage-uint8array-privatekey-uint8array--data-noncefn---data-uint8array-noncefn-message-uint8array-privatekey-uint8array-algo-null-data-uint8array-counter-number--uint8array----output-uint8array--len-number--uint8array--signature-uint8array-recid-number-) +- [`.ecdsaVerify(signature: Uint8Array, message: Uint8Array, publicKey: Uint8Array): boolean`](#ecdsaverifysignature-uint8array-message-uint8array-publickey-uint8array-boolean) +- [`.ecdsaRecover(signature: Uint8Array, recid: number, message: Uint8Array, compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array`](#ecdsarecoversignature-uint8array-recid-number-message-uint8array-compressed-boolean--true-output-uint8array--len-number--uint8array--len--new-uint8arraylen-uint8array) +- [`.ecdh(publicKey: Uint8Array, privateKey: Uint8Array, { data, xbuf, ybuf, hashfn }: { data?: Uint8Array, xbuf?: Uint8Array, ybuf?: Uint8Array, hashfn?: ((x: Uint8Array, y: Uint8Array, data: Uint8Array) => Uint8Array) } = {}, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array`](#ecdhpublickey-uint8array-privatekey-uint8array--data-xbuf-ybuf-hashfn---data-uint8array-xbuf-uint8array-ybuf-uint8array-hashfn-x-uint8array-y-uint8array-data-uint8array--uint8array----output-uint8array--len-number--uint8array--len--new-uint8arraylen-uint8array) + +##### .contextRandomize(seed: Uint8Array): void + +Updates the context randomization to protect against side-channel leakage, `seed` should be Uint8Array with length 32. + +##### .privateKeyVerify(privateKey: Uint8Array): boolean + +Verify a private key. + +##### .privateKeyNegate(privateKey: Uint8Array): Uint8Array + +Negate a private key in place and return result. + +##### .privateKeyTweakAdd(privateKey: Uint8Array, tweak: Uint8Array): Uint8Array + +Tweak a private key in place by adding tweak to it. + +##### .privateKeyTweakMul(privateKey: Uint8Array, tweak: Uint8Array): Uint8Array + +Tweak a private key in place by multiplying it by a tweak. + +##### .publicKeyVerify(publicKey: Uint8Array): boolean + +Verify a public key. + +##### .publicKeyCreate(privateKey: Uint8Array, compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array + +Compute the public key for a secret key. + +##### .publicKeyConvert(publicKey: Uint8Array, compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array + +Reserialize public key to another format. + +##### .publicKeyNegate(publicKey: Uint8Array, compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array + +Negates a public key in place. + +##### .publicKeyCombine(publicKeys: Uint8Array[], compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array + +Add a number of public keys together. + +##### .publicKeyTweakAdd(publicKey: Uint8Array, tweak: Uint8Array, compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array + +Tweak a public key by adding tweak times the generator to it. + +##### .publicKeyTweakMul(publicKey: Uint8Array, tweak: Uint8Array, compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array + +Tweak a public key by multiplying it by a tweak value. + +##### .signatureNormalize(signature: Uint8Array): Uint8Array + +Convert a signature to a normalized lower-S form in place. + +##### .signatureExport(signature, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array + +Export an ECDSA signature to DER format. + +##### .signatureImport(signature, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array + +Parse a DER ECDSA signature. + +##### .ecdsaSign(message: Uint8Array, privateKey: Uint8Array, { data, noncefn }: { data?: Uint8Array, noncefn?: ((message: Uint8Array, privateKey: Uint8Array, algo: null, data: Uint8Array, counter: number) => Uint8Array) } = {}, output: Uint8Array | ((len: number) => Uint8Array)): { signature: Uint8Array, recid: number } + +Create an ECDSA signature. + +##### .ecdsaVerify(signature: Uint8Array, message: Uint8Array, publicKey: Uint8Array): boolean + +Verify an ECDSA signature. + +##### .ecdsaRecover(signature: Uint8Array, recid: number, message: Uint8Array, compressed: boolean = true, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array + +Recover an ECDSA public key from a signature. + +##### .ecdh(publicKey: Uint8Array, privateKey: Uint8Array, { data, xbuf, ybuf, hashfn }: { data?: Uint8Array, xbuf?: Uint8Array, ybuf?: Uint8Array, hashfn?: ((x: Uint8Array, y: Uint8Array, data: Uint8Array) => Uint8Array) } = {}, output: Uint8Array | ((len: number) => Uint8Array) = (len) => new Uint8Array(len)): Uint8Array + +Compute an EC Diffie-Hellman secret in constant time. diff --git a/node_modules/secp256k1/LICENSE b/node_modules/secp256k1/LICENSE new file mode 100644 index 0000000..f2369eb --- /dev/null +++ b/node_modules/secp256k1/LICENSE @@ -0,0 +1,24 @@ +The MIT License (MIT) + +Copyright (c) 2014-2016 secp256k1-node contributors + +Parts of this software are based on bn.js, elliptic, hash.js +Copyright (c) 2014-2016 Fedor Indutny + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/node_modules/secp256k1/README.md b/node_modules/secp256k1/README.md new file mode 100644 index 0000000..954098c --- /dev/null +++ b/node_modules/secp256k1/README.md @@ -0,0 +1,117 @@ +# secp256k1-node + +This module provides native bindings to [bitcoin-core/secp256k1](https://github.com/bitcoin-core/secp256k1). In browser [elliptic](https://github.com/indutny/elliptic) will be used as fallback. + +Works on node version 14.0.0 or greater, because use [N-API](https://nodejs.org/api/n-api.html). + +## Installation + +##### from npm + +`npm install secp256k1` + +##### from git + +``` +git clone git@github.com:cryptocoinjs/secp256k1-node.git +cd secp256k1-node +git submodule update --init +npm install +``` + +##### Windows + +The easiest way to build the package on windows is to install [windows-build-tools](https://github.com/felixrieseberg/windows-build-tools). + +Or install the following software: + + * Git: https://git-scm.com/download/win + * nvm: https://github.com/coreybutler/nvm-windows + * Python 2.7: https://www.python.org/downloads/release/python-2712/ + * Visual C++ Build Tools: http://landinghub.visualstudio.com/visual-cpp-build-tools (Custom Install, and select both Windows 8.1 and Windows 10 SDKs) + +And run commands: + +``` +npm config set msvs_version 2015 --global +npm install npm@next -g +``` + +Based on: + + * https://github.com/nodejs/node-gyp/issues/629#issuecomment-153196245 + * https://github.com/nodejs/node-gyp/issues/972 + +## Usage + +* [API Reference (v4.x)](API.md) (current version) +* [API Reference (v3.x)](https://github.com/cryptocoinjs/secp256k1-node/blob/v3.x/API.md) +* [API Reference (v2.x)](https://github.com/cryptocoinjs/secp256k1-node/blob/v2.x/API.md) + +##### Private Key generation, Public Key creation, signature creation, signature verification + +```js +const { randomBytes } = require('crypto') +const secp256k1 = require('secp256k1') +// or require('secp256k1/elliptic') +// if you want to use pure js implementation in node + +// generate message to sign +// message should have 32-byte length, if you have some other length you can hash message +// for example `msg = sha256(rawMessage)` +const msg = randomBytes(32) + +// generate privKey +let privKey +do { + privKey = randomBytes(32) +} while (!secp256k1.privateKeyVerify(privKey)) + +// get the public key in a compressed format +const pubKey = secp256k1.publicKeyCreate(privKey) + +// sign the message +const sigObj = secp256k1.ecdsaSign(msg, privKey) + +// verify the signature +console.log(secp256k1.ecdsaVerify(sigObj.signature, msg, pubKey)) +// => true +``` + +\* **.verify return false for high signatures** + +##### Get X point of ECDH + +```js +const { randomBytes } = require('crypto') +// const secp256k1 = require('./elliptic') +const secp256k1 = require('./') + +// generate privKey +function getPrivateKey () { + while (true) { + const privKey = randomBytes(32) + if (secp256k1.privateKeyVerify(privKey)) return privKey + } +} + +// generate private and public keys +const privKey = getPrivateKey() +const pubKey = secp256k1.publicKeyCreate(privKey) + +// compressed public key from X and Y +function hashfn (x, y) { + const pubKey = new Uint8Array(33) + pubKey[0] = (y[31] & 1) === 0 ? 0x02 : 0x03 + pubKey.set(x, 1) + return pubKey +} + +// get X point of ecdh +const ecdhPointX = secp256k1.ecdh(pubKey, privKey, { hashfn }, Buffer.alloc(33)) +console.log(ecdhPointX.toString('hex')) +``` + +## LICENSE + +This library is free and open-source software released under the MIT license. diff --git a/node_modules/secp256k1/binding.gyp b/node_modules/secp256k1/binding.gyp new file mode 100644 index 0000000..9ca0c3c --- /dev/null +++ b/node_modules/secp256k1/binding.gyp @@ -0,0 +1,100 @@ +{ + 'variables': { + 'arch': '= 0) return null + x = x.toRed(ecparams.red) + + // compute corresponding Y + let y = x.redSqr().redIMul(x).redIAdd(ecparams.b).redSqrt() + if ((first === 0x03) !== y.isOdd()) y = y.redNeg() + + return ec.keyPair({ pub: { x: x, y: y } }) +} + +function loadUncompressedPublicKey (first, xbuf, ybuf) { + let x = new BN(xbuf) + let y = new BN(ybuf) + + // overflow + if (x.cmp(ecparams.p) >= 0 || y.cmp(ecparams.p) >= 0) return null + + x = x.toRed(ecparams.red) + y = y.toRed(ecparams.red) + + // is odd flag + if ((first === 0x06 || first === 0x07) && y.isOdd() !== (first === 0x07)) return null + + // x*x*x + b = y*y + const x3 = x.redSqr().redIMul(x) + if (!y.redSqr().redISub(x3.redIAdd(ecparams.b)).isZero()) return null + + return ec.keyPair({ pub: { x: x, y: y } }) +} + +function loadPublicKey (pubkey) { + // length should be validated in interface + const first = pubkey[0] + switch (first) { + case 0x02: + case 0x03: + if (pubkey.length !== 33) return null + return loadCompressedPublicKey(first, pubkey.subarray(1, 33)) + case 0x04: + case 0x06: + case 0x07: + if (pubkey.length !== 65) return null + return loadUncompressedPublicKey(first, pubkey.subarray(1, 33), pubkey.subarray(33, 65)) + default: + return null + } +} + +function savePublicKey (output, point) { + const pubkey = point.encode(null, output.length === 33) + // Loop should be faster because we do not need create extra Uint8Array + // output.set(new Uint8Array(pubkey)) + for (let i = 0; i < output.length; ++i) output[i] = pubkey[i] +} + +module.exports = { + contextRandomize () { + return 0 + }, + + privateKeyVerify (seckey) { + const bn = new BN(seckey) + return bn.cmp(ecparams.n) < 0 && !bn.isZero() ? 0 : 1 + }, + + privateKeyNegate (seckey) { + const bn = new BN(seckey) + const negate = ecparams.n.sub(bn).umod(ecparams.n).toArrayLike(Uint8Array, 'be', 32) + seckey.set(negate) + return 0 + }, + + privateKeyTweakAdd (seckey, tweak) { + const bn = new BN(tweak) + if (bn.cmp(ecparams.n) >= 0) return 1 + + bn.iadd(new BN(seckey)) + if (bn.cmp(ecparams.n) >= 0) bn.isub(ecparams.n) + if (bn.isZero()) return 1 + + const tweaked = bn.toArrayLike(Uint8Array, 'be', 32) + seckey.set(tweaked) + + return 0 + }, + + privateKeyTweakMul (seckey, tweak) { + let bn = new BN(tweak) + if (bn.cmp(ecparams.n) >= 0 || bn.isZero()) return 1 + + bn.imul(new BN(seckey)) + if (bn.cmp(ecparams.n) >= 0) bn = bn.umod(ecparams.n) + + const tweaked = bn.toArrayLike(Uint8Array, 'be', 32) + seckey.set(tweaked) + + return 0 + }, + + publicKeyVerify (pubkey) { + const pair = loadPublicKey(pubkey) + return pair === null ? 1 : 0 + }, + + publicKeyCreate (output, seckey) { + const bn = new BN(seckey) + if (bn.cmp(ecparams.n) >= 0 || bn.isZero()) return 1 + + const point = ec.keyFromPrivate(seckey).getPublic() + savePublicKey(output, point) + + return 0 + }, + + publicKeyConvert (output, pubkey) { + const pair = loadPublicKey(pubkey) + if (pair === null) return 1 + + const point = pair.getPublic() + savePublicKey(output, point) + + return 0 + }, + + publicKeyNegate (output, pubkey) { + const pair = loadPublicKey(pubkey) + if (pair === null) return 1 + + const point = pair.getPublic() + point.y = point.y.redNeg() + savePublicKey(output, point) + + return 0 + }, + + publicKeyCombine (output, pubkeys) { + const pairs = new Array(pubkeys.length) + for (let i = 0; i < pubkeys.length; ++i) { + pairs[i] = loadPublicKey(pubkeys[i]) + if (pairs[i] === null) return 1 + } + + let point = pairs[0].getPublic() + for (let i = 1; i < pairs.length; ++i) point = point.add(pairs[i].pub) + if (point.isInfinity()) return 2 + + savePublicKey(output, point) + + return 0 + }, + + publicKeyTweakAdd (output, pubkey, tweak) { + const pair = loadPublicKey(pubkey) + if (pair === null) return 1 + + tweak = new BN(tweak) + if (tweak.cmp(ecparams.n) >= 0) return 2 + + const point = pair.getPublic().add(ecparams.g.mul(tweak)) + if (point.isInfinity()) return 2 + + savePublicKey(output, point) + + return 0 + }, + + publicKeyTweakMul (output, pubkey, tweak) { + const pair = loadPublicKey(pubkey) + if (pair === null) return 1 + + tweak = new BN(tweak) + if (tweak.cmp(ecparams.n) >= 0 || tweak.isZero()) return 2 + + const point = pair.getPublic().mul(tweak) + savePublicKey(output, point) + + return 0 + }, + + signatureNormalize (sig) { + const r = new BN(sig.subarray(0, 32)) + const s = new BN(sig.subarray(32, 64)) + if (r.cmp(ecparams.n) >= 0 || s.cmp(ecparams.n) >= 0) return 1 + + if (s.cmp(ec.nh) === 1) { + sig.set(ecparams.n.sub(s).toArrayLike(Uint8Array, 'be', 32), 32) + } + + return 0 + }, + + // Copied 1-to-1 from https://github.com/bitcoinjs/bip66/blob/master/index.js + // Adapted for Uint8Array instead Buffer + signatureExport (obj, sig) { + const sigR = sig.subarray(0, 32) + const sigS = sig.subarray(32, 64) + if (new BN(sigR).cmp(ecparams.n) >= 0) return 1 + if (new BN(sigS).cmp(ecparams.n) >= 0) return 1 + + const { output } = obj + + // Prepare R + let r = output.subarray(4, 4 + 33) + r[0] = 0x00 + r.set(sigR, 1) + + let lenR = 33 + let posR = 0 + for (; lenR > 1 && r[posR] === 0x00 && !(r[posR + 1] & 0x80); --lenR, ++posR); + + r = r.subarray(posR) + if (r[0] & 0x80) return 1 + if (lenR > 1 && (r[0] === 0x00) && !(r[1] & 0x80)) return 1 + + // Prepare S + let s = output.subarray(6 + 33, 6 + 33 + 33) + s[0] = 0x00 + s.set(sigS, 1) + + let lenS = 33 + let posS = 0 + for (; lenS > 1 && s[posS] === 0x00 && !(s[posS + 1] & 0x80); --lenS, ++posS); + + s = s.subarray(posS) + if (s[0] & 0x80) return 1 + if (lenS > 1 && (s[0] === 0x00) && !(s[1] & 0x80)) return 1 + + // Set output length for return + obj.outputlen = 6 + lenR + lenS + + // Output in specified format + // 0x30 [total-length] 0x02 [R-length] [R] 0x02 [S-length] [S] + output[0] = 0x30 + output[1] = obj.outputlen - 2 + output[2] = 0x02 + output[3] = r.length + output.set(r, 4) + output[4 + lenR] = 0x02 + output[5 + lenR] = s.length + output.set(s, 6 + lenR) + + return 0 + }, + + // Copied 1-to-1 from https://github.com/bitcoinjs/bip66/blob/master/index.js + // Adapted for Uint8Array instead Buffer + signatureImport (output, sig) { + if (sig.length < 8) return 1 + if (sig.length > 72) return 1 + if (sig[0] !== 0x30) return 1 + if (sig[1] !== sig.length - 2) return 1 + if (sig[2] !== 0x02) return 1 + + const lenR = sig[3] + if (lenR === 0) return 1 + if (5 + lenR >= sig.length) return 1 + if (sig[4 + lenR] !== 0x02) return 1 + + const lenS = sig[5 + lenR] + if (lenS === 0) return 1 + if ((6 + lenR + lenS) !== sig.length) return 1 + + if (sig[4] & 0x80) return 1 + if (lenR > 1 && (sig[4] === 0x00) && !(sig[5] & 0x80)) return 1 + + if (sig[lenR + 6] & 0x80) return 1 + if (lenS > 1 && (sig[lenR + 6] === 0x00) && !(sig[lenR + 7] & 0x80)) return 1 + + let sigR = sig.subarray(4, 4 + lenR) + if (sigR.length === 33 && sigR[0] === 0x00) sigR = sigR.subarray(1) + if (sigR.length > 32) return 1 + + let sigS = sig.subarray(6 + lenR) + if (sigS.length === 33 && sigS[0] === 0x00) sigS = sigS.slice(1) + if (sigS.length > 32) throw new Error('S length is too long') + + let r = new BN(sigR) + if (r.cmp(ecparams.n) >= 0) r = new BN(0) + + let s = new BN(sig.subarray(6 + lenR)) + if (s.cmp(ecparams.n) >= 0) s = new BN(0) + + output.set(r.toArrayLike(Uint8Array, 'be', 32), 0) + output.set(s.toArrayLike(Uint8Array, 'be', 32), 32) + + return 0 + }, + + ecdsaSign (obj, message, seckey, data, noncefn) { + if (noncefn) { + const _noncefn = noncefn + noncefn = (counter) => { + const nonce = _noncefn(message, seckey, null, data, counter) + + const isValid = nonce instanceof Uint8Array && nonce.length === 32 + if (!isValid) throw new Error('This is the way') + + return new BN(nonce) + } + } + + const d = new BN(seckey) + if (d.cmp(ecparams.n) >= 0 || d.isZero()) return 1 + + let sig + try { + sig = ec.sign(message, seckey, { canonical: true, k: noncefn, pers: data }) + } catch (err) { + return 1 + } + + obj.signature.set(sig.r.toArrayLike(Uint8Array, 'be', 32), 0) + obj.signature.set(sig.s.toArrayLike(Uint8Array, 'be', 32), 32) + obj.recid = sig.recoveryParam + + return 0 + }, + + ecdsaVerify (sig, msg32, pubkey) { + const sigObj = { r: sig.subarray(0, 32), s: sig.subarray(32, 64) } + + const sigr = new BN(sigObj.r) + const sigs = new BN(sigObj.s) + if (sigr.cmp(ecparams.n) >= 0 || sigs.cmp(ecparams.n) >= 0) return 1 + if (sigs.cmp(ec.nh) === 1 || sigr.isZero() || sigs.isZero()) return 3 + + const pair = loadPublicKey(pubkey) + if (pair === null) return 2 + + const point = pair.getPublic() + const isValid = ec.verify(msg32, sigObj, point) + return isValid ? 0 : 3 + }, + + ecdsaRecover (output, sig, recid, msg32) { + const sigObj = { r: sig.slice(0, 32), s: sig.slice(32, 64) } + + const sigr = new BN(sigObj.r) + const sigs = new BN(sigObj.s) + if (sigr.cmp(ecparams.n) >= 0 || sigs.cmp(ecparams.n) >= 0) return 1 + + if (sigr.isZero() || sigs.isZero()) return 2 + + // Can throw `throw new Error('Unable to find sencond key candinate');` + let point + try { + point = ec.recoverPubKey(msg32, sigObj, recid) + } catch (err) { + return 2 + } + + savePublicKey(output, point) + + return 0 + }, + + ecdh (output, pubkey, seckey, data, hashfn, xbuf, ybuf) { + const pair = loadPublicKey(pubkey) + if (pair === null) return 1 + + const scalar = new BN(seckey) + if (scalar.cmp(ecparams.n) >= 0 || scalar.isZero()) return 2 + + const point = pair.getPublic().mul(scalar) + + if (hashfn === undefined) { + const data = point.encode(null, true) + const sha256 = ec.hash().update(data).digest() + for (let i = 0; i < 32; ++i) output[i] = sha256[i] + } else { + if (!xbuf) xbuf = new Uint8Array(32) + const x = point.getX().toArray('be', 32) + for (let i = 0; i < 32; ++i) xbuf[i] = x[i] + + if (!ybuf) ybuf = new Uint8Array(32) + const y = point.getY().toArray('be', 32) + for (let i = 0; i < 32; ++i) ybuf[i] = y[i] + + const hash = hashfn(xbuf, ybuf, data) + + const isValid = hash instanceof Uint8Array && hash.length === output.length + if (!isValid) return 2 + + output.set(hash) + } + + return 0 + } +} diff --git a/node_modules/secp256k1/lib/index.js b/node_modules/secp256k1/lib/index.js new file mode 100644 index 0000000..8187e17 --- /dev/null +++ b/node_modules/secp256k1/lib/index.js @@ -0,0 +1,336 @@ +const errors = { + IMPOSSIBLE_CASE: 'Impossible case. Please create issue.', + TWEAK_ADD: + 'The tweak was out of range or the resulted private key is invalid', + TWEAK_MUL: 'The tweak was out of range or equal to zero', + CONTEXT_RANDOMIZE_UNKNOW: 'Unknow error on context randomization', + SECKEY_INVALID: 'Private Key is invalid', + PUBKEY_PARSE: 'Public Key could not be parsed', + PUBKEY_SERIALIZE: 'Public Key serialization error', + PUBKEY_COMBINE: 'The sum of the public keys is not valid', + SIG_PARSE: 'Signature could not be parsed', + SIGN: 'The nonce generation function failed, or the private key was invalid', + RECOVER: 'Public key could not be recover', + ECDH: 'Scalar was invalid (zero or overflow)' +} + +function assert (cond, msg) { + if (!cond) throw new Error(msg) +} + +function isUint8Array (name, value, length) { + assert(value instanceof Uint8Array, `Expected ${name} to be an Uint8Array`) + + if (length !== undefined) { + if (Array.isArray(length)) { + const numbers = length.join(', ') + const msg = `Expected ${name} to be an Uint8Array with length [${numbers}]` + assert(length.includes(value.length), msg) + } else { + const msg = `Expected ${name} to be an Uint8Array with length ${length}` + assert(value.length === length, msg) + } + } +} + +function isCompressed (value) { + assert(toTypeString(value) === 'Boolean', 'Expected compressed to be a Boolean') +} + +function getAssertedOutput (output = (len) => new Uint8Array(len), length) { + if (typeof output === 'function') output = output(length) + isUint8Array('output', output, length) + return output +} + +function toTypeString (value) { + return Object.prototype.toString.call(value).slice(8, -1) +} + +module.exports = (secp256k1) => { + return { + contextRandomize (seed) { + assert( + seed === null || seed instanceof Uint8Array, + 'Expected seed to be an Uint8Array or null' + ) + if (seed !== null) isUint8Array('seed', seed, 32) + + switch (secp256k1.contextRandomize(seed)) { + case 1: + throw new Error(errors.CONTEXT_RANDOMIZE_UNKNOW) + } + }, + + privateKeyVerify (seckey) { + isUint8Array('private key', seckey, 32) + + return secp256k1.privateKeyVerify(seckey) === 0 + }, + + privateKeyNegate (seckey) { + isUint8Array('private key', seckey, 32) + + switch (secp256k1.privateKeyNegate(seckey)) { + case 0: + return seckey + case 1: + throw new Error(errors.IMPOSSIBLE_CASE) + } + }, + + privateKeyTweakAdd (seckey, tweak) { + isUint8Array('private key', seckey, 32) + isUint8Array('tweak', tweak, 32) + + switch (secp256k1.privateKeyTweakAdd(seckey, tweak)) { + case 0: + return seckey + case 1: + throw new Error(errors.TWEAK_ADD) + } + }, + + privateKeyTweakMul (seckey, tweak) { + isUint8Array('private key', seckey, 32) + isUint8Array('tweak', tweak, 32) + + switch (secp256k1.privateKeyTweakMul(seckey, tweak)) { + case 0: + return seckey + case 1: + throw new Error(errors.TWEAK_MUL) + } + }, + + publicKeyVerify (pubkey) { + isUint8Array('public key', pubkey, [33, 65]) + + return secp256k1.publicKeyVerify(pubkey) === 0 + }, + + publicKeyCreate (seckey, compressed = true, output) { + isUint8Array('private key', seckey, 32) + isCompressed(compressed) + output = getAssertedOutput(output, compressed ? 33 : 65) + + switch (secp256k1.publicKeyCreate(output, seckey)) { + case 0: + return output + case 1: + throw new Error(errors.SECKEY_INVALID) + case 2: + throw new Error(errors.PUBKEY_SERIALIZE) + } + }, + + publicKeyConvert (pubkey, compressed = true, output) { + isUint8Array('public key', pubkey, [33, 65]) + isCompressed(compressed) + output = getAssertedOutput(output, compressed ? 33 : 65) + + switch (secp256k1.publicKeyConvert(output, pubkey)) { + case 0: + return output + case 1: + throw new Error(errors.PUBKEY_PARSE) + case 2: + throw new Error(errors.PUBKEY_SERIALIZE) + } + }, + + publicKeyNegate (pubkey, compressed = true, output) { + isUint8Array('public key', pubkey, [33, 65]) + isCompressed(compressed) + output = getAssertedOutput(output, compressed ? 33 : 65) + + switch (secp256k1.publicKeyNegate(output, pubkey)) { + case 0: + return output + case 1: + throw new Error(errors.PUBKEY_PARSE) + case 2: + throw new Error(errors.IMPOSSIBLE_CASE) + case 3: + throw new Error(errors.PUBKEY_SERIALIZE) + } + }, + + publicKeyCombine (pubkeys, compressed = true, output) { + assert(Array.isArray(pubkeys), 'Expected public keys to be an Array') + assert(pubkeys.length > 0, 'Expected public keys array will have more than zero items') + for (const pubkey of pubkeys) { + isUint8Array('public key', pubkey, [33, 65]) + } + isCompressed(compressed) + output = getAssertedOutput(output, compressed ? 33 : 65) + + switch (secp256k1.publicKeyCombine(output, pubkeys)) { + case 0: + return output + case 1: + throw new Error(errors.PUBKEY_PARSE) + case 2: + throw new Error(errors.PUBKEY_COMBINE) + case 3: + throw new Error(errors.PUBKEY_SERIALIZE) + } + }, + + publicKeyTweakAdd (pubkey, tweak, compressed = true, output) { + isUint8Array('public key', pubkey, [33, 65]) + isUint8Array('tweak', tweak, 32) + isCompressed(compressed) + output = getAssertedOutput(output, compressed ? 33 : 65) + + switch (secp256k1.publicKeyTweakAdd(output, pubkey, tweak)) { + case 0: + return output + case 1: + throw new Error(errors.PUBKEY_PARSE) + case 2: + throw new Error(errors.TWEAK_ADD) + } + }, + + publicKeyTweakMul (pubkey, tweak, compressed = true, output) { + isUint8Array('public key', pubkey, [33, 65]) + isUint8Array('tweak', tweak, 32) + isCompressed(compressed) + output = getAssertedOutput(output, compressed ? 33 : 65) + + switch (secp256k1.publicKeyTweakMul(output, pubkey, tweak)) { + case 0: + return output + case 1: + throw new Error(errors.PUBKEY_PARSE) + case 2: + throw new Error(errors.TWEAK_MUL) + } + }, + + signatureNormalize (sig) { + isUint8Array('signature', sig, 64) + + switch (secp256k1.signatureNormalize(sig)) { + case 0: + return sig + case 1: + throw new Error(errors.SIG_PARSE) + } + }, + + signatureExport (sig, output) { + isUint8Array('signature', sig, 64) + output = getAssertedOutput(output, 72) + + const obj = { output, outputlen: 72 } + switch (secp256k1.signatureExport(obj, sig)) { + case 0: + return output.slice(0, obj.outputlen) + case 1: + throw new Error(errors.SIG_PARSE) + case 2: + throw new Error(errors.IMPOSSIBLE_CASE) + } + }, + + signatureImport (sig, output) { + isUint8Array('signature', sig) + output = getAssertedOutput(output, 64) + + switch (secp256k1.signatureImport(output, sig)) { + case 0: + return output + case 1: + throw new Error(errors.SIG_PARSE) + case 2: + throw new Error(errors.IMPOSSIBLE_CASE) + } + }, + + ecdsaSign (msg32, seckey, options = {}, output) { + isUint8Array('message', msg32, 32) + isUint8Array('private key', seckey, 32) + assert(toTypeString(options) === 'Object', 'Expected options to be an Object') + if (options.data !== undefined) isUint8Array('options.data', options.data) + if (options.noncefn !== undefined) assert(toTypeString(options.noncefn) === 'Function', 'Expected options.noncefn to be a Function') + output = getAssertedOutput(output, 64) + + const obj = { signature: output, recid: null } + switch (secp256k1.ecdsaSign(obj, msg32, seckey, options.data, options.noncefn)) { + case 0: + return obj + case 1: + throw new Error(errors.SIGN) + case 2: + throw new Error(errors.IMPOSSIBLE_CASE) + } + }, + + ecdsaVerify (sig, msg32, pubkey) { + isUint8Array('signature', sig, 64) + isUint8Array('message', msg32, 32) + isUint8Array('public key', pubkey, [33, 65]) + + switch (secp256k1.ecdsaVerify(sig, msg32, pubkey)) { + case 0: + return true + case 3: + return false + case 1: + throw new Error(errors.SIG_PARSE) + case 2: + throw new Error(errors.PUBKEY_PARSE) + } + }, + + ecdsaRecover (sig, recid, msg32, compressed = true, output) { + isUint8Array('signature', sig, 64) + assert( + toTypeString(recid) === 'Number' && + recid >= 0 && + recid <= 3, + 'Expected recovery id to be a Number within interval [0, 3]' + ) + isUint8Array('message', msg32, 32) + isCompressed(compressed) + output = getAssertedOutput(output, compressed ? 33 : 65) + + switch (secp256k1.ecdsaRecover(output, sig, recid, msg32)) { + case 0: + return output + case 1: + throw new Error(errors.SIG_PARSE) + case 2: + throw new Error(errors.RECOVER) + case 3: + throw new Error(errors.IMPOSSIBLE_CASE) + } + }, + + ecdh (pubkey, seckey, options = {}, output) { + isUint8Array('public key', pubkey, [33, 65]) + isUint8Array('private key', seckey, 32) + assert(toTypeString(options) === 'Object', 'Expected options to be an Object') + if (options.data !== undefined) isUint8Array('options.data', options.data) + if (options.hashfn !== undefined) { + assert(toTypeString(options.hashfn) === 'Function', 'Expected options.hashfn to be a Function') + if (options.xbuf !== undefined) isUint8Array('options.xbuf', options.xbuf, 32) + if (options.ybuf !== undefined) isUint8Array('options.ybuf', options.ybuf, 32) + isUint8Array('output', output) + } else { + output = getAssertedOutput(output, 32) + } + + switch (secp256k1.ecdh(output, pubkey, seckey, options.data, options.hashfn, options.xbuf, options.ybuf)) { + case 0: + return output + case 1: + throw new Error(errors.PUBKEY_PARSE) + case 2: + throw new Error(errors.ECDH) + } + } + } +} diff --git a/node_modules/secp256k1/package.json b/node_modules/secp256k1/package.json new file mode 100644 index 0000000..d369ff6 --- /dev/null +++ b/node_modules/secp256k1/package.json @@ -0,0 +1,43 @@ +{ + "name": "secp256k1", + "version": "5.0.0", + "description": "This module provides native bindings to ecdsa secp256k1 functions", + "keywords": [ + "ec", + "ecdh", + "ecdsa", + "secp256k1" + ], + "bugs": { + "url": "https://github.com/cryptocoinjs/secp256k1-node/issues" + }, + "repository": { + "type": "git", + "url": "https://github.com/cryptocoinjs/secp256k1-node.git" + }, + "license": "MIT", + "author": "Martin Becze ", + "contributors": [ + "Aaron Davis (https://github.com/kumavis)", + "Ian Matyssik (https://github.com/phpb-com)", + "Kagami Hiiragi (http://kagami.genshiken.org/)", + "ethers (https://github.com/ethers)", + "Kirill Fomichev (https://github.com/fanatid)" + ], + "main": "./index.js", + "browser": { + "./index.js": "./elliptic.js" + }, + "scripts": { + "install": "node-gyp-build || exit 0" + }, + "dependencies": { + "elliptic": "^6.5.4", + "node-addon-api": "^5.0.0", + "node-gyp-build": "^4.2.0" + }, + "engines": { + "node": ">=14.0.0" + }, + "gypfile": true +} \ No newline at end of file diff --git a/node_modules/secp256k1/prebuilds/darwin-x64/node.napi.node b/node_modules/secp256k1/prebuilds/darwin-x64/node.napi.node new file mode 100644 index 0000000..b136c48 Binary files /dev/null and b/node_modules/secp256k1/prebuilds/darwin-x64/node.napi.node differ diff --git a/node_modules/secp256k1/prebuilds/linux-x64/node.napi.glibc.node b/node_modules/secp256k1/prebuilds/linux-x64/node.napi.glibc.node new file mode 100644 index 0000000..b61205c Binary files /dev/null and b/node_modules/secp256k1/prebuilds/linux-x64/node.napi.glibc.node differ diff --git a/node_modules/secp256k1/prebuilds/linux-x64/node.napi.musl.node b/node_modules/secp256k1/prebuilds/linux-x64/node.napi.musl.node new file mode 100644 index 0000000..8f00988 Binary files /dev/null and b/node_modules/secp256k1/prebuilds/linux-x64/node.napi.musl.node differ diff --git a/node_modules/secp256k1/prebuilds/win32-x64/node.napi.node b/node_modules/secp256k1/prebuilds/win32-x64/node.napi.node new file mode 100644 index 0000000..3c0652d Binary files /dev/null and b/node_modules/secp256k1/prebuilds/win32-x64/node.napi.node differ diff --git a/node_modules/secp256k1/src/addon.cc b/node_modules/secp256k1/src/addon.cc new file mode 100644 index 0000000..8c5badd --- /dev/null +++ b/node_modules/secp256k1/src/addon.cc @@ -0,0 +1,10 @@ +#include + +#include + +Napi::Object Init(Napi::Env env, Napi::Object exports) { + exports.Set("Secp256k1", Secp256k1Addon::Init(env)); + return exports; +} + +NODE_API_MODULE(NODE_GYP_MODULE_NAME, Init) diff --git a/node_modules/secp256k1/src/secp256k1.cc b/node_modules/secp256k1/src/secp256k1.cc new file mode 100644 index 0000000..40ea6a5 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1.cc @@ -0,0 +1,415 @@ +#include +#include +#include +#include + +// Local helpers +#define RETURN(result) return Napi::Number::New(info.Env(), result) + +#define RETURN_INVERTED(result) RETURN(result == 1 ? 0 : 1) + +#define RETURN_IF_ZERO(result, retcode) \ + do { \ + if (result == 0) { \ + RETURN(retcode); \ + } \ + } while (0) + +#define PUBKEY_SERIALIZE(retcode) \ + do { \ + size_t outputlen = output.Length(); \ + int flags = \ + outputlen == 33 ? SECP256K1_EC_COMPRESSED : SECP256K1_EC_UNCOMPRESSED; \ + RETURN_IF_ZERO(secp256k1_ec_pubkey_serialize( \ + this->ctx_, output.Data(), &outputlen, &pubkey, flags), \ + retcode); \ + } while (0) + +// Secp256k1 +const unsigned int Secp256k1Addon::secp256k1_context_flags = + SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY; + +Napi::Value Secp256k1Addon::Init(Napi::Env env) { + Napi::Function func = DefineClass( + env, + "Secp256k1Addon", + { + InstanceMethod("contextRandomize", &Secp256k1Addon::ContextRandomize), + + InstanceMethod("privateKeyVerify", &Secp256k1Addon::PrivateKeyVerify), + InstanceMethod("privateKeyNegate", &Secp256k1Addon::PrivateKeyNegate), + InstanceMethod("privateKeyTweakAdd", + &Secp256k1Addon::PrivateKeyTweakAdd), + InstanceMethod("privateKeyTweakMul", + &Secp256k1Addon::PrivateKeyTweakMul), + + InstanceMethod("publicKeyVerify", &Secp256k1Addon::PublicKeyVerify), + InstanceMethod("publicKeyCreate", &Secp256k1Addon::PublicKeyCreate), + InstanceMethod("publicKeyConvert", &Secp256k1Addon::PublicKeyConvert), + InstanceMethod("publicKeyNegate", &Secp256k1Addon::PublicKeyNegate), + InstanceMethod("publicKeyCombine", &Secp256k1Addon::PublicKeyCombine), + InstanceMethod("publicKeyTweakAdd", + &Secp256k1Addon::PublicKeyTweakAdd), + InstanceMethod("publicKeyTweakMul", + &Secp256k1Addon::PublicKeyTweakMul), + + InstanceMethod("signatureNormalize", + &Secp256k1Addon::SignatureNormalize), + InstanceMethod("signatureExport", &Secp256k1Addon::SignatureExport), + InstanceMethod("signatureImport", &Secp256k1Addon::SignatureImport), + + InstanceMethod("ecdsaSign", &Secp256k1Addon::ECDSASign), + InstanceMethod("ecdsaVerify", &Secp256k1Addon::ECDSAVerify), + InstanceMethod("ecdsaRecover", &Secp256k1Addon::ECDSARecover), + + InstanceMethod("ecdh", &Secp256k1Addon::ECDH), + }); + + Napi::FunctionReference* constructor = new Napi::FunctionReference(); + *constructor = Napi::Persistent(func); + env.SetInstanceData(constructor); + + return func; +} + +Secp256k1Addon::Secp256k1Addon(const Napi::CallbackInfo& info) + : Napi::ObjectWrap(info) { + ctx_ = secp256k1_context_create(secp256k1_context_flags); + + size_t size = secp256k1_context_preallocated_size(secp256k1_context_flags); + Napi::MemoryManagement::AdjustExternalMemory(info.Env(), size); +} + +void Secp256k1Addon::Finalize(Napi::Env env) { + secp256k1_context_destroy(const_cast(ctx_)); + + size_t size = secp256k1_context_preallocated_size(secp256k1_context_flags); + Napi::MemoryManagement::AdjustExternalMemory(env, -size); +} + +Napi::Value Secp256k1Addon::ContextRandomize(const Napi::CallbackInfo& info) { + const unsigned char* seed32 = NULL; + if (!info[0].IsNull()) { + seed32 = info[0].As>().Data(); + } + + RETURN_INVERTED(secp256k1_context_randomize( + const_cast(this->ctx_), seed32)); +} + +// PrivateKey +Napi::Value Secp256k1Addon::PrivateKeyVerify(const Napi::CallbackInfo& info) { + auto seckey = info[0].As>().Data(); + + RETURN_INVERTED(secp256k1_ec_seckey_verify(this->ctx_, seckey)); +} + +Napi::Value Secp256k1Addon::PrivateKeyNegate(const Napi::CallbackInfo& info) { + auto seckey = info[0].As>().Data(); + + RETURN_IF_ZERO(secp256k1_ec_privkey_negate(this->ctx_, seckey), 1); + RETURN(0); +} + +Napi::Value Secp256k1Addon::PrivateKeyTweakAdd(const Napi::CallbackInfo& info) { + auto seckey = info[0].As>().Data(); + auto tweak = info[1].As>().Data(); + + RETURN_INVERTED(secp256k1_ec_privkey_tweak_add(this->ctx_, seckey, tweak)); +} + +Napi::Value Secp256k1Addon::PrivateKeyTweakMul(const Napi::CallbackInfo& info) { + auto seckey = info[0].As>().Data(); + auto tweak = info[1].As>().Data(); + + RETURN_INVERTED(secp256k1_ec_privkey_tweak_mul(this->ctx_, seckey, tweak)); +} + +// PublicKey +Napi::Value Secp256k1Addon::PublicKeyVerify(const Napi::CallbackInfo& info) { + auto input = info[0].As>(); + + secp256k1_pubkey pubkey; + RETURN_IF_ZERO(secp256k1_ec_pubkey_parse( + this->ctx_, &pubkey, input.Data(), input.Length()), + 1); + RETURN(0); +} + +Napi::Value Secp256k1Addon::PublicKeyCreate(const Napi::CallbackInfo& info) { + auto output = info[0].As>(); + auto seckey = info[1].As>().Data(); + + secp256k1_pubkey pubkey; + RETURN_IF_ZERO(secp256k1_ec_pubkey_create(this->ctx_, &pubkey, seckey), 1); + PUBKEY_SERIALIZE(2); + RETURN(0); +} + +Napi::Value Secp256k1Addon::PublicKeyConvert(const Napi::CallbackInfo& info) { + auto output = info[0].As>(); + auto input = info[1].As>(); + + secp256k1_pubkey pubkey; + RETURN_IF_ZERO(secp256k1_ec_pubkey_parse( + this->ctx_, &pubkey, input.Data(), input.Length()), + 1); + PUBKEY_SERIALIZE(2); + RETURN(0); +} + +Napi::Value Secp256k1Addon::PublicKeyNegate(const Napi::CallbackInfo& info) { + auto output = info[0].As>(); + auto input = info[1].As>(); + + secp256k1_pubkey pubkey; + RETURN_IF_ZERO(secp256k1_ec_pubkey_parse( + this->ctx_, &pubkey, input.Data(), input.Length()), + 1); + RETURN_IF_ZERO(secp256k1_ec_pubkey_negate(this->ctx_, &pubkey), 2); + PUBKEY_SERIALIZE(3); + RETURN(0); +} + +Napi::Value Secp256k1Addon::PublicKeyCombine(const Napi::CallbackInfo& info) { + auto output = info[0].As>(); + auto inputs = info[1].As(); + + std::unique_ptr pubkeys( + new secp256k1_pubkey[inputs.Length()]); + std::unique_ptr ptrs( + new secp256k1_pubkey*[inputs.Length()]); + for (unsigned int i = 0; i < inputs.Length(); ++i) { + auto input = inputs.Get(i).As>(); + RETURN_IF_ZERO(secp256k1_ec_pubkey_parse( + this->ctx_, &pubkeys[i], input.Data(), input.Length()), + 1); + ptrs[i] = &pubkeys[i]; + } + + secp256k1_pubkey pubkey; + RETURN_IF_ZERO(secp256k1_ec_pubkey_combine( + this->ctx_, &pubkey, ptrs.get(), inputs.Length()), + 2); + PUBKEY_SERIALIZE(3); + RETURN(0); +} + +Napi::Value Secp256k1Addon::PublicKeyTweakAdd(const Napi::CallbackInfo& info) { + auto output = info[0].As>(); + auto input = info[1].As>(); + auto tweak = info[2].As>().Data(); + + secp256k1_pubkey pubkey; + RETURN_IF_ZERO(secp256k1_ec_pubkey_parse( + this->ctx_, &pubkey, input.Data(), input.Length()), + 1); + RETURN_IF_ZERO(secp256k1_ec_pubkey_tweak_add(this->ctx_, &pubkey, tweak), 2); + PUBKEY_SERIALIZE(3); + RETURN(0); +} + +Napi::Value Secp256k1Addon::PublicKeyTweakMul(const Napi::CallbackInfo& info) { + auto output = info[0].As>(); + auto input = info[1].As>(); + auto tweak = info[2].As>().Data(); + + secp256k1_pubkey pubkey; + RETURN_IF_ZERO(secp256k1_ec_pubkey_parse( + this->ctx_, &pubkey, input.Data(), input.Length()), + 1); + RETURN_IF_ZERO(secp256k1_ec_pubkey_tweak_mul(this->ctx_, &pubkey, tweak), 2); + PUBKEY_SERIALIZE(3); + RETURN(0); +} + +// Signature +Napi::Value Secp256k1Addon::SignatureNormalize(const Napi::CallbackInfo& info) { + auto sig = info[0].As>().Data(); + + secp256k1_ecdsa_signature sigin, sigout; + RETURN_IF_ZERO( + secp256k1_ecdsa_signature_parse_compact(this->ctx_, &sigin, sig), 1); + secp256k1_ecdsa_signature_normalize(this->ctx_, &sigout, &sigin); + secp256k1_ecdsa_signature_serialize_compact(this->ctx_, sig, &sigout); + RETURN(0); +} + +Napi::Value Secp256k1Addon::SignatureExport(const Napi::CallbackInfo& info) { + auto obj = info[0].As(); + auto output = obj.Get("output").As>().Data(); + size_t outputlen = 72; + auto input = info[1].As>().Data(); + + secp256k1_ecdsa_signature sig; + RETURN_IF_ZERO( + secp256k1_ecdsa_signature_parse_compact(this->ctx_, &sig, input), 1); + RETURN_IF_ZERO(secp256k1_ecdsa_signature_serialize_der( + this->ctx_, output, &outputlen, &sig), + 2); + + obj.Set("outputlen", outputlen); + RETURN(0); +} + +Napi::Value Secp256k1Addon::SignatureImport(const Napi::CallbackInfo& info) { + auto output = info[0].As>().Data(); + auto input = info[1].As>(); + + secp256k1_ecdsa_signature sig; + RETURN_IF_ZERO(secp256k1_ecdsa_signature_parse_der( + this->ctx_, &sig, input.Data(), input.Length()), + 1); + RETURN_IF_ZERO( + secp256k1_ecdsa_signature_serialize_compact(this->ctx_, output, &sig), 2); + RETURN(0); +} + +// ECDSA +int ecdsa_sign_nonce_function(unsigned char* nonce32, + const unsigned char* msg32, + const unsigned char* key32, + const unsigned char* algo16, + void* data, + unsigned int counter) { + auto obj = static_cast(data); + auto env = Napi::Env(obj->env); + + auto result = obj->fn.Call({obj->msg32, + obj->key32, + env.Null(), + obj->data, + Napi::Number::New(env, counter)}); + if (!result.IsTypedArray()) return 0; + if (result.As().ByteLength() != 32) return 0; + + memcpy(nonce32, result.As().Data(), 32); + return 1; +} + +Napi::Value Secp256k1Addon::ECDSASign(const Napi::CallbackInfo& info) { + auto obj = info[0].As(); + auto output = obj.Get("signature").As>().Data(); + int recid; + auto msg32 = info[1].As>().Data(); + auto seckey = info[2].As>().Data(); + + void* data = NULL; + if (!info[3].IsUndefined()) { + data = info[3].As>().Data(); + } + + secp256k1_nonce_function noncefn = secp256k1_nonce_function_rfc6979; + if (!info[4].IsUndefined()) { + this->ecdsa_sign_data.env = info.Env(); + this->ecdsa_sign_data.fn = info[4].As(); + this->ecdsa_sign_data.msg32 = info[1]; + this->ecdsa_sign_data.key32 = info[2]; + this->ecdsa_sign_data.data = + info[3].IsUndefined() ? info.Env().Null() : info[3]; + + noncefn = ecdsa_sign_nonce_function; + data = static_cast(&this->ecdsa_sign_data); + } + + secp256k1_ecdsa_recoverable_signature sig; + RETURN_IF_ZERO(secp256k1_ecdsa_sign_recoverable( + this->ctx_, &sig, msg32, seckey, noncefn, data), + 1); + + RETURN_IF_ZERO(secp256k1_ecdsa_recoverable_signature_serialize_compact( + this->ctx_, output, &recid, &sig), + 2); + + obj.Set("recid", recid); + RETURN(0); +} + +Napi::Value Secp256k1Addon::ECDSAVerify(const Napi::CallbackInfo& info) { + auto sigraw = info[0].As>().Data(); + auto msg32 = info[1].As>().Data(); + auto input = info[2].As>(); + + secp256k1_ecdsa_signature sig; + RETURN_IF_ZERO( + secp256k1_ecdsa_signature_parse_compact(this->ctx_, &sig, sigraw), 1); + + secp256k1_pubkey pubkey; + RETURN_IF_ZERO(secp256k1_ec_pubkey_parse( + this->ctx_, &pubkey, input.Data(), input.Length()), + 2); + + RETURN_IF_ZERO(secp256k1_ecdsa_verify(this->ctx_, &sig, msg32, &pubkey), 3); + RETURN(0); +} + +Napi::Value Secp256k1Addon::ECDSARecover(const Napi::CallbackInfo& info) { + auto output = info[0].As>(); + auto sigraw = info[1].As>().Data(); + auto recid = info[2].As().Int32Value(); + auto msg32 = info[3].As>().Data(); + + secp256k1_ecdsa_recoverable_signature sig; + RETURN_IF_ZERO(secp256k1_ecdsa_recoverable_signature_parse_compact( + this->ctx_, &sig, sigraw, recid), + 1); + + secp256k1_pubkey pubkey; + RETURN_IF_ZERO(secp256k1_ecdsa_recover(this->ctx_, &pubkey, &sig, msg32), 2); + + PUBKEY_SERIALIZE(3); + RETURN(0); +} + +// ECDH +int ecdh_hash_function(unsigned char* output, + const unsigned char* x, + const unsigned char* y, + void* data) { + auto obj = static_cast(data); + + memcpy(obj->xbuf.As().Data(), x, 32); + memcpy(obj->ybuf.As().Data(), y, 32); + + auto result = obj->fn.Call({obj->xbuf, obj->ybuf, obj->data}); + if (!result.IsTypedArray()) return 0; + if (result.As().ByteLength() != obj->outputlen) return 0; + + memcpy(output, result.As().Data(), obj->outputlen); + return 1; +} + +Napi::Value Secp256k1Addon::ECDH(const Napi::CallbackInfo& info) { + auto output = info[0].As>(); + auto input = info[1].As>(); + auto seckey = info[2].As>().Data(); + + void* data = NULL; + if (!info[3].IsUndefined()) { + data = info[3].As>().Data(); + } + + secp256k1_ecdh_hash_function hashfn = secp256k1_ecdh_hash_function_sha256; + if (!info[4].IsUndefined()) { + auto env = info.Env(); + this->ecdh_data.fn = info[4].As(); + this->ecdh_data.xbuf = + info[5].IsUndefined() ? Napi::Uint8Array::New(env, 32) : info[5]; + this->ecdh_data.ybuf = + info[6].IsUndefined() ? Napi::Uint8Array::New(env, 32) : info[6]; + this->ecdh_data.data = info[3].IsUndefined() ? env.Null() : info[3]; + this->ecdh_data.outputlen = output.Length(); + + hashfn = ecdh_hash_function; + data = static_cast(&this->ecdh_data); + } + + secp256k1_pubkey pubkey; + RETURN_IF_ZERO(secp256k1_ec_pubkey_parse( + this->ctx_, &pubkey, input.Data(), input.Length()), + 1); + RETURN_IF_ZERO( + secp256k1_ecdh(this->ctx_, output.Data(), &pubkey, seckey, hashfn, data), + 2); + RETURN(0); +} diff --git a/node_modules/secp256k1/src/secp256k1.h b/node_modules/secp256k1/src/secp256k1.h new file mode 100644 index 0000000..5f3e125 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1.h @@ -0,0 +1,62 @@ +#ifndef ADDON_SECP256K1 +#define ADDON_SECP256K1 + +#include +#include + +class Secp256k1Addon : public Napi::ObjectWrap { + public: + static Napi::Value Init(Napi::Env env); + + Secp256k1Addon(const Napi::CallbackInfo& info); + void Finalize(Napi::Env env); + + struct ECDSASignData { + napi_env env; + Napi::Function fn; + Napi::Value msg32; + Napi::Value key32; + Napi::Value data; + }; + + struct ECDHData { + Napi::Function fn; + Napi::Value xbuf; + Napi::Value ybuf; + Napi::Value data; + size_t outputlen; + }; + + private: + static const unsigned int secp256k1_context_flags; + const secp256k1_context* ctx_; + ECDSASignData ecdsa_sign_data; + ECDHData ecdh_data; + + Napi::Value ContextRandomize(const Napi::CallbackInfo& info); + + Napi::Value PrivateKeyVerify(const Napi::CallbackInfo& info); + Napi::Value PrivateKeyNegate(const Napi::CallbackInfo& info); + Napi::Value PrivateKeyTweakAdd(const Napi::CallbackInfo& info); + Napi::Value PrivateKeyTweakMul(const Napi::CallbackInfo& info); + + Napi::Value PublicKeyVerify(const Napi::CallbackInfo& info); + Napi::Value PublicKeyCreate(const Napi::CallbackInfo& info); + Napi::Value PublicKeyConvert(const Napi::CallbackInfo& info); + Napi::Value PublicKeyNegate(const Napi::CallbackInfo& info); + Napi::Value PublicKeyCombine(const Napi::CallbackInfo& info); + Napi::Value PublicKeyTweakAdd(const Napi::CallbackInfo& info); + Napi::Value PublicKeyTweakMul(const Napi::CallbackInfo& info); + + Napi::Value SignatureNormalize(const Napi::CallbackInfo& info); + Napi::Value SignatureExport(const Napi::CallbackInfo& info); + Napi::Value SignatureImport(const Napi::CallbackInfo& info); + + Napi::Value ECDSASign(const Napi::CallbackInfo& info); + Napi::Value ECDSAVerify(const Napi::CallbackInfo& info); + Napi::Value ECDSARecover(const Napi::CallbackInfo& info); + + Napi::Value ECDH(const Napi::CallbackInfo& info); +}; + +#endif // ADDON_SECP256K1 diff --git a/node_modules/secp256k1/src/secp256k1/.npmignore b/node_modules/secp256k1/src/secp256k1/.npmignore new file mode 100644 index 0000000..55d325a --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/.npmignore @@ -0,0 +1,50 @@ +bench_inv +bench_ecdh +bench_ecmult +bench_sign +bench_verify +bench_schnorr_verify +bench_recover +bench_internal +tests +exhaustive_tests +gen_context +*.exe +*.so +*.a +!.gitignore + +Makefile +configure +.libs/ +Makefile.in +aclocal.m4 +autom4te.cache/ +config.log +config.status +*.tar.gz +*.la +libtool +.deps/ +.dirstamp +*.lo +*.o +*~ +src/libsecp256k1-config.h +src/libsecp256k1-config.h.in +src/ecmult_static_context.h +build-aux/config.guess +build-aux/config.sub +build-aux/depcomp +build-aux/install-sh +build-aux/ltmain.sh +build-aux/m4/libtool.m4 +build-aux/m4/lt~obsolete.m4 +build-aux/m4/ltoptions.m4 +build-aux/m4/ltsugar.m4 +build-aux/m4/ltversion.m4 +build-aux/missing +build-aux/compile +build-aux/test-driver +src/stamp-h1 +libsecp256k1.pc diff --git a/node_modules/secp256k1/src/secp256k1/.travis.yml b/node_modules/secp256k1/src/secp256k1/.travis.yml new file mode 100644 index 0000000..4d5adf7 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/.travis.yml @@ -0,0 +1,100 @@ +language: c +os: linux +addons: + apt: + packages: libgmp-dev +compiler: + - clang + - gcc +cache: + directories: + - src/java/guava/ +env: + global: + - FIELD=auto BIGNUM=auto SCALAR=auto ENDOMORPHISM=no STATICPRECOMPUTATION=yes ECMULTGENPRECISION=auto ASM=no BUILD=check EXTRAFLAGS= HOST= ECDH=no RECOVERY=no EXPERIMENTAL=no JNI=no + - GUAVA_URL=https://search.maven.org/remotecontent?filepath=com/google/guava/guava/18.0/guava-18.0.jar GUAVA_JAR=src/java/guava/guava-18.0.jar + matrix: + - SCALAR=32bit RECOVERY=yes + - SCALAR=32bit FIELD=32bit ECDH=yes EXPERIMENTAL=yes + - SCALAR=64bit + - FIELD=64bit RECOVERY=yes + - FIELD=64bit ENDOMORPHISM=yes + - FIELD=64bit ENDOMORPHISM=yes ECDH=yes EXPERIMENTAL=yes + - FIELD=64bit ASM=x86_64 + - FIELD=64bit ENDOMORPHISM=yes ASM=x86_64 + - FIELD=32bit ENDOMORPHISM=yes + - BIGNUM=no + - BIGNUM=no ENDOMORPHISM=yes RECOVERY=yes EXPERIMENTAL=yes + - BIGNUM=no STATICPRECOMPUTATION=no + - BUILD=distcheck + - EXTRAFLAGS=CPPFLAGS=-DDETERMINISTIC + - EXTRAFLAGS=CFLAGS=-O0 + - BUILD=check-java JNI=yes ECDH=yes EXPERIMENTAL=yes + - ECMULTGENPRECISION=2 + - ECMULTGENPRECISION=8 +matrix: + fast_finish: true + include: + - compiler: clang + env: HOST=i686-linux-gnu ENDOMORPHISM=yes + addons: + apt: + packages: + - gcc-multilib + - libgmp-dev:i386 + - compiler: clang + env: HOST=i686-linux-gnu + addons: + apt: + packages: + - gcc-multilib + - compiler: gcc + env: HOST=i686-linux-gnu ENDOMORPHISM=yes + addons: + apt: + packages: + - gcc-multilib + - compiler: gcc + env: HOST=i686-linux-gnu + addons: + apt: + packages: + - gcc-multilib + - libgmp-dev:i386 + - compiler: gcc + env: + - BIGNUM=no ENDOMORPHISM=yes ASM=x86_64 EXPERIMENTAL=yes ECDH=yes RECOVERY=yes + - VALGRIND=yes EXTRAFLAGS="--disable-openssl-tests CPPFLAGS=-DVALGRIND" BUILD= + addons: + apt: + packages: + - valgrind + - compiler: gcc + env: # The same as above but without endomorphism. + - BIGNUM=no ENDOMORPHISM=no ASM=x86_64 EXPERIMENTAL=yes ECDH=yes RECOVERY=yes + - VALGRIND=yes EXTRAFLAGS="--disable-openssl-tests CPPFLAGS=-DVALGRIND" BUILD= + addons: + apt: + packages: + - valgrind + +before_install: mkdir -p `dirname $GUAVA_JAR` +install: if [ ! -f $GUAVA_JAR ]; then wget $GUAVA_URL -O $GUAVA_JAR; fi +before_script: ./autogen.sh + +script: + - if [ -n "$HOST" ]; then export USE_HOST="--host=$HOST"; fi + - if [ "x$HOST" = "xi686-linux-gnu" ]; then export CC="$CC -m32"; fi + - ./configure --enable-experimental=$EXPERIMENTAL --enable-endomorphism=$ENDOMORPHISM --with-field=$FIELD --with-bignum=$BIGNUM --with-asm=$ASM --with-scalar=$SCALAR --enable-ecmult-static-precomputation=$STATICPRECOMPUTATION --with-ecmult-gen-precision=$ECMULTGENPRECISION --enable-module-ecdh=$ECDH --enable-module-recovery=$RECOVERY --enable-jni=$JNI $EXTRAFLAGS $USE_HOST + - if [ -n "$BUILD" ]; then make -j2 $BUILD; fi + - # travis_wait extends the 10 minutes without output allowed (https://docs.travis-ci.com/user/common-build-problems/#build-times-out-because-no-output-was-received) + - # the `--error-exitcode` is required to make the test fail if valgrind found errors, otherwise it'll return 0 (http://valgrind.org/docs/manual/manual-core.html) + - if [ -n "$VALGRIND" ]; then + make -j2 && + travis_wait 30 valgrind --error-exitcode=42 ./tests 16 && + travis_wait 30 valgrind --error-exitcode=42 ./exhaustive_tests; + fi + +after_script: + - cat ./tests.log + - cat ./exhaustive_tests.log diff --git a/node_modules/secp256k1/src/secp256k1/COPYING b/node_modules/secp256k1/src/secp256k1/COPYING new file mode 100644 index 0000000..4522a59 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/COPYING @@ -0,0 +1,19 @@ +Copyright (c) 2013 Pieter Wuille + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/node_modules/secp256k1/src/secp256k1/Makefile.am b/node_modules/secp256k1/src/secp256k1/Makefile.am new file mode 100644 index 0000000..f420944 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/Makefile.am @@ -0,0 +1,183 @@ +ACLOCAL_AMFLAGS = -I build-aux/m4 + +lib_LTLIBRARIES = libsecp256k1.la +if USE_JNI +JNI_LIB = libsecp256k1_jni.la +noinst_LTLIBRARIES = $(JNI_LIB) +else +JNI_LIB = +endif +include_HEADERS = include/secp256k1.h +include_HEADERS += include/secp256k1_preallocated.h +noinst_HEADERS = +noinst_HEADERS += src/scalar.h +noinst_HEADERS += src/scalar_4x64.h +noinst_HEADERS += src/scalar_8x32.h +noinst_HEADERS += src/scalar_low.h +noinst_HEADERS += src/scalar_impl.h +noinst_HEADERS += src/scalar_4x64_impl.h +noinst_HEADERS += src/scalar_8x32_impl.h +noinst_HEADERS += src/scalar_low_impl.h +noinst_HEADERS += src/group.h +noinst_HEADERS += src/group_impl.h +noinst_HEADERS += src/num_gmp.h +noinst_HEADERS += src/num_gmp_impl.h +noinst_HEADERS += src/ecdsa.h +noinst_HEADERS += src/ecdsa_impl.h +noinst_HEADERS += src/eckey.h +noinst_HEADERS += src/eckey_impl.h +noinst_HEADERS += src/ecmult.h +noinst_HEADERS += src/ecmult_impl.h +noinst_HEADERS += src/ecmult_const.h +noinst_HEADERS += src/ecmult_const_impl.h +noinst_HEADERS += src/ecmult_gen.h +noinst_HEADERS += src/ecmult_gen_impl.h +noinst_HEADERS += src/num.h +noinst_HEADERS += src/num_impl.h +noinst_HEADERS += src/field_10x26.h +noinst_HEADERS += src/field_10x26_impl.h +noinst_HEADERS += src/field_5x52.h +noinst_HEADERS += src/field_5x52_impl.h +noinst_HEADERS += src/field_5x52_int128_impl.h +noinst_HEADERS += src/field_5x52_asm_impl.h +noinst_HEADERS += src/java/org_bitcoin_NativeSecp256k1.h +noinst_HEADERS += src/java/org_bitcoin_Secp256k1Context.h +noinst_HEADERS += src/util.h +noinst_HEADERS += src/scratch.h +noinst_HEADERS += src/scratch_impl.h +noinst_HEADERS += src/testrand.h +noinst_HEADERS += src/testrand_impl.h +noinst_HEADERS += src/hash.h +noinst_HEADERS += src/hash_impl.h +noinst_HEADERS += src/field.h +noinst_HEADERS += src/field_impl.h +noinst_HEADERS += src/bench.h +noinst_HEADERS += contrib/lax_der_parsing.h +noinst_HEADERS += contrib/lax_der_parsing.c +noinst_HEADERS += contrib/lax_der_privatekey_parsing.h +noinst_HEADERS += contrib/lax_der_privatekey_parsing.c + +if USE_EXTERNAL_ASM +COMMON_LIB = libsecp256k1_common.la +noinst_LTLIBRARIES = $(COMMON_LIB) +else +COMMON_LIB = +endif + +pkgconfigdir = $(libdir)/pkgconfig +pkgconfig_DATA = libsecp256k1.pc + +if USE_EXTERNAL_ASM +if USE_ASM_ARM +libsecp256k1_common_la_SOURCES = src/asm/field_10x26_arm.s +endif +endif + +libsecp256k1_la_SOURCES = src/secp256k1.c +libsecp256k1_la_CPPFLAGS = -DSECP256K1_BUILD -I$(top_srcdir)/include -I$(top_srcdir)/src $(SECP_INCLUDES) +libsecp256k1_la_LIBADD = $(JNI_LIB) $(SECP_LIBS) $(COMMON_LIB) + +libsecp256k1_jni_la_SOURCES = src/java/org_bitcoin_NativeSecp256k1.c src/java/org_bitcoin_Secp256k1Context.c +libsecp256k1_jni_la_CPPFLAGS = -DSECP256K1_BUILD $(JNI_INCLUDES) + +noinst_PROGRAMS = +if USE_BENCHMARK +noinst_PROGRAMS += bench_verify bench_sign bench_internal bench_ecmult +bench_verify_SOURCES = src/bench_verify.c +bench_verify_LDADD = libsecp256k1.la $(SECP_LIBS) $(SECP_TEST_LIBS) $(COMMON_LIB) +bench_sign_SOURCES = src/bench_sign.c +bench_sign_LDADD = libsecp256k1.la $(SECP_LIBS) $(SECP_TEST_LIBS) $(COMMON_LIB) +bench_internal_SOURCES = src/bench_internal.c +bench_internal_LDADD = $(SECP_LIBS) $(COMMON_LIB) +bench_internal_CPPFLAGS = -DSECP256K1_BUILD $(SECP_INCLUDES) +bench_ecmult_SOURCES = src/bench_ecmult.c +bench_ecmult_LDADD = $(SECP_LIBS) $(COMMON_LIB) +bench_ecmult_CPPFLAGS = -DSECP256K1_BUILD $(SECP_INCLUDES) +endif + +TESTS = +if USE_TESTS +noinst_PROGRAMS += tests +tests_SOURCES = src/tests.c +tests_CPPFLAGS = -DSECP256K1_BUILD -I$(top_srcdir)/src -I$(top_srcdir)/include $(SECP_INCLUDES) $(SECP_TEST_INCLUDES) +if !ENABLE_COVERAGE +tests_CPPFLAGS += -DVERIFY +endif +tests_LDADD = $(SECP_LIBS) $(SECP_TEST_LIBS) $(COMMON_LIB) +tests_LDFLAGS = -static +TESTS += tests +endif + +if USE_EXHAUSTIVE_TESTS +noinst_PROGRAMS += exhaustive_tests +exhaustive_tests_SOURCES = src/tests_exhaustive.c +exhaustive_tests_CPPFLAGS = -DSECP256K1_BUILD -I$(top_srcdir)/src $(SECP_INCLUDES) +if !ENABLE_COVERAGE +exhaustive_tests_CPPFLAGS += -DVERIFY +endif +exhaustive_tests_LDADD = $(SECP_LIBS) $(COMMON_LIB) +exhaustive_tests_LDFLAGS = -static +TESTS += exhaustive_tests +endif + +JAVAROOT=src/java +JAVAORG=org/bitcoin +JAVA_GUAVA=$(srcdir)/$(JAVAROOT)/guava/guava-18.0.jar +CLASSPATH_ENV=CLASSPATH=$(JAVA_GUAVA) +JAVA_FILES= \ + $(JAVAROOT)/$(JAVAORG)/NativeSecp256k1.java \ + $(JAVAROOT)/$(JAVAORG)/NativeSecp256k1Test.java \ + $(JAVAROOT)/$(JAVAORG)/NativeSecp256k1Util.java \ + $(JAVAROOT)/$(JAVAORG)/Secp256k1Context.java + +if USE_JNI + +$(JAVA_GUAVA): + @echo Guava is missing. Fetch it via: \ + wget https://search.maven.org/remotecontent?filepath=com/google/guava/guava/18.0/guava-18.0.jar -O $(@) + @false + +.stamp-java: $(JAVA_FILES) + @echo Compiling $^ + $(AM_V_at)$(CLASSPATH_ENV) javac $^ + @touch $@ + +if USE_TESTS + +check-java: libsecp256k1.la $(JAVA_GUAVA) .stamp-java + $(AM_V_at)java -Djava.library.path="./:./src:./src/.libs:.libs/" -cp "$(JAVA_GUAVA):$(JAVAROOT)" $(JAVAORG)/NativeSecp256k1Test + +endif +endif + +if USE_ECMULT_STATIC_PRECOMPUTATION +CPPFLAGS_FOR_BUILD +=-I$(top_srcdir) -I$(builddir)/src + +gen_context_OBJECTS = gen_context.o +gen_context_BIN = gen_context$(BUILD_EXEEXT) +gen_%.o: src/gen_%.c src/libsecp256k1-config.h + $(CC_FOR_BUILD) $(CPPFLAGS_FOR_BUILD) $(CFLAGS_FOR_BUILD) -c $< -o $@ + +$(gen_context_BIN): $(gen_context_OBJECTS) + $(CC_FOR_BUILD) $(CFLAGS_FOR_BUILD) $(LDFLAGS_FOR_BUILD) $^ -o $@ + +$(libsecp256k1_la_OBJECTS): src/ecmult_static_context.h +$(tests_OBJECTS): src/ecmult_static_context.h +$(bench_internal_OBJECTS): src/ecmult_static_context.h +$(bench_ecmult_OBJECTS): src/ecmult_static_context.h + +src/ecmult_static_context.h: $(gen_context_BIN) + ./$(gen_context_BIN) + +CLEANFILES = $(gen_context_BIN) src/ecmult_static_context.h $(JAVAROOT)/$(JAVAORG)/*.class .stamp-java +endif + +EXTRA_DIST = autogen.sh src/gen_context.c src/basic-config.h $(JAVA_FILES) + +if ENABLE_MODULE_ECDH +include src/modules/ecdh/Makefile.am.include +endif + +if ENABLE_MODULE_RECOVERY +include src/modules/recovery/Makefile.am.include +endif diff --git a/node_modules/secp256k1/src/secp256k1/README.md b/node_modules/secp256k1/src/secp256k1/README.md new file mode 100644 index 0000000..e7eebe2 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/README.md @@ -0,0 +1,78 @@ +libsecp256k1 +============ + +[![Build Status](https://travis-ci.org/bitcoin-core/secp256k1.svg?branch=master)](https://travis-ci.org/bitcoin-core/secp256k1) + +Optimized C library for EC operations on curve secp256k1. + +This library is a work in progress and is being used to research best practices. Use at your own risk. + +Features: +* secp256k1 ECDSA signing/verification and key generation. +* Adding/multiplying private/public keys. +* Serialization/parsing of private keys, public keys, signatures. +* Constant time, constant memory access signing and pubkey generation. +* Derandomized DSA (via RFC6979 or with a caller provided function.) +* Very efficient implementation. + +Implementation details +---------------------- + +* General + * No runtime heap allocation. + * Extensive testing infrastructure. + * Structured to facilitate review and analysis. + * Intended to be portable to any system with a C89 compiler and uint64_t support. + * No use of floating types. + * Expose only higher level interfaces to minimize the API surface and improve application security. ("Be difficult to use insecurely.") +* Field operations + * Optimized implementation of arithmetic modulo the curve's field size (2^256 - 0x1000003D1). + * Using 5 52-bit limbs (including hand-optimized assembly for x86_64, by Diederik Huys). + * Using 10 26-bit limbs. + * Field inverses and square roots using a sliding window over blocks of 1s (by Peter Dettman). +* Scalar operations + * Optimized implementation without data-dependent branches of arithmetic modulo the curve's order. + * Using 4 64-bit limbs (relying on __int128 support in the compiler). + * Using 8 32-bit limbs. +* Group operations + * Point addition formula specifically simplified for the curve equation (y^2 = x^3 + 7). + * Use addition between points in Jacobian and affine coordinates where possible. + * Use a unified addition/doubling formula where necessary to avoid data-dependent branches. + * Point/x comparison without a field inversion by comparison in the Jacobian coordinate space. +* Point multiplication for verification (a*P + b*G). + * Use wNAF notation for point multiplicands. + * Use a much larger window for multiples of G, using precomputed multiples. + * Use Shamir's trick to do the multiplication with the public key and the generator simultaneously. + * Optionally (off by default) use secp256k1's efficiently-computable endomorphism to split the P multiplicand into 2 half-sized ones. +* Point multiplication for signing + * Use a precomputed table of multiples of powers of 16 multiplied with the generator, so general multiplication becomes a series of additions. + * Intended to be completely free of timing sidechannels for secret-key operations (on reasonable hardware/toolchains) + * Access the table with branch-free conditional moves so memory access is uniform. + * No data-dependent branches + * Optional runtime blinding which attempts to frustrate differential power analysis. + * The precomputed tables add and eventually subtract points for which no known scalar (private key) is known, preventing even an attacker with control over the private key used to control the data internally. + +Build steps +----------- + +libsecp256k1 is built using autotools: + + $ ./autogen.sh + $ ./configure + $ make + $ make check + $ sudo make install # optional + +Exhaustive tests +----------- + + $ ./exhaustive_tests + +With valgrind, you might need to increase the max stack size: + + $ valgrind --max-stackframe=2500000 ./exhaustive_tests + +Reporting a vulnerability +------------ + +See [SECURITY.md](SECURITY.md) diff --git a/node_modules/secp256k1/src/secp256k1/SECURITY.md b/node_modules/secp256k1/src/secp256k1/SECURITY.md new file mode 100644 index 0000000..0e4d588 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/SECURITY.md @@ -0,0 +1,15 @@ +# Security Policy + +## Reporting a Vulnerability + +To report security issues send an email to secp256k1-security@bitcoincore.org (not for support). + +The following keys may be used to communicate sensitive information to developers: + +| Name | Fingerprint | +|------|-------------| +| Pieter Wuille | 133E AC17 9436 F14A 5CF1 B794 860F EB80 4E66 9320 | +| Andrew Poelstra | 699A 63EF C17A D3A9 A34C FFC0 7AD0 A91C 40BD 0091 | +| Tim Ruffing | 09E0 3F87 1092 E40E 106E 902B 33BC 86AB 80FF 5516 | + +You can import a key by running the following command with that individual’s fingerprint: `gpg --recv-keys ""` Ensure that you put quotes around fingerprints containing spaces. diff --git a/node_modules/secp256k1/src/secp256k1/TODO b/node_modules/secp256k1/src/secp256k1/TODO new file mode 100644 index 0000000..a300e1c --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/TODO @@ -0,0 +1,3 @@ +* Unit tests for fieldelem/groupelem, including ones intended to + trigger fieldelem's boundary cases. +* Complete constant-time operations for signing/keygen diff --git a/node_modules/secp256k1/src/secp256k1/autogen.sh b/node_modules/secp256k1/src/secp256k1/autogen.sh new file mode 100755 index 0000000..65286b9 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/autogen.sh @@ -0,0 +1,3 @@ +#!/bin/sh +set -e +autoreconf -if --warnings=all diff --git a/node_modules/secp256k1/src/secp256k1/build-aux/m4/ax_jni_include_dir.m4 b/node_modules/secp256k1/src/secp256k1/build-aux/m4/ax_jni_include_dir.m4 new file mode 100644 index 0000000..cdc78d8 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/build-aux/m4/ax_jni_include_dir.m4 @@ -0,0 +1,145 @@ +# =========================================================================== +# https://www.gnu.org/software/autoconf-archive/ax_jni_include_dir.html +# =========================================================================== +# +# SYNOPSIS +# +# AX_JNI_INCLUDE_DIR +# +# DESCRIPTION +# +# AX_JNI_INCLUDE_DIR finds include directories needed for compiling +# programs using the JNI interface. +# +# JNI include directories are usually in the Java distribution. This is +# deduced from the value of $JAVA_HOME, $JAVAC, or the path to "javac", in +# that order. When this macro completes, a list of directories is left in +# the variable JNI_INCLUDE_DIRS. +# +# Example usage follows: +# +# AX_JNI_INCLUDE_DIR +# +# for JNI_INCLUDE_DIR in $JNI_INCLUDE_DIRS +# do +# CPPFLAGS="$CPPFLAGS -I$JNI_INCLUDE_DIR" +# done +# +# If you want to force a specific compiler: +# +# - at the configure.in level, set JAVAC=yourcompiler before calling +# AX_JNI_INCLUDE_DIR +# +# - at the configure level, setenv JAVAC +# +# Note: This macro can work with the autoconf M4 macros for Java programs. +# This particular macro is not part of the original set of macros. +# +# LICENSE +# +# Copyright (c) 2008 Don Anderson +# +# Copying and distribution of this file, with or without modification, are +# permitted in any medium without royalty provided the copyright notice +# and this notice are preserved. This file is offered as-is, without any +# warranty. + +#serial 14 + +AU_ALIAS([AC_JNI_INCLUDE_DIR], [AX_JNI_INCLUDE_DIR]) +AC_DEFUN([AX_JNI_INCLUDE_DIR],[ + +JNI_INCLUDE_DIRS="" + +if test "x$JAVA_HOME" != x; then + _JTOPDIR="$JAVA_HOME" +else + if test "x$JAVAC" = x; then + JAVAC=javac + fi + AC_PATH_PROG([_ACJNI_JAVAC], [$JAVAC], [no]) + if test "x$_ACJNI_JAVAC" = xno; then + AC_MSG_WARN([cannot find JDK; try setting \$JAVAC or \$JAVA_HOME]) + fi + _ACJNI_FOLLOW_SYMLINKS("$_ACJNI_JAVAC") + _JTOPDIR=`echo "$_ACJNI_FOLLOWED" | sed -e 's://*:/:g' -e 's:/[[^/]]*$::'` +fi + +case "$host_os" in + darwin*) # Apple Java headers are inside the Xcode bundle. + macos_version=$(sw_vers -productVersion | sed -n -e 's/^@<:@0-9@:>@*.\(@<:@0-9@:>@*\).@<:@0-9@:>@*/\1/p') + if @<:@ "$macos_version" -gt "7" @:>@; then + _JTOPDIR="$(xcrun --show-sdk-path)/System/Library/Frameworks/JavaVM.framework" + _JINC="$_JTOPDIR/Headers" + else + _JTOPDIR="/System/Library/Frameworks/JavaVM.framework" + _JINC="$_JTOPDIR/Headers" + fi + ;; + *) _JINC="$_JTOPDIR/include";; +esac +_AS_ECHO_LOG([_JTOPDIR=$_JTOPDIR]) +_AS_ECHO_LOG([_JINC=$_JINC]) + +# On Mac OS X 10.6.4, jni.h is a symlink: +# /System/Library/Frameworks/JavaVM.framework/Versions/Current/Headers/jni.h +# -> ../../CurrentJDK/Headers/jni.h. +AC_CACHE_CHECK(jni headers, ac_cv_jni_header_path, +[ + if test -f "$_JINC/jni.h"; then + ac_cv_jni_header_path="$_JINC" + JNI_INCLUDE_DIRS="$JNI_INCLUDE_DIRS $ac_cv_jni_header_path" + else + _JTOPDIR=`echo "$_JTOPDIR" | sed -e 's:/[[^/]]*$::'` + if test -f "$_JTOPDIR/include/jni.h"; then + ac_cv_jni_header_path="$_JTOPDIR/include" + JNI_INCLUDE_DIRS="$JNI_INCLUDE_DIRS $ac_cv_jni_header_path" + else + ac_cv_jni_header_path=none + fi + fi +]) + +# get the likely subdirectories for system specific java includes +case "$host_os" in +bsdi*) _JNI_INC_SUBDIRS="bsdos";; +freebsd*) _JNI_INC_SUBDIRS="freebsd";; +darwin*) _JNI_INC_SUBDIRS="darwin";; +linux*) _JNI_INC_SUBDIRS="linux genunix";; +osf*) _JNI_INC_SUBDIRS="alpha";; +solaris*) _JNI_INC_SUBDIRS="solaris";; +mingw*) _JNI_INC_SUBDIRS="win32";; +cygwin*) _JNI_INC_SUBDIRS="win32";; +*) _JNI_INC_SUBDIRS="genunix";; +esac + +if test "x$ac_cv_jni_header_path" != "xnone"; then + # add any subdirectories that are present + for JINCSUBDIR in $_JNI_INC_SUBDIRS + do + if test -d "$_JTOPDIR/include/$JINCSUBDIR"; then + JNI_INCLUDE_DIRS="$JNI_INCLUDE_DIRS $_JTOPDIR/include/$JINCSUBDIR" + fi + done +fi +]) + +# _ACJNI_FOLLOW_SYMLINKS +# Follows symbolic links on , +# finally setting variable _ACJNI_FOLLOWED +# ---------------------------------------- +AC_DEFUN([_ACJNI_FOLLOW_SYMLINKS],[ +# find the include directory relative to the javac executable +_cur="$1" +while ls -ld "$_cur" 2>/dev/null | grep " -> " >/dev/null; do + AC_MSG_CHECKING([symlink for $_cur]) + _slink=`ls -ld "$_cur" | sed 's/.* -> //'` + case "$_slink" in + /*) _cur="$_slink";; + # 'X' avoids triggering unwanted echo options. + *) _cur=`echo "X$_cur" | sed -e 's/^X//' -e 's:[[^/]]*$::'`"$_slink";; + esac + AC_MSG_RESULT([$_cur]) +done +_ACJNI_FOLLOWED="$_cur" +])# _ACJNI diff --git a/node_modules/secp256k1/src/secp256k1/build-aux/m4/ax_prog_cc_for_build.m4 b/node_modules/secp256k1/src/secp256k1/build-aux/m4/ax_prog_cc_for_build.m4 new file mode 100644 index 0000000..77fd346 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/build-aux/m4/ax_prog_cc_for_build.m4 @@ -0,0 +1,125 @@ +# =========================================================================== +# http://www.gnu.org/software/autoconf-archive/ax_prog_cc_for_build.html +# =========================================================================== +# +# SYNOPSIS +# +# AX_PROG_CC_FOR_BUILD +# +# DESCRIPTION +# +# This macro searches for a C compiler that generates native executables, +# that is a C compiler that surely is not a cross-compiler. This can be +# useful if you have to generate source code at compile-time like for +# example GCC does. +# +# The macro sets the CC_FOR_BUILD and CPP_FOR_BUILD macros to anything +# needed to compile or link (CC_FOR_BUILD) and preprocess (CPP_FOR_BUILD). +# The value of these variables can be overridden by the user by specifying +# a compiler with an environment variable (like you do for standard CC). +# +# It also sets BUILD_EXEEXT and BUILD_OBJEXT to the executable and object +# file extensions for the build platform, and GCC_FOR_BUILD to `yes' if +# the compiler we found is GCC. All these variables but GCC_FOR_BUILD are +# substituted in the Makefile. +# +# LICENSE +# +# Copyright (c) 2008 Paolo Bonzini +# +# Copying and distribution of this file, with or without modification, are +# permitted in any medium without royalty provided the copyright notice +# and this notice are preserved. This file is offered as-is, without any +# warranty. + +#serial 8 + +AU_ALIAS([AC_PROG_CC_FOR_BUILD], [AX_PROG_CC_FOR_BUILD]) +AC_DEFUN([AX_PROG_CC_FOR_BUILD], [dnl +AC_REQUIRE([AC_PROG_CC])dnl +AC_REQUIRE([AC_PROG_CPP])dnl +AC_REQUIRE([AC_EXEEXT])dnl +AC_REQUIRE([AC_CANONICAL_HOST])dnl + +dnl Use the standard macros, but make them use other variable names +dnl +pushdef([ac_cv_prog_CPP], ac_cv_build_prog_CPP)dnl +pushdef([ac_cv_prog_gcc], ac_cv_build_prog_gcc)dnl +pushdef([ac_cv_prog_cc_works], ac_cv_build_prog_cc_works)dnl +pushdef([ac_cv_prog_cc_cross], ac_cv_build_prog_cc_cross)dnl +pushdef([ac_cv_prog_cc_g], ac_cv_build_prog_cc_g)dnl +pushdef([ac_cv_exeext], ac_cv_build_exeext)dnl +pushdef([ac_cv_objext], ac_cv_build_objext)dnl +pushdef([ac_exeext], ac_build_exeext)dnl +pushdef([ac_objext], ac_build_objext)dnl +pushdef([CC], CC_FOR_BUILD)dnl +pushdef([CPP], CPP_FOR_BUILD)dnl +pushdef([CFLAGS], CFLAGS_FOR_BUILD)dnl +pushdef([CPPFLAGS], CPPFLAGS_FOR_BUILD)dnl +pushdef([LDFLAGS], LDFLAGS_FOR_BUILD)dnl +pushdef([host], build)dnl +pushdef([host_alias], build_alias)dnl +pushdef([host_cpu], build_cpu)dnl +pushdef([host_vendor], build_vendor)dnl +pushdef([host_os], build_os)dnl +pushdef([ac_cv_host], ac_cv_build)dnl +pushdef([ac_cv_host_alias], ac_cv_build_alias)dnl +pushdef([ac_cv_host_cpu], ac_cv_build_cpu)dnl +pushdef([ac_cv_host_vendor], ac_cv_build_vendor)dnl +pushdef([ac_cv_host_os], ac_cv_build_os)dnl +pushdef([ac_cpp], ac_build_cpp)dnl +pushdef([ac_compile], ac_build_compile)dnl +pushdef([ac_link], ac_build_link)dnl + +save_cross_compiling=$cross_compiling +save_ac_tool_prefix=$ac_tool_prefix +cross_compiling=no +ac_tool_prefix= + +AC_PROG_CC +AC_PROG_CPP +AC_EXEEXT + +ac_tool_prefix=$save_ac_tool_prefix +cross_compiling=$save_cross_compiling + +dnl Restore the old definitions +dnl +popdef([ac_link])dnl +popdef([ac_compile])dnl +popdef([ac_cpp])dnl +popdef([ac_cv_host_os])dnl +popdef([ac_cv_host_vendor])dnl +popdef([ac_cv_host_cpu])dnl +popdef([ac_cv_host_alias])dnl +popdef([ac_cv_host])dnl +popdef([host_os])dnl +popdef([host_vendor])dnl +popdef([host_cpu])dnl +popdef([host_alias])dnl +popdef([host])dnl +popdef([LDFLAGS])dnl +popdef([CPPFLAGS])dnl +popdef([CFLAGS])dnl +popdef([CPP])dnl +popdef([CC])dnl +popdef([ac_objext])dnl +popdef([ac_exeext])dnl +popdef([ac_cv_objext])dnl +popdef([ac_cv_exeext])dnl +popdef([ac_cv_prog_cc_g])dnl +popdef([ac_cv_prog_cc_cross])dnl +popdef([ac_cv_prog_cc_works])dnl +popdef([ac_cv_prog_gcc])dnl +popdef([ac_cv_prog_CPP])dnl + +dnl Finally, set Makefile variables +dnl +BUILD_EXEEXT=$ac_build_exeext +BUILD_OBJEXT=$ac_build_objext +AC_SUBST(BUILD_EXEEXT)dnl +AC_SUBST(BUILD_OBJEXT)dnl +AC_SUBST([CFLAGS_FOR_BUILD])dnl +AC_SUBST([CPPFLAGS_FOR_BUILD])dnl +AC_SUBST([LDFLAGS_FOR_BUILD])dnl +]) diff --git a/node_modules/secp256k1/src/secp256k1/build-aux/m4/bitcoin_secp.m4 b/node_modules/secp256k1/src/secp256k1/build-aux/m4/bitcoin_secp.m4 new file mode 100644 index 0000000..3b3975c --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/build-aux/m4/bitcoin_secp.m4 @@ -0,0 +1,68 @@ +dnl libsecp25k1 helper checks +AC_DEFUN([SECP_INT128_CHECK],[ +has_int128=$ac_cv_type___int128 +]) + +dnl escape "$0x" below using the m4 quadrigaph @S|@, and escape it again with a \ for the shell. +AC_DEFUN([SECP_64BIT_ASM_CHECK],[ +AC_MSG_CHECKING(for x86_64 assembly availability) +AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[ + #include ]],[[ + uint64_t a = 11, tmp; + __asm__ __volatile__("movq \@S|@0x100000000,%1; mulq %%rsi" : "+a"(a) : "S"(tmp) : "cc", "%rdx"); + ]])],[has_64bit_asm=yes],[has_64bit_asm=no]) +AC_MSG_RESULT([$has_64bit_asm]) +]) + +dnl +AC_DEFUN([SECP_OPENSSL_CHECK],[ + has_libcrypto=no + m4_ifdef([PKG_CHECK_MODULES],[ + PKG_CHECK_MODULES([CRYPTO], [libcrypto], [has_libcrypto=yes],[has_libcrypto=no]) + if test x"$has_libcrypto" = x"yes"; then + TEMP_LIBS="$LIBS" + LIBS="$LIBS $CRYPTO_LIBS" + AC_CHECK_LIB(crypto, main,[AC_DEFINE(HAVE_LIBCRYPTO,1,[Define this symbol if libcrypto is installed])],[has_libcrypto=no]) + LIBS="$TEMP_LIBS" + fi + ]) + if test x$has_libcrypto = xno; then + AC_CHECK_HEADER(openssl/crypto.h,[ + AC_CHECK_LIB(crypto, main,[ + has_libcrypto=yes + CRYPTO_LIBS=-lcrypto + AC_DEFINE(HAVE_LIBCRYPTO,1,[Define this symbol if libcrypto is installed]) + ]) + ]) + LIBS= + fi +if test x"$has_libcrypto" = x"yes" && test x"$has_openssl_ec" = x; then + AC_MSG_CHECKING(for EC functions in libcrypto) + AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[ + #include + #include + #include ]],[[ + EC_KEY *eckey = EC_KEY_new_by_curve_name(NID_secp256k1); + ECDSA_sign(0, NULL, 0, NULL, NULL, eckey); + ECDSA_verify(0, NULL, 0, NULL, 0, eckey); + EC_KEY_free(eckey); + ECDSA_SIG *sig_openssl; + sig_openssl = ECDSA_SIG_new(); + ECDSA_SIG_free(sig_openssl); + ]])],[has_openssl_ec=yes],[has_openssl_ec=no]) + AC_MSG_RESULT([$has_openssl_ec]) +fi +]) + +dnl +AC_DEFUN([SECP_GMP_CHECK],[ +if test x"$has_gmp" != x"yes"; then + CPPFLAGS_TEMP="$CPPFLAGS" + CPPFLAGS="$GMP_CPPFLAGS $CPPFLAGS" + LIBS_TEMP="$LIBS" + LIBS="$GMP_LIBS $LIBS" + AC_CHECK_HEADER(gmp.h,[AC_CHECK_LIB(gmp, __gmpz_init,[has_gmp=yes; GMP_LIBS="$GMP_LIBS -lgmp"; AC_DEFINE(HAVE_LIBGMP,1,[Define this symbol if libgmp is installed])])]) + CPPFLAGS="$CPPFLAGS_TEMP" + LIBS="$LIBS_TEMP" +fi +]) diff --git a/node_modules/secp256k1/src/secp256k1/configure.ac b/node_modules/secp256k1/src/secp256k1/configure.ac new file mode 100644 index 0000000..2a8db0a --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/configure.ac @@ -0,0 +1,591 @@ +AC_PREREQ([2.60]) +AC_INIT([libsecp256k1],[0.1]) +AC_CONFIG_AUX_DIR([build-aux]) +AC_CONFIG_MACRO_DIR([build-aux/m4]) +AC_CANONICAL_HOST +AH_TOP([#ifndef LIBSECP256K1_CONFIG_H]) +AH_TOP([#define LIBSECP256K1_CONFIG_H]) +AH_BOTTOM([#endif /*LIBSECP256K1_CONFIG_H*/]) +AM_INIT_AUTOMAKE([foreign subdir-objects]) +LT_INIT + +dnl make the compilation flags quiet unless V=1 is used +m4_ifdef([AM_SILENT_RULES], [AM_SILENT_RULES([yes])]) + +PKG_PROG_PKG_CONFIG + +AC_PATH_TOOL(AR, ar) +AC_PATH_TOOL(RANLIB, ranlib) +AC_PATH_TOOL(STRIP, strip) +AX_PROG_CC_FOR_BUILD + +if test "x$CFLAGS" = "x"; then + CFLAGS="-g" +fi + +AM_PROG_CC_C_O + +AC_PROG_CC_C89 +if test x"$ac_cv_prog_cc_c89" = x"no"; then + AC_MSG_ERROR([c89 compiler support required]) +fi +AM_PROG_AS + +case $host_os in + *darwin*) + if test x$cross_compiling != xyes; then + AC_PATH_PROG([BREW],brew,) + if test x$BREW != x; then + dnl These Homebrew packages may be keg-only, meaning that they won't be found + dnl in expected paths because they may conflict with system files. Ask + dnl Homebrew where each one is located, then adjust paths accordingly. + + openssl_prefix=`$BREW --prefix openssl 2>/dev/null` + gmp_prefix=`$BREW --prefix gmp 2>/dev/null` + if test x$openssl_prefix != x; then + PKG_CONFIG_PATH="$openssl_prefix/lib/pkgconfig:$PKG_CONFIG_PATH" + export PKG_CONFIG_PATH + fi + if test x$gmp_prefix != x; then + GMP_CPPFLAGS="-I$gmp_prefix/include" + GMP_LIBS="-L$gmp_prefix/lib" + fi + else + AC_PATH_PROG([PORT],port,) + dnl if homebrew isn't installed and macports is, add the macports default paths + dnl as a last resort. + if test x$PORT != x; then + CPPFLAGS="$CPPFLAGS -isystem /opt/local/include" + LDFLAGS="$LDFLAGS -L/opt/local/lib" + fi + fi + fi + ;; +esac + +CFLAGS="$CFLAGS -W" + +warn_CFLAGS="-std=c89 -pedantic -Wall -Wextra -Wcast-align -Wnested-externs -Wshadow -Wstrict-prototypes -Wno-unused-function -Wno-long-long -Wno-overlength-strings" +saved_CFLAGS="$CFLAGS" +CFLAGS="$CFLAGS $warn_CFLAGS" +AC_MSG_CHECKING([if ${CC} supports ${warn_CFLAGS}]) +AC_COMPILE_IFELSE([AC_LANG_SOURCE([[char foo;]])], + [ AC_MSG_RESULT([yes]) ], + [ AC_MSG_RESULT([no]) + CFLAGS="$saved_CFLAGS" + ]) + +saved_CFLAGS="$CFLAGS" +CFLAGS="$CFLAGS -fvisibility=hidden" +AC_MSG_CHECKING([if ${CC} supports -fvisibility=hidden]) +AC_COMPILE_IFELSE([AC_LANG_SOURCE([[char foo;]])], + [ AC_MSG_RESULT([yes]) ], + [ AC_MSG_RESULT([no]) + CFLAGS="$saved_CFLAGS" + ]) + +AC_ARG_ENABLE(benchmark, + AS_HELP_STRING([--enable-benchmark],[compile benchmark [default=yes]]), + [use_benchmark=$enableval], + [use_benchmark=yes]) + +AC_ARG_ENABLE(coverage, + AS_HELP_STRING([--enable-coverage],[enable compiler flags to support kcov coverage analysis [default=no]]), + [enable_coverage=$enableval], + [enable_coverage=no]) + +AC_ARG_ENABLE(tests, + AS_HELP_STRING([--enable-tests],[compile tests [default=yes]]), + [use_tests=$enableval], + [use_tests=yes]) + +AC_ARG_ENABLE(openssl_tests, + AS_HELP_STRING([--enable-openssl-tests],[enable OpenSSL tests [default=auto]]), + [enable_openssl_tests=$enableval], + [enable_openssl_tests=auto]) + +AC_ARG_ENABLE(experimental, + AS_HELP_STRING([--enable-experimental],[allow experimental configure options [default=no]]), + [use_experimental=$enableval], + [use_experimental=no]) + +AC_ARG_ENABLE(exhaustive_tests, + AS_HELP_STRING([--enable-exhaustive-tests],[compile exhaustive tests [default=yes]]), + [use_exhaustive_tests=$enableval], + [use_exhaustive_tests=yes]) + +AC_ARG_ENABLE(endomorphism, + AS_HELP_STRING([--enable-endomorphism],[enable endomorphism [default=no]]), + [use_endomorphism=$enableval], + [use_endomorphism=no]) + +AC_ARG_ENABLE(ecmult_static_precomputation, + AS_HELP_STRING([--enable-ecmult-static-precomputation],[enable precomputed ecmult table for signing [default=auto]]), + [use_ecmult_static_precomputation=$enableval], + [use_ecmult_static_precomputation=auto]) + +AC_ARG_ENABLE(module_ecdh, + AS_HELP_STRING([--enable-module-ecdh],[enable ECDH shared secret computation (experimental)]), + [enable_module_ecdh=$enableval], + [enable_module_ecdh=no]) + +AC_ARG_ENABLE(module_recovery, + AS_HELP_STRING([--enable-module-recovery],[enable ECDSA pubkey recovery module [default=no]]), + [enable_module_recovery=$enableval], + [enable_module_recovery=no]) + +AC_ARG_ENABLE(external_default_callbacks, + AS_HELP_STRING([--enable-external-default-callbacks],[enable external default callback functions [default=no]]), + [use_external_default_callbacks=$enableval], + [use_external_default_callbacks=no]) + +AC_ARG_ENABLE(jni, + AS_HELP_STRING([--enable-jni],[enable libsecp256k1_jni [default=no]]), + [use_jni=$enableval], + [use_jni=no]) + +AC_ARG_WITH([field], [AS_HELP_STRING([--with-field=64bit|32bit|auto], +[finite field implementation to use [default=auto]])],[req_field=$withval], [req_field=auto]) + +AC_ARG_WITH([bignum], [AS_HELP_STRING([--with-bignum=gmp|no|auto], +[bignum implementation to use [default=auto]])],[req_bignum=$withval], [req_bignum=auto]) + +AC_ARG_WITH([scalar], [AS_HELP_STRING([--with-scalar=64bit|32bit|auto], +[scalar implementation to use [default=auto]])],[req_scalar=$withval], [req_scalar=auto]) + +AC_ARG_WITH([asm], [AS_HELP_STRING([--with-asm=x86_64|arm|no|auto], +[assembly optimizations to use (experimental: arm) [default=auto]])],[req_asm=$withval], [req_asm=auto]) + +AC_ARG_WITH([ecmult-window], [AS_HELP_STRING([--with-ecmult-window=SIZE|auto], +[window size for ecmult precomputation for verification, specified as integer in range [2..24].] +[Larger values result in possibly better performance at the cost of an exponentially larger precomputed table.] +[The table will store 2^(SIZE-2) * 64 bytes of data but can be larger in memory due to platform-specific padding and alignment.] +[If the endomorphism optimization is enabled, two tables of this size are used instead of only one.] +["auto" is a reasonable setting for desktop machines (currently 15). [default=auto]] +)], +[req_ecmult_window=$withval], [req_ecmult_window=auto]) + +AC_ARG_WITH([ecmult-gen-precision], [AS_HELP_STRING([--with-ecmult-gen-precision=2|4|8|auto], +[Precision bits to tune the precomputed table size for signing.] +[The size of the table is 32kB for 2 bits, 64kB for 4 bits, 512kB for 8 bits of precision.] +[A larger table size usually results in possible faster signing.] +["auto" is a reasonable setting for desktop machines (currently 4). [default=auto]] +)], +[req_ecmult_gen_precision=$withval], [req_ecmult_gen_precision=auto]) + +AC_CHECK_TYPES([__int128]) + +if test x"$enable_coverage" = x"yes"; then + AC_DEFINE(COVERAGE, 1, [Define this symbol to compile out all VERIFY code]) + CFLAGS="$CFLAGS -O0 --coverage" + LDFLAGS="$LDFLAGS --coverage" +else + CFLAGS="$CFLAGS -O3" +fi + +if test x"$use_ecmult_static_precomputation" != x"no"; then + # Temporarily switch to an environment for the native compiler + save_cross_compiling=$cross_compiling + cross_compiling=no + SAVE_CC="$CC" + CC="$CC_FOR_BUILD" + SAVE_CFLAGS="$CFLAGS" + CFLAGS="$CFLAGS_FOR_BUILD" + SAVE_CPPFLAGS="$CPPFLAGS" + CPPFLAGS="$CPPFLAGS_FOR_BUILD" + SAVE_LDFLAGS="$LDFLAGS" + LDFLAGS="$LDFLAGS_FOR_BUILD" + + warn_CFLAGS_FOR_BUILD="-Wall -Wextra -Wno-unused-function" + saved_CFLAGS="$CFLAGS" + CFLAGS="$CFLAGS $warn_CFLAGS_FOR_BUILD" + AC_MSG_CHECKING([if native ${CC_FOR_BUILD} supports ${warn_CFLAGS_FOR_BUILD}]) + AC_COMPILE_IFELSE([AC_LANG_SOURCE([[char foo;]])], + [ AC_MSG_RESULT([yes]) ], + [ AC_MSG_RESULT([no]) + CFLAGS="$saved_CFLAGS" + ]) + + AC_MSG_CHECKING([for working native compiler: ${CC_FOR_BUILD}]) + AC_RUN_IFELSE( + [AC_LANG_PROGRAM([], [])], + [working_native_cc=yes], + [working_native_cc=no],[dnl]) + + CFLAGS_FOR_BUILD="$CFLAGS" + + # Restore the environment + cross_compiling=$save_cross_compiling + CC="$SAVE_CC" + CFLAGS="$SAVE_CFLAGS" + CPPFLAGS="$SAVE_CPPFLAGS" + LDFLAGS="$SAVE_LDFLAGS" + + if test x"$working_native_cc" = x"no"; then + AC_MSG_RESULT([no]) + set_precomp=no + m4_define([please_set_for_build], [Please set CC_FOR_BUILD, CFLAGS_FOR_BUILD, CPPFLAGS_FOR_BUILD, and/or LDFLAGS_FOR_BUILD.]) + if test x"$use_ecmult_static_precomputation" = x"yes"; then + AC_MSG_ERROR([native compiler ${CC_FOR_BUILD} does not produce working binaries. please_set_for_build]) + else + AC_MSG_WARN([Disabling statically generated ecmult table because the native compiler ${CC_FOR_BUILD} does not produce working binaries. please_set_for_build]) + fi + else + AC_MSG_RESULT([yes]) + set_precomp=yes + fi +else + set_precomp=no +fi + +if test x"$req_asm" = x"auto"; then + SECP_64BIT_ASM_CHECK + if test x"$has_64bit_asm" = x"yes"; then + set_asm=x86_64 + fi + if test x"$set_asm" = x; then + set_asm=no + fi +else + set_asm=$req_asm + case $set_asm in + x86_64) + SECP_64BIT_ASM_CHECK + if test x"$has_64bit_asm" != x"yes"; then + AC_MSG_ERROR([x86_64 assembly optimization requested but not available]) + fi + ;; + arm) + ;; + no) + ;; + *) + AC_MSG_ERROR([invalid assembly optimization selection]) + ;; + esac +fi + +if test x"$req_field" = x"auto"; then + if test x"set_asm" = x"x86_64"; then + set_field=64bit + fi + if test x"$set_field" = x; then + SECP_INT128_CHECK + if test x"$has_int128" = x"yes"; then + set_field=64bit + fi + fi + if test x"$set_field" = x; then + set_field=32bit + fi +else + set_field=$req_field + case $set_field in + 64bit) + if test x"$set_asm" != x"x86_64"; then + SECP_INT128_CHECK + if test x"$has_int128" != x"yes"; then + AC_MSG_ERROR([64bit field explicitly requested but neither __int128 support or x86_64 assembly available]) + fi + fi + ;; + 32bit) + ;; + *) + AC_MSG_ERROR([invalid field implementation selection]) + ;; + esac +fi + +if test x"$req_scalar" = x"auto"; then + SECP_INT128_CHECK + if test x"$has_int128" = x"yes"; then + set_scalar=64bit + fi + if test x"$set_scalar" = x; then + set_scalar=32bit + fi +else + set_scalar=$req_scalar + case $set_scalar in + 64bit) + SECP_INT128_CHECK + if test x"$has_int128" != x"yes"; then + AC_MSG_ERROR([64bit scalar explicitly requested but __int128 support not available]) + fi + ;; + 32bit) + ;; + *) + AC_MSG_ERROR([invalid scalar implementation selected]) + ;; + esac +fi + +if test x"$req_bignum" = x"auto"; then + SECP_GMP_CHECK + if test x"$has_gmp" = x"yes"; then + set_bignum=gmp + fi + + if test x"$set_bignum" = x; then + set_bignum=no + fi +else + set_bignum=$req_bignum + case $set_bignum in + gmp) + SECP_GMP_CHECK + if test x"$has_gmp" != x"yes"; then + AC_MSG_ERROR([gmp bignum explicitly requested but libgmp not available]) + fi + ;; + no) + ;; + *) + AC_MSG_ERROR([invalid bignum implementation selection]) + ;; + esac +fi + +# select assembly optimization +use_external_asm=no + +case $set_asm in +x86_64) + AC_DEFINE(USE_ASM_X86_64, 1, [Define this symbol to enable x86_64 assembly optimizations]) + ;; +arm) + use_external_asm=yes + ;; +no) + ;; +*) + AC_MSG_ERROR([invalid assembly optimizations]) + ;; +esac + +# select field implementation +case $set_field in +64bit) + AC_DEFINE(USE_FIELD_5X52, 1, [Define this symbol to use the FIELD_5X52 implementation]) + ;; +32bit) + AC_DEFINE(USE_FIELD_10X26, 1, [Define this symbol to use the FIELD_10X26 implementation]) + ;; +*) + AC_MSG_ERROR([invalid field implementation]) + ;; +esac + +# select bignum implementation +case $set_bignum in +gmp) + AC_DEFINE(HAVE_LIBGMP, 1, [Define this symbol if libgmp is installed]) + AC_DEFINE(USE_NUM_GMP, 1, [Define this symbol to use the gmp implementation for num]) + AC_DEFINE(USE_FIELD_INV_NUM, 1, [Define this symbol to use the num-based field inverse implementation]) + AC_DEFINE(USE_SCALAR_INV_NUM, 1, [Define this symbol to use the num-based scalar inverse implementation]) + ;; +no) + AC_DEFINE(USE_NUM_NONE, 1, [Define this symbol to use no num implementation]) + AC_DEFINE(USE_FIELD_INV_BUILTIN, 1, [Define this symbol to use the native field inverse implementation]) + AC_DEFINE(USE_SCALAR_INV_BUILTIN, 1, [Define this symbol to use the native scalar inverse implementation]) + ;; +*) + AC_MSG_ERROR([invalid bignum implementation]) + ;; +esac + +#select scalar implementation +case $set_scalar in +64bit) + AC_DEFINE(USE_SCALAR_4X64, 1, [Define this symbol to use the 4x64 scalar implementation]) + ;; +32bit) + AC_DEFINE(USE_SCALAR_8X32, 1, [Define this symbol to use the 8x32 scalar implementation]) + ;; +*) + AC_MSG_ERROR([invalid scalar implementation]) + ;; +esac + +#set ecmult window size +if test x"$req_ecmult_window" = x"auto"; then + set_ecmult_window=15 +else + set_ecmult_window=$req_ecmult_window +fi + +error_window_size=['window size for ecmult precomputation not an integer in range [2..24] or "auto"'] +case $set_ecmult_window in +''|*[[!0-9]]*) + # no valid integer + AC_MSG_ERROR($error_window_size) + ;; +*) + if test "$set_ecmult_window" -lt 2 -o "$set_ecmult_window" -gt 24 ; then + # not in range + AC_MSG_ERROR($error_window_size) + fi + AC_DEFINE_UNQUOTED(ECMULT_WINDOW_SIZE, $set_ecmult_window, [Set window size for ecmult precomputation]) + ;; +esac + +#set ecmult gen precision +if test x"$req_ecmult_gen_precision" = x"auto"; then + set_ecmult_gen_precision=4 +else + set_ecmult_gen_precision=$req_ecmult_gen_precision +fi + +case $set_ecmult_gen_precision in +2|4|8) + AC_DEFINE_UNQUOTED(ECMULT_GEN_PREC_BITS, $set_ecmult_gen_precision, [Set ecmult gen precision bits]) + ;; +*) + AC_MSG_ERROR(['ecmult gen precision not 2, 4, 8 or "auto"']) + ;; +esac + +if test x"$use_tests" = x"yes"; then + SECP_OPENSSL_CHECK + if test x"$has_openssl_ec" = x"yes"; then + if test x"$enable_openssl_tests" != x"no"; then + AC_DEFINE(ENABLE_OPENSSL_TESTS, 1, [Define this symbol if OpenSSL EC functions are available]) + SECP_TEST_INCLUDES="$SSL_CFLAGS $CRYPTO_CFLAGS" + SECP_TEST_LIBS="$CRYPTO_LIBS" + + case $host in + *mingw*) + SECP_TEST_LIBS="$SECP_TEST_LIBS -lgdi32" + ;; + esac + fi + else + if test x"$enable_openssl_tests" = x"yes"; then + AC_MSG_ERROR([OpenSSL tests requested but OpenSSL with EC support is not available]) + fi + fi +else + if test x"$enable_openssl_tests" = x"yes"; then + AC_MSG_ERROR([OpenSSL tests requested but tests are not enabled]) + fi +fi + +if test x"$use_jni" != x"no"; then + AX_JNI_INCLUDE_DIR + have_jni_dependencies=yes + if test x"$enable_module_ecdh" = x"no"; then + have_jni_dependencies=no + fi + if test "x$JNI_INCLUDE_DIRS" = "x"; then + have_jni_dependencies=no + fi + if test "x$have_jni_dependencies" = "xno"; then + if test x"$use_jni" = x"yes"; then + AC_MSG_ERROR([jni support explicitly requested but headers/dependencies were not found. Enable ECDH and try again.]) + fi + AC_MSG_WARN([jni headers/dependencies not found. jni support disabled]) + use_jni=no + else + use_jni=yes + for JNI_INCLUDE_DIR in $JNI_INCLUDE_DIRS; do + JNI_INCLUDES="$JNI_INCLUDES -I$JNI_INCLUDE_DIR" + done + fi +fi + +if test x"$set_bignum" = x"gmp"; then + SECP_LIBS="$SECP_LIBS $GMP_LIBS" + SECP_INCLUDES="$SECP_INCLUDES $GMP_CPPFLAGS" +fi + +if test x"$use_endomorphism" = x"yes"; then + AC_DEFINE(USE_ENDOMORPHISM, 1, [Define this symbol to use endomorphism optimization]) +fi + +if test x"$set_precomp" = x"yes"; then + AC_DEFINE(USE_ECMULT_STATIC_PRECOMPUTATION, 1, [Define this symbol to use a statically generated ecmult table]) +fi + +if test x"$enable_module_ecdh" = x"yes"; then + AC_DEFINE(ENABLE_MODULE_ECDH, 1, [Define this symbol to enable the ECDH module]) +fi + +if test x"$enable_module_recovery" = x"yes"; then + AC_DEFINE(ENABLE_MODULE_RECOVERY, 1, [Define this symbol to enable the ECDSA pubkey recovery module]) +fi + +AC_C_BIGENDIAN() + +if test x"$use_external_asm" = x"yes"; then + AC_DEFINE(USE_EXTERNAL_ASM, 1, [Define this symbol if an external (non-inline) assembly implementation is used]) +fi + +if test x"$use_external_default_callbacks" = x"yes"; then + AC_DEFINE(USE_EXTERNAL_DEFAULT_CALLBACKS, 1, [Define this symbol if an external implementation of the default callbacks is used]) +fi + +if test x"$enable_experimental" = x"yes"; then + AC_MSG_NOTICE([******]) + AC_MSG_NOTICE([WARNING: experimental build]) + AC_MSG_NOTICE([Experimental features do not have stable APIs or properties, and may not be safe for production use.]) + AC_MSG_NOTICE([Building ECDH module: $enable_module_ecdh]) + AC_MSG_NOTICE([******]) +else + if test x"$enable_module_ecdh" = x"yes"; then + AC_MSG_ERROR([ECDH module is experimental. Use --enable-experimental to allow.]) + fi + if test x"$set_asm" = x"arm"; then + AC_MSG_ERROR([ARM assembly optimization is experimental. Use --enable-experimental to allow.]) + fi +fi + +AC_CONFIG_HEADERS([src/libsecp256k1-config.h]) +AC_CONFIG_FILES([Makefile libsecp256k1.pc]) +AC_SUBST(JNI_INCLUDES) +AC_SUBST(SECP_INCLUDES) +AC_SUBST(SECP_LIBS) +AC_SUBST(SECP_TEST_LIBS) +AC_SUBST(SECP_TEST_INCLUDES) +AM_CONDITIONAL([ENABLE_COVERAGE], [test x"$enable_coverage" = x"yes"]) +AM_CONDITIONAL([USE_TESTS], [test x"$use_tests" != x"no"]) +AM_CONDITIONAL([USE_EXHAUSTIVE_TESTS], [test x"$use_exhaustive_tests" != x"no"]) +AM_CONDITIONAL([USE_BENCHMARK], [test x"$use_benchmark" = x"yes"]) +AM_CONDITIONAL([USE_ECMULT_STATIC_PRECOMPUTATION], [test x"$set_precomp" = x"yes"]) +AM_CONDITIONAL([ENABLE_MODULE_ECDH], [test x"$enable_module_ecdh" = x"yes"]) +AM_CONDITIONAL([ENABLE_MODULE_RECOVERY], [test x"$enable_module_recovery" = x"yes"]) +AM_CONDITIONAL([USE_JNI], [test x"$use_jni" = x"yes"]) +AM_CONDITIONAL([USE_EXTERNAL_ASM], [test x"$use_external_asm" = x"yes"]) +AM_CONDITIONAL([USE_ASM_ARM], [test x"$set_asm" = x"arm"]) + +dnl make sure nothing new is exported so that we don't break the cache +PKGCONFIG_PATH_TEMP="$PKG_CONFIG_PATH" +unset PKG_CONFIG_PATH +PKG_CONFIG_PATH="$PKGCONFIG_PATH_TEMP" + +AC_OUTPUT + +echo +echo "Build Options:" +echo " with endomorphism = $use_endomorphism" +echo " with ecmult precomp = $set_precomp" +echo " with external callbacks = $use_external_default_callbacks" +echo " with jni = $use_jni" +echo " with benchmarks = $use_benchmark" +echo " with coverage = $enable_coverage" +echo " module ecdh = $enable_module_ecdh" +echo " module recovery = $enable_module_recovery" +echo +echo " asm = $set_asm" +echo " bignum = $set_bignum" +echo " field = $set_field" +echo " scalar = $set_scalar" +echo " ecmult window size = $set_ecmult_window" +echo " ecmult gen prec. bits = $set_ecmult_gen_precision" +echo +echo " CC = $CC" +echo " CFLAGS = $CFLAGS" +echo " CPPFLAGS = $CPPFLAGS" +echo " LDFLAGS = $LDFLAGS" +echo diff --git a/node_modules/secp256k1/src/secp256k1/contrib/lax_der_parsing.c b/node_modules/secp256k1/src/secp256k1/contrib/lax_der_parsing.c new file mode 100644 index 0000000..e177a05 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/contrib/lax_der_parsing.c @@ -0,0 +1,150 @@ +/********************************************************************** + * Copyright (c) 2015 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#include +#include + +#include "lax_der_parsing.h" + +int ecdsa_signature_parse_der_lax(const secp256k1_context* ctx, secp256k1_ecdsa_signature* sig, const unsigned char *input, size_t inputlen) { + size_t rpos, rlen, spos, slen; + size_t pos = 0; + size_t lenbyte; + unsigned char tmpsig[64] = {0}; + int overflow = 0; + + /* Hack to initialize sig with a correctly-parsed but invalid signature. */ + secp256k1_ecdsa_signature_parse_compact(ctx, sig, tmpsig); + + /* Sequence tag byte */ + if (pos == inputlen || input[pos] != 0x30) { + return 0; + } + pos++; + + /* Sequence length bytes */ + if (pos == inputlen) { + return 0; + } + lenbyte = input[pos++]; + if (lenbyte & 0x80) { + lenbyte -= 0x80; + if (lenbyte > inputlen - pos) { + return 0; + } + pos += lenbyte; + } + + /* Integer tag byte for R */ + if (pos == inputlen || input[pos] != 0x02) { + return 0; + } + pos++; + + /* Integer length for R */ + if (pos == inputlen) { + return 0; + } + lenbyte = input[pos++]; + if (lenbyte & 0x80) { + lenbyte -= 0x80; + if (lenbyte > inputlen - pos) { + return 0; + } + while (lenbyte > 0 && input[pos] == 0) { + pos++; + lenbyte--; + } + if (lenbyte >= sizeof(size_t)) { + return 0; + } + rlen = 0; + while (lenbyte > 0) { + rlen = (rlen << 8) + input[pos]; + pos++; + lenbyte--; + } + } else { + rlen = lenbyte; + } + if (rlen > inputlen - pos) { + return 0; + } + rpos = pos; + pos += rlen; + + /* Integer tag byte for S */ + if (pos == inputlen || input[pos] != 0x02) { + return 0; + } + pos++; + + /* Integer length for S */ + if (pos == inputlen) { + return 0; + } + lenbyte = input[pos++]; + if (lenbyte & 0x80) { + lenbyte -= 0x80; + if (lenbyte > inputlen - pos) { + return 0; + } + while (lenbyte > 0 && input[pos] == 0) { + pos++; + lenbyte--; + } + if (lenbyte >= sizeof(size_t)) { + return 0; + } + slen = 0; + while (lenbyte > 0) { + slen = (slen << 8) + input[pos]; + pos++; + lenbyte--; + } + } else { + slen = lenbyte; + } + if (slen > inputlen - pos) { + return 0; + } + spos = pos; + pos += slen; + + /* Ignore leading zeroes in R */ + while (rlen > 0 && input[rpos] == 0) { + rlen--; + rpos++; + } + /* Copy R value */ + if (rlen > 32) { + overflow = 1; + } else { + memcpy(tmpsig + 32 - rlen, input + rpos, rlen); + } + + /* Ignore leading zeroes in S */ + while (slen > 0 && input[spos] == 0) { + slen--; + spos++; + } + /* Copy S value */ + if (slen > 32) { + overflow = 1; + } else { + memcpy(tmpsig + 64 - slen, input + spos, slen); + } + + if (!overflow) { + overflow = !secp256k1_ecdsa_signature_parse_compact(ctx, sig, tmpsig); + } + if (overflow) { + memset(tmpsig, 0, 64); + secp256k1_ecdsa_signature_parse_compact(ctx, sig, tmpsig); + } + return 1; +} + diff --git a/node_modules/secp256k1/src/secp256k1/contrib/lax_der_parsing.h b/node_modules/secp256k1/src/secp256k1/contrib/lax_der_parsing.h new file mode 100644 index 0000000..7eaf63b --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/contrib/lax_der_parsing.h @@ -0,0 +1,91 @@ +/********************************************************************** + * Copyright (c) 2015 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +/**** + * Please do not link this file directly. It is not part of the libsecp256k1 + * project and does not promise any stability in its API, functionality or + * presence. Projects which use this code should instead copy this header + * and its accompanying .c file directly into their codebase. + ****/ + +/* This file defines a function that parses DER with various errors and + * violations. This is not a part of the library itself, because the allowed + * violations are chosen arbitrarily and do not follow or establish any + * standard. + * + * In many places it matters that different implementations do not only accept + * the same set of valid signatures, but also reject the same set of signatures. + * The only means to accomplish that is by strictly obeying a standard, and not + * accepting anything else. + * + * Nonetheless, sometimes there is a need for compatibility with systems that + * use signatures which do not strictly obey DER. The snippet below shows how + * certain violations are easily supported. You may need to adapt it. + * + * Do not use this for new systems. Use well-defined DER or compact signatures + * instead if you have the choice (see secp256k1_ecdsa_signature_parse_der and + * secp256k1_ecdsa_signature_parse_compact). + * + * The supported violations are: + * - All numbers are parsed as nonnegative integers, even though X.609-0207 + * section 8.3.3 specifies that integers are always encoded as two's + * complement. + * - Integers can have length 0, even though section 8.3.1 says they can't. + * - Integers with overly long padding are accepted, violation section + * 8.3.2. + * - 127-byte long length descriptors are accepted, even though section + * 8.1.3.5.c says that they are not. + * - Trailing garbage data inside or after the signature is ignored. + * - The length descriptor of the sequence is ignored. + * + * Compared to for example OpenSSL, many violations are NOT supported: + * - Using overly long tag descriptors for the sequence or integers inside, + * violating section 8.1.2.2. + * - Encoding primitive integers as constructed values, violating section + * 8.3.1. + */ + +#ifndef SECP256K1_CONTRIB_LAX_DER_PARSING_H +#define SECP256K1_CONTRIB_LAX_DER_PARSING_H + +#include + +#ifdef __cplusplus +extern "C" { +#endif + +/** Parse a signature in "lax DER" format + * + * Returns: 1 when the signature could be parsed, 0 otherwise. + * Args: ctx: a secp256k1 context object + * Out: sig: a pointer to a signature object + * In: input: a pointer to the signature to be parsed + * inputlen: the length of the array pointed to be input + * + * This function will accept any valid DER encoded signature, even if the + * encoded numbers are out of range. In addition, it will accept signatures + * which violate the DER spec in various ways. Its purpose is to allow + * validation of the Bitcoin blockchain, which includes non-DER signatures + * from before the network rules were updated to enforce DER. Note that + * the set of supported violations is a strict subset of what OpenSSL will + * accept. + * + * After the call, sig will always be initialized. If parsing failed or the + * encoded numbers are out of range, signature validation with it is + * guaranteed to fail for every message and public key. + */ +int ecdsa_signature_parse_der_lax( + const secp256k1_context* ctx, + secp256k1_ecdsa_signature* sig, + const unsigned char *input, + size_t inputlen +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +#ifdef __cplusplus +} +#endif + +#endif /* SECP256K1_CONTRIB_LAX_DER_PARSING_H */ diff --git a/node_modules/secp256k1/src/secp256k1/contrib/lax_der_privatekey_parsing.c b/node_modules/secp256k1/src/secp256k1/contrib/lax_der_privatekey_parsing.c new file mode 100644 index 0000000..c2e63b4 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/contrib/lax_der_privatekey_parsing.c @@ -0,0 +1,113 @@ +/********************************************************************** + * Copyright (c) 2014, 2015 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#include +#include + +#include "lax_der_privatekey_parsing.h" + +int ec_privkey_import_der(const secp256k1_context* ctx, unsigned char *out32, const unsigned char *privkey, size_t privkeylen) { + const unsigned char *end = privkey + privkeylen; + int lenb = 0; + int len = 0; + memset(out32, 0, 32); + /* sequence header */ + if (end < privkey+1 || *privkey != 0x30) { + return 0; + } + privkey++; + /* sequence length constructor */ + if (end < privkey+1 || !(*privkey & 0x80)) { + return 0; + } + lenb = *privkey & ~0x80; privkey++; + if (lenb < 1 || lenb > 2) { + return 0; + } + if (end < privkey+lenb) { + return 0; + } + /* sequence length */ + len = privkey[lenb-1] | (lenb > 1 ? privkey[lenb-2] << 8 : 0); + privkey += lenb; + if (end < privkey+len) { + return 0; + } + /* sequence element 0: version number (=1) */ + if (end < privkey+3 || privkey[0] != 0x02 || privkey[1] != 0x01 || privkey[2] != 0x01) { + return 0; + } + privkey += 3; + /* sequence element 1: octet string, up to 32 bytes */ + if (end < privkey+2 || privkey[0] != 0x04 || privkey[1] > 0x20 || end < privkey+2+privkey[1]) { + return 0; + } + memcpy(out32 + 32 - privkey[1], privkey + 2, privkey[1]); + if (!secp256k1_ec_seckey_verify(ctx, out32)) { + memset(out32, 0, 32); + return 0; + } + return 1; +} + +int ec_privkey_export_der(const secp256k1_context *ctx, unsigned char *privkey, size_t *privkeylen, const unsigned char *key32, int compressed) { + secp256k1_pubkey pubkey; + size_t pubkeylen = 0; + if (!secp256k1_ec_pubkey_create(ctx, &pubkey, key32)) { + *privkeylen = 0; + return 0; + } + if (compressed) { + static const unsigned char begin[] = { + 0x30,0x81,0xD3,0x02,0x01,0x01,0x04,0x20 + }; + static const unsigned char middle[] = { + 0xA0,0x81,0x85,0x30,0x81,0x82,0x02,0x01,0x01,0x30,0x2C,0x06,0x07,0x2A,0x86,0x48, + 0xCE,0x3D,0x01,0x01,0x02,0x21,0x00,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFE,0xFF,0xFF,0xFC,0x2F,0x30,0x06,0x04,0x01,0x00,0x04,0x01,0x07,0x04, + 0x21,0x02,0x79,0xBE,0x66,0x7E,0xF9,0xDC,0xBB,0xAC,0x55,0xA0,0x62,0x95,0xCE,0x87, + 0x0B,0x07,0x02,0x9B,0xFC,0xDB,0x2D,0xCE,0x28,0xD9,0x59,0xF2,0x81,0x5B,0x16,0xF8, + 0x17,0x98,0x02,0x21,0x00,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFF,0xFE,0xBA,0xAE,0xDC,0xE6,0xAF,0x48,0xA0,0x3B,0xBF,0xD2,0x5E, + 0x8C,0xD0,0x36,0x41,0x41,0x02,0x01,0x01,0xA1,0x24,0x03,0x22,0x00 + }; + unsigned char *ptr = privkey; + memcpy(ptr, begin, sizeof(begin)); ptr += sizeof(begin); + memcpy(ptr, key32, 32); ptr += 32; + memcpy(ptr, middle, sizeof(middle)); ptr += sizeof(middle); + pubkeylen = 33; + secp256k1_ec_pubkey_serialize(ctx, ptr, &pubkeylen, &pubkey, SECP256K1_EC_COMPRESSED); + ptr += pubkeylen; + *privkeylen = ptr - privkey; + } else { + static const unsigned char begin[] = { + 0x30,0x82,0x01,0x13,0x02,0x01,0x01,0x04,0x20 + }; + static const unsigned char middle[] = { + 0xA0,0x81,0xA5,0x30,0x81,0xA2,0x02,0x01,0x01,0x30,0x2C,0x06,0x07,0x2A,0x86,0x48, + 0xCE,0x3D,0x01,0x01,0x02,0x21,0x00,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFE,0xFF,0xFF,0xFC,0x2F,0x30,0x06,0x04,0x01,0x00,0x04,0x01,0x07,0x04, + 0x41,0x04,0x79,0xBE,0x66,0x7E,0xF9,0xDC,0xBB,0xAC,0x55,0xA0,0x62,0x95,0xCE,0x87, + 0x0B,0x07,0x02,0x9B,0xFC,0xDB,0x2D,0xCE,0x28,0xD9,0x59,0xF2,0x81,0x5B,0x16,0xF8, + 0x17,0x98,0x48,0x3A,0xDA,0x77,0x26,0xA3,0xC4,0x65,0x5D,0xA4,0xFB,0xFC,0x0E,0x11, + 0x08,0xA8,0xFD,0x17,0xB4,0x48,0xA6,0x85,0x54,0x19,0x9C,0x47,0xD0,0x8F,0xFB,0x10, + 0xD4,0xB8,0x02,0x21,0x00,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFF,0xFE,0xBA,0xAE,0xDC,0xE6,0xAF,0x48,0xA0,0x3B,0xBF,0xD2,0x5E, + 0x8C,0xD0,0x36,0x41,0x41,0x02,0x01,0x01,0xA1,0x44,0x03,0x42,0x00 + }; + unsigned char *ptr = privkey; + memcpy(ptr, begin, sizeof(begin)); ptr += sizeof(begin); + memcpy(ptr, key32, 32); ptr += 32; + memcpy(ptr, middle, sizeof(middle)); ptr += sizeof(middle); + pubkeylen = 65; + secp256k1_ec_pubkey_serialize(ctx, ptr, &pubkeylen, &pubkey, SECP256K1_EC_UNCOMPRESSED); + ptr += pubkeylen; + *privkeylen = ptr - privkey; + } + return 1; +} diff --git a/node_modules/secp256k1/src/secp256k1/contrib/lax_der_privatekey_parsing.h b/node_modules/secp256k1/src/secp256k1/contrib/lax_der_privatekey_parsing.h new file mode 100644 index 0000000..fece261 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/contrib/lax_der_privatekey_parsing.h @@ -0,0 +1,90 @@ +/********************************************************************** + * Copyright (c) 2014, 2015 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +/**** + * Please do not link this file directly. It is not part of the libsecp256k1 + * project and does not promise any stability in its API, functionality or + * presence. Projects which use this code should instead copy this header + * and its accompanying .c file directly into their codebase. + ****/ + +/* This file contains code snippets that parse DER private keys with + * various errors and violations. This is not a part of the library + * itself, because the allowed violations are chosen arbitrarily and + * do not follow or establish any standard. + * + * It also contains code to serialize private keys in a compatible + * manner. + * + * These functions are meant for compatibility with applications + * that require BER encoded keys. When working with secp256k1-specific + * code, the simple 32-byte private keys normally used by the + * library are sufficient. + */ + +#ifndef SECP256K1_CONTRIB_BER_PRIVATEKEY_H +#define SECP256K1_CONTRIB_BER_PRIVATEKEY_H + +#include + +#ifdef __cplusplus +extern "C" { +#endif + +/** Export a private key in DER format. + * + * Returns: 1 if the private key was valid. + * Args: ctx: pointer to a context object, initialized for signing (cannot + * be NULL) + * Out: privkey: pointer to an array for storing the private key in BER. + * Should have space for 279 bytes, and cannot be NULL. + * privkeylen: Pointer to an int where the length of the private key in + * privkey will be stored. + * In: seckey: pointer to a 32-byte secret key to export. + * compressed: 1 if the key should be exported in + * compressed format, 0 otherwise + * + * This function is purely meant for compatibility with applications that + * require BER encoded keys. When working with secp256k1-specific code, the + * simple 32-byte private keys are sufficient. + * + * Note that this function does not guarantee correct DER output. It is + * guaranteed to be parsable by secp256k1_ec_privkey_import_der + */ +SECP256K1_WARN_UNUSED_RESULT int ec_privkey_export_der( + const secp256k1_context* ctx, + unsigned char *privkey, + size_t *privkeylen, + const unsigned char *seckey, + int compressed +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3) SECP256K1_ARG_NONNULL(4); + +/** Import a private key in DER format. + * Returns: 1 if a private key was extracted. + * Args: ctx: pointer to a context object (cannot be NULL). + * Out: seckey: pointer to a 32-byte array for storing the private key. + * (cannot be NULL). + * In: privkey: pointer to a private key in DER format (cannot be NULL). + * privkeylen: length of the DER private key pointed to be privkey. + * + * This function will accept more than just strict DER, and even allow some BER + * violations. The public key stored inside the DER-encoded private key is not + * verified for correctness, nor are the curve parameters. Use this function + * only if you know in advance it is supposed to contain a secp256k1 private + * key. + */ +SECP256K1_WARN_UNUSED_RESULT int ec_privkey_import_der( + const secp256k1_context* ctx, + unsigned char *seckey, + const unsigned char *privkey, + size_t privkeylen +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +#ifdef __cplusplus +} +#endif + +#endif /* SECP256K1_CONTRIB_BER_PRIVATEKEY_H */ diff --git a/node_modules/secp256k1/src/secp256k1/include/secp256k1.h b/node_modules/secp256k1/src/secp256k1/include/secp256k1.h new file mode 100644 index 0000000..36020e5 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/include/secp256k1.h @@ -0,0 +1,708 @@ +#ifndef SECP256K1_H +#define SECP256K1_H + +#ifdef __cplusplus +extern "C" { +#endif + +#include + +/* These rules specify the order of arguments in API calls: + * + * 1. Context pointers go first, followed by output arguments, combined + * output/input arguments, and finally input-only arguments. + * 2. Array lengths always immediately the follow the argument whose length + * they describe, even if this violates rule 1. + * 3. Within the OUT/OUTIN/IN groups, pointers to data that is typically generated + * later go first. This means: signatures, public nonces, private nonces, + * messages, public keys, secret keys, tweaks. + * 4. Arguments that are not data pointers go last, from more complex to less + * complex: function pointers, algorithm names, messages, void pointers, + * counts, flags, booleans. + * 5. Opaque data pointers follow the function pointer they are to be passed to. + */ + +/** Opaque data structure that holds context information (precomputed tables etc.). + * + * The purpose of context structures is to cache large precomputed data tables + * that are expensive to construct, and also to maintain the randomization data + * for blinding. + * + * Do not create a new context object for each operation, as construction is + * far slower than all other API calls (~100 times slower than an ECDSA + * verification). + * + * A constructed context can safely be used from multiple threads + * simultaneously, but API calls that take a non-const pointer to a context + * need exclusive access to it. In particular this is the case for + * secp256k1_context_destroy, secp256k1_context_preallocated_destroy, + * and secp256k1_context_randomize. + * + * Regarding randomization, either do it once at creation time (in which case + * you do not need any locking for the other calls), or use a read-write lock. + */ +typedef struct secp256k1_context_struct secp256k1_context; + +/** Opaque data structure that holds rewriteable "scratch space" + * + * The purpose of this structure is to replace dynamic memory allocations, + * because we target architectures where this may not be available. It is + * essentially a resizable (within specified parameters) block of bytes, + * which is initially created either by memory allocation or TODO as a pointer + * into some fixed rewritable space. + * + * Unlike the context object, this cannot safely be shared between threads + * without additional synchronization logic. + */ +typedef struct secp256k1_scratch_space_struct secp256k1_scratch_space; + +/** Opaque data structure that holds a parsed and valid public key. + * + * The exact representation of data inside is implementation defined and not + * guaranteed to be portable between different platforms or versions. It is + * however guaranteed to be 64 bytes in size, and can be safely copied/moved. + * If you need to convert to a format suitable for storage, transmission, or + * comparison, use secp256k1_ec_pubkey_serialize and secp256k1_ec_pubkey_parse. + */ +typedef struct { + unsigned char data[64]; +} secp256k1_pubkey; + +/** Opaque data structured that holds a parsed ECDSA signature. + * + * The exact representation of data inside is implementation defined and not + * guaranteed to be portable between different platforms or versions. It is + * however guaranteed to be 64 bytes in size, and can be safely copied/moved. + * If you need to convert to a format suitable for storage, transmission, or + * comparison, use the secp256k1_ecdsa_signature_serialize_* and + * secp256k1_ecdsa_signature_parse_* functions. + */ +typedef struct { + unsigned char data[64]; +} secp256k1_ecdsa_signature; + +/** A pointer to a function to deterministically generate a nonce. + * + * Returns: 1 if a nonce was successfully generated. 0 will cause signing to fail. + * Out: nonce32: pointer to a 32-byte array to be filled by the function. + * In: msg32: the 32-byte message hash being verified (will not be NULL) + * key32: pointer to a 32-byte secret key (will not be NULL) + * algo16: pointer to a 16-byte array describing the signature + * algorithm (will be NULL for ECDSA for compatibility). + * data: Arbitrary data pointer that is passed through. + * attempt: how many iterations we have tried to find a nonce. + * This will almost always be 0, but different attempt values + * are required to result in a different nonce. + * + * Except for test cases, this function should compute some cryptographic hash of + * the message, the algorithm, the key and the attempt. + */ +typedef int (*secp256k1_nonce_function)( + unsigned char *nonce32, + const unsigned char *msg32, + const unsigned char *key32, + const unsigned char *algo16, + void *data, + unsigned int attempt +); + +# if !defined(SECP256K1_GNUC_PREREQ) +# if defined(__GNUC__)&&defined(__GNUC_MINOR__) +# define SECP256K1_GNUC_PREREQ(_maj,_min) \ + ((__GNUC__<<16)+__GNUC_MINOR__>=((_maj)<<16)+(_min)) +# else +# define SECP256K1_GNUC_PREREQ(_maj,_min) 0 +# endif +# endif + +# if (!defined(__STDC_VERSION__) || (__STDC_VERSION__ < 199901L) ) +# if SECP256K1_GNUC_PREREQ(2,7) +# define SECP256K1_INLINE __inline__ +# elif (defined(_MSC_VER)) +# define SECP256K1_INLINE __inline +# else +# define SECP256K1_INLINE +# endif +# else +# define SECP256K1_INLINE inline +# endif + +#ifndef SECP256K1_API +# if defined(_WIN32) +# ifdef SECP256K1_BUILD +# define SECP256K1_API __declspec(dllexport) +# else +# define SECP256K1_API +# endif +# elif defined(__GNUC__) && defined(SECP256K1_BUILD) +# define SECP256K1_API __attribute__ ((visibility ("default"))) +# else +# define SECP256K1_API +# endif +#endif + +/**Warning attributes + * NONNULL is not used if SECP256K1_BUILD is set to avoid the compiler optimizing out + * some paranoid null checks. */ +# if defined(__GNUC__) && SECP256K1_GNUC_PREREQ(3, 4) +# define SECP256K1_WARN_UNUSED_RESULT __attribute__ ((__warn_unused_result__)) +# else +# define SECP256K1_WARN_UNUSED_RESULT +# endif +# if !defined(SECP256K1_BUILD) && defined(__GNUC__) && SECP256K1_GNUC_PREREQ(3, 4) +# define SECP256K1_ARG_NONNULL(_x) __attribute__ ((__nonnull__(_x))) +# else +# define SECP256K1_ARG_NONNULL(_x) +# endif + +/** All flags' lower 8 bits indicate what they're for. Do not use directly. */ +#define SECP256K1_FLAGS_TYPE_MASK ((1 << 8) - 1) +#define SECP256K1_FLAGS_TYPE_CONTEXT (1 << 0) +#define SECP256K1_FLAGS_TYPE_COMPRESSION (1 << 1) +/** The higher bits contain the actual data. Do not use directly. */ +#define SECP256K1_FLAGS_BIT_CONTEXT_VERIFY (1 << 8) +#define SECP256K1_FLAGS_BIT_CONTEXT_SIGN (1 << 9) +#define SECP256K1_FLAGS_BIT_COMPRESSION (1 << 8) + +/** Flags to pass to secp256k1_context_create, secp256k1_context_preallocated_size, and + * secp256k1_context_preallocated_create. */ +#define SECP256K1_CONTEXT_VERIFY (SECP256K1_FLAGS_TYPE_CONTEXT | SECP256K1_FLAGS_BIT_CONTEXT_VERIFY) +#define SECP256K1_CONTEXT_SIGN (SECP256K1_FLAGS_TYPE_CONTEXT | SECP256K1_FLAGS_BIT_CONTEXT_SIGN) +#define SECP256K1_CONTEXT_NONE (SECP256K1_FLAGS_TYPE_CONTEXT) + +/** Flag to pass to secp256k1_ec_pubkey_serialize. */ +#define SECP256K1_EC_COMPRESSED (SECP256K1_FLAGS_TYPE_COMPRESSION | SECP256K1_FLAGS_BIT_COMPRESSION) +#define SECP256K1_EC_UNCOMPRESSED (SECP256K1_FLAGS_TYPE_COMPRESSION) + +/** Prefix byte used to tag various encoded curvepoints for specific purposes */ +#define SECP256K1_TAG_PUBKEY_EVEN 0x02 +#define SECP256K1_TAG_PUBKEY_ODD 0x03 +#define SECP256K1_TAG_PUBKEY_UNCOMPRESSED 0x04 +#define SECP256K1_TAG_PUBKEY_HYBRID_EVEN 0x06 +#define SECP256K1_TAG_PUBKEY_HYBRID_ODD 0x07 + +/** A simple secp256k1 context object with no precomputed tables. These are useful for + * type serialization/parsing functions which require a context object to maintain + * API consistency, but currently do not require expensive precomputations or dynamic + * allocations. + */ +SECP256K1_API extern const secp256k1_context *secp256k1_context_no_precomp; + +/** Create a secp256k1 context object (in dynamically allocated memory). + * + * This function uses malloc to allocate memory. It is guaranteed that malloc is + * called at most once for every call of this function. If you need to avoid dynamic + * memory allocation entirely, see the functions in secp256k1_preallocated.h. + * + * Returns: a newly created context object. + * In: flags: which parts of the context to initialize. + * + * See also secp256k1_context_randomize. + */ +SECP256K1_API secp256k1_context* secp256k1_context_create( + unsigned int flags +) SECP256K1_WARN_UNUSED_RESULT; + +/** Copy a secp256k1 context object (into dynamically allocated memory). + * + * This function uses malloc to allocate memory. It is guaranteed that malloc is + * called at most once for every call of this function. If you need to avoid dynamic + * memory allocation entirely, see the functions in secp256k1_preallocated.h. + * + * Returns: a newly created context object. + * Args: ctx: an existing context to copy (cannot be NULL) + */ +SECP256K1_API secp256k1_context* secp256k1_context_clone( + const secp256k1_context* ctx +) SECP256K1_ARG_NONNULL(1) SECP256K1_WARN_UNUSED_RESULT; + +/** Destroy a secp256k1 context object (created in dynamically allocated memory). + * + * The context pointer may not be used afterwards. + * + * The context to destroy must have been created using secp256k1_context_create + * or secp256k1_context_clone. If the context has instead been created using + * secp256k1_context_preallocated_create or secp256k1_context_preallocated_clone, the + * behaviour is undefined. In that case, secp256k1_context_preallocated_destroy must + * be used instead. + * + * Args: ctx: an existing context to destroy, constructed using + * secp256k1_context_create or secp256k1_context_clone + */ +SECP256K1_API void secp256k1_context_destroy( + secp256k1_context* ctx +); + +/** Set a callback function to be called when an illegal argument is passed to + * an API call. It will only trigger for violations that are mentioned + * explicitly in the header. + * + * The philosophy is that these shouldn't be dealt with through a + * specific return value, as calling code should not have branches to deal with + * the case that this code itself is broken. + * + * On the other hand, during debug stage, one would want to be informed about + * such mistakes, and the default (crashing) may be inadvisable. + * When this callback is triggered, the API function called is guaranteed not + * to cause a crash, though its return value and output arguments are + * undefined. + * + * When this function has not been called (or called with fn==NULL), then the + * default handler will be used. The library provides a default handler which + * writes the message to stderr and calls abort. This default handler can be + * replaced at link time if the preprocessor macro + * USE_EXTERNAL_DEFAULT_CALLBACKS is defined, which is the case if the build + * has been configured with --enable-external-default-callbacks. Then the + * following two symbols must be provided to link against: + * - void secp256k1_default_illegal_callback_fn(const char* message, void* data); + * - void secp256k1_default_error_callback_fn(const char* message, void* data); + * The library can call these default handlers even before a proper callback data + * pointer could have been set using secp256k1_context_set_illegal_callback or + * secp256k1_context_set_error_callback, e.g., when the creation of a context + * fails. In this case, the corresponding default handler will be called with + * the data pointer argument set to NULL. + * + * Args: ctx: an existing context object (cannot be NULL) + * In: fun: a pointer to a function to call when an illegal argument is + * passed to the API, taking a message and an opaque pointer. + * (NULL restores the default handler.) + * data: the opaque pointer to pass to fun above. + * + * See also secp256k1_context_set_error_callback. + */ +SECP256K1_API void secp256k1_context_set_illegal_callback( + secp256k1_context* ctx, + void (*fun)(const char* message, void* data), + const void* data +) SECP256K1_ARG_NONNULL(1); + +/** Set a callback function to be called when an internal consistency check + * fails. The default is crashing. + * + * This can only trigger in case of a hardware failure, miscompilation, + * memory corruption, serious bug in the library, or other error would can + * otherwise result in undefined behaviour. It will not trigger due to mere + * incorrect usage of the API (see secp256k1_context_set_illegal_callback + * for that). After this callback returns, anything may happen, including + * crashing. + * + * Args: ctx: an existing context object (cannot be NULL) + * In: fun: a pointer to a function to call when an internal error occurs, + * taking a message and an opaque pointer (NULL restores the + * default handler, see secp256k1_context_set_illegal_callback + * for details). + * data: the opaque pointer to pass to fun above. + * + * See also secp256k1_context_set_illegal_callback. + */ +SECP256K1_API void secp256k1_context_set_error_callback( + secp256k1_context* ctx, + void (*fun)(const char* message, void* data), + const void* data +) SECP256K1_ARG_NONNULL(1); + +/** Create a secp256k1 scratch space object. + * + * Returns: a newly created scratch space. + * Args: ctx: an existing context object (cannot be NULL) + * In: size: amount of memory to be available as scratch space. Some extra + * (<100 bytes) will be allocated for extra accounting. + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT secp256k1_scratch_space* secp256k1_scratch_space_create( + const secp256k1_context* ctx, + size_t size +) SECP256K1_ARG_NONNULL(1); + +/** Destroy a secp256k1 scratch space. + * + * The pointer may not be used afterwards. + * Args: ctx: a secp256k1 context object. + * scratch: space to destroy + */ +SECP256K1_API void secp256k1_scratch_space_destroy( + const secp256k1_context* ctx, + secp256k1_scratch_space* scratch +) SECP256K1_ARG_NONNULL(1); + +/** Parse a variable-length public key into the pubkey object. + * + * Returns: 1 if the public key was fully valid. + * 0 if the public key could not be parsed or is invalid. + * Args: ctx: a secp256k1 context object. + * Out: pubkey: pointer to a pubkey object. If 1 is returned, it is set to a + * parsed version of input. If not, its value is undefined. + * In: input: pointer to a serialized public key + * inputlen: length of the array pointed to by input + * + * This function supports parsing compressed (33 bytes, header byte 0x02 or + * 0x03), uncompressed (65 bytes, header byte 0x04), or hybrid (65 bytes, header + * byte 0x06 or 0x07) format public keys. + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ec_pubkey_parse( + const secp256k1_context* ctx, + secp256k1_pubkey* pubkey, + const unsigned char *input, + size_t inputlen +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +/** Serialize a pubkey object into a serialized byte sequence. + * + * Returns: 1 always. + * Args: ctx: a secp256k1 context object. + * Out: output: a pointer to a 65-byte (if compressed==0) or 33-byte (if + * compressed==1) byte array to place the serialized key + * in. + * In/Out: outputlen: a pointer to an integer which is initially set to the + * size of output, and is overwritten with the written + * size. + * In: pubkey: a pointer to a secp256k1_pubkey containing an + * initialized public key. + * flags: SECP256K1_EC_COMPRESSED if serialization should be in + * compressed format, otherwise SECP256K1_EC_UNCOMPRESSED. + */ +SECP256K1_API int secp256k1_ec_pubkey_serialize( + const secp256k1_context* ctx, + unsigned char *output, + size_t *outputlen, + const secp256k1_pubkey* pubkey, + unsigned int flags +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3) SECP256K1_ARG_NONNULL(4); + +/** Parse an ECDSA signature in compact (64 bytes) format. + * + * Returns: 1 when the signature could be parsed, 0 otherwise. + * Args: ctx: a secp256k1 context object + * Out: sig: a pointer to a signature object + * In: input64: a pointer to the 64-byte array to parse + * + * The signature must consist of a 32-byte big endian R value, followed by a + * 32-byte big endian S value. If R or S fall outside of [0..order-1], the + * encoding is invalid. R and S with value 0 are allowed in the encoding. + * + * After the call, sig will always be initialized. If parsing failed or R or + * S are zero, the resulting sig value is guaranteed to fail validation for any + * message and public key. + */ +SECP256K1_API int secp256k1_ecdsa_signature_parse_compact( + const secp256k1_context* ctx, + secp256k1_ecdsa_signature* sig, + const unsigned char *input64 +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +/** Parse a DER ECDSA signature. + * + * Returns: 1 when the signature could be parsed, 0 otherwise. + * Args: ctx: a secp256k1 context object + * Out: sig: a pointer to a signature object + * In: input: a pointer to the signature to be parsed + * inputlen: the length of the array pointed to be input + * + * This function will accept any valid DER encoded signature, even if the + * encoded numbers are out of range. + * + * After the call, sig will always be initialized. If parsing failed or the + * encoded numbers are out of range, signature validation with it is + * guaranteed to fail for every message and public key. + */ +SECP256K1_API int secp256k1_ecdsa_signature_parse_der( + const secp256k1_context* ctx, + secp256k1_ecdsa_signature* sig, + const unsigned char *input, + size_t inputlen +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +/** Serialize an ECDSA signature in DER format. + * + * Returns: 1 if enough space was available to serialize, 0 otherwise + * Args: ctx: a secp256k1 context object + * Out: output: a pointer to an array to store the DER serialization + * In/Out: outputlen: a pointer to a length integer. Initially, this integer + * should be set to the length of output. After the call + * it will be set to the length of the serialization (even + * if 0 was returned). + * In: sig: a pointer to an initialized signature object + */ +SECP256K1_API int secp256k1_ecdsa_signature_serialize_der( + const secp256k1_context* ctx, + unsigned char *output, + size_t *outputlen, + const secp256k1_ecdsa_signature* sig +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3) SECP256K1_ARG_NONNULL(4); + +/** Serialize an ECDSA signature in compact (64 byte) format. + * + * Returns: 1 + * Args: ctx: a secp256k1 context object + * Out: output64: a pointer to a 64-byte array to store the compact serialization + * In: sig: a pointer to an initialized signature object + * + * See secp256k1_ecdsa_signature_parse_compact for details about the encoding. + */ +SECP256K1_API int secp256k1_ecdsa_signature_serialize_compact( + const secp256k1_context* ctx, + unsigned char *output64, + const secp256k1_ecdsa_signature* sig +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +/** Verify an ECDSA signature. + * + * Returns: 1: correct signature + * 0: incorrect or unparseable signature + * Args: ctx: a secp256k1 context object, initialized for verification. + * In: sig: the signature being verified (cannot be NULL) + * msg32: the 32-byte message hash being verified (cannot be NULL) + * pubkey: pointer to an initialized public key to verify with (cannot be NULL) + * + * To avoid accepting malleable signatures, only ECDSA signatures in lower-S + * form are accepted. + * + * If you need to accept ECDSA signatures from sources that do not obey this + * rule, apply secp256k1_ecdsa_signature_normalize to the signature prior to + * validation, but be aware that doing so results in malleable signatures. + * + * For details, see the comments for that function. + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ecdsa_verify( + const secp256k1_context* ctx, + const secp256k1_ecdsa_signature *sig, + const unsigned char *msg32, + const secp256k1_pubkey *pubkey +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3) SECP256K1_ARG_NONNULL(4); + +/** Convert a signature to a normalized lower-S form. + * + * Returns: 1 if sigin was not normalized, 0 if it already was. + * Args: ctx: a secp256k1 context object + * Out: sigout: a pointer to a signature to fill with the normalized form, + * or copy if the input was already normalized. (can be NULL if + * you're only interested in whether the input was already + * normalized). + * In: sigin: a pointer to a signature to check/normalize (cannot be NULL, + * can be identical to sigout) + * + * With ECDSA a third-party can forge a second distinct signature of the same + * message, given a single initial signature, but without knowing the key. This + * is done by negating the S value modulo the order of the curve, 'flipping' + * the sign of the random point R which is not included in the signature. + * + * Forgery of the same message isn't universally problematic, but in systems + * where message malleability or uniqueness of signatures is important this can + * cause issues. This forgery can be blocked by all verifiers forcing signers + * to use a normalized form. + * + * The lower-S form reduces the size of signatures slightly on average when + * variable length encodings (such as DER) are used and is cheap to verify, + * making it a good choice. Security of always using lower-S is assured because + * anyone can trivially modify a signature after the fact to enforce this + * property anyway. + * + * The lower S value is always between 0x1 and + * 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0, + * inclusive. + * + * No other forms of ECDSA malleability are known and none seem likely, but + * there is no formal proof that ECDSA, even with this additional restriction, + * is free of other malleability. Commonly used serialization schemes will also + * accept various non-unique encodings, so care should be taken when this + * property is required for an application. + * + * The secp256k1_ecdsa_sign function will by default create signatures in the + * lower-S form, and secp256k1_ecdsa_verify will not accept others. In case + * signatures come from a system that cannot enforce this property, + * secp256k1_ecdsa_signature_normalize must be called before verification. + */ +SECP256K1_API int secp256k1_ecdsa_signature_normalize( + const secp256k1_context* ctx, + secp256k1_ecdsa_signature *sigout, + const secp256k1_ecdsa_signature *sigin +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(3); + +/** An implementation of RFC6979 (using HMAC-SHA256) as nonce generation function. + * If a data pointer is passed, it is assumed to be a pointer to 32 bytes of + * extra entropy. + */ +SECP256K1_API extern const secp256k1_nonce_function secp256k1_nonce_function_rfc6979; + +/** A default safe nonce generation function (currently equal to secp256k1_nonce_function_rfc6979). */ +SECP256K1_API extern const secp256k1_nonce_function secp256k1_nonce_function_default; + +/** Create an ECDSA signature. + * + * Returns: 1: signature created + * 0: the nonce generation function failed, or the private key was invalid. + * Args: ctx: pointer to a context object, initialized for signing (cannot be NULL) + * Out: sig: pointer to an array where the signature will be placed (cannot be NULL) + * In: msg32: the 32-byte message hash being signed (cannot be NULL) + * seckey: pointer to a 32-byte secret key (cannot be NULL) + * noncefp:pointer to a nonce generation function. If NULL, secp256k1_nonce_function_default is used + * ndata: pointer to arbitrary data used by the nonce generation function (can be NULL) + * + * The created signature is always in lower-S form. See + * secp256k1_ecdsa_signature_normalize for more details. + */ +SECP256K1_API int secp256k1_ecdsa_sign( + const secp256k1_context* ctx, + secp256k1_ecdsa_signature *sig, + const unsigned char *msg32, + const unsigned char *seckey, + secp256k1_nonce_function noncefp, + const void *ndata +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3) SECP256K1_ARG_NONNULL(4); + +/** Verify an ECDSA secret key. + * + * Returns: 1: secret key is valid + * 0: secret key is invalid + * Args: ctx: pointer to a context object (cannot be NULL) + * In: seckey: pointer to a 32-byte secret key (cannot be NULL) + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ec_seckey_verify( + const secp256k1_context* ctx, + const unsigned char *seckey +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2); + +/** Compute the public key for a secret key. + * + * Returns: 1: secret was valid, public key stores + * 0: secret was invalid, try again + * Args: ctx: pointer to a context object, initialized for signing (cannot be NULL) + * Out: pubkey: pointer to the created public key (cannot be NULL) + * In: seckey: pointer to a 32-byte private key (cannot be NULL) + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ec_pubkey_create( + const secp256k1_context* ctx, + secp256k1_pubkey *pubkey, + const unsigned char *seckey +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +/** Negates a private key in place. + * + * Returns: 1 always + * Args: ctx: pointer to a context object + * In/Out: seckey: pointer to the 32-byte private key to be negated (cannot be NULL) + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ec_privkey_negate( + const secp256k1_context* ctx, + unsigned char *seckey +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2); + +/** Negates a public key in place. + * + * Returns: 1 always + * Args: ctx: pointer to a context object + * In/Out: pubkey: pointer to the public key to be negated (cannot be NULL) + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ec_pubkey_negate( + const secp256k1_context* ctx, + secp256k1_pubkey *pubkey +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2); + +/** Tweak a private key by adding tweak to it. + * Returns: 0 if the tweak was out of range (chance of around 1 in 2^128 for + * uniformly random 32-byte arrays, or if the resulting private key + * would be invalid (only when the tweak is the complement of the + * private key). 1 otherwise. + * Args: ctx: pointer to a context object (cannot be NULL). + * In/Out: seckey: pointer to a 32-byte private key. + * In: tweak: pointer to a 32-byte tweak. + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ec_privkey_tweak_add( + const secp256k1_context* ctx, + unsigned char *seckey, + const unsigned char *tweak +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +/** Tweak a public key by adding tweak times the generator to it. + * Returns: 0 if the tweak was out of range (chance of around 1 in 2^128 for + * uniformly random 32-byte arrays, or if the resulting public key + * would be invalid (only when the tweak is the complement of the + * corresponding private key). 1 otherwise. + * Args: ctx: pointer to a context object initialized for validation + * (cannot be NULL). + * In/Out: pubkey: pointer to a public key object. + * In: tweak: pointer to a 32-byte tweak. + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ec_pubkey_tweak_add( + const secp256k1_context* ctx, + secp256k1_pubkey *pubkey, + const unsigned char *tweak +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +/** Tweak a private key by multiplying it by a tweak. + * Returns: 0 if the tweak was out of range (chance of around 1 in 2^128 for + * uniformly random 32-byte arrays, or equal to zero. 1 otherwise. + * Args: ctx: pointer to a context object (cannot be NULL). + * In/Out: seckey: pointer to a 32-byte private key. + * In: tweak: pointer to a 32-byte tweak. + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ec_privkey_tweak_mul( + const secp256k1_context* ctx, + unsigned char *seckey, + const unsigned char *tweak +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +/** Tweak a public key by multiplying it by a tweak value. + * Returns: 0 if the tweak was out of range (chance of around 1 in 2^128 for + * uniformly random 32-byte arrays, or equal to zero. 1 otherwise. + * Args: ctx: pointer to a context object initialized for validation + * (cannot be NULL). + * In/Out: pubkey: pointer to a public key object. + * In: tweak: pointer to a 32-byte tweak. + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ec_pubkey_tweak_mul( + const secp256k1_context* ctx, + secp256k1_pubkey *pubkey, + const unsigned char *tweak +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +/** Updates the context randomization to protect against side-channel leakage. + * Returns: 1: randomization successfully updated or nothing to randomize + * 0: error + * Args: ctx: pointer to a context object (cannot be NULL) + * In: seed32: pointer to a 32-byte random seed (NULL resets to initial state) + * + * While secp256k1 code is written to be constant-time no matter what secret + * values are, it's possible that a future compiler may output code which isn't, + * and also that the CPU may not emit the same radio frequencies or draw the same + * amount power for all values. + * + * This function provides a seed which is combined into the blinding value: that + * blinding value is added before each multiplication (and removed afterwards) so + * that it does not affect function results, but shields against attacks which + * rely on any input-dependent behaviour. + * + * This function has currently an effect only on contexts initialized for signing + * because randomization is currently used only for signing. However, this is not + * guaranteed and may change in the future. It is safe to call this function on + * contexts not initialized for signing; then it will have no effect and return 1. + * + * You should call this after secp256k1_context_create or + * secp256k1_context_clone (and secp256k1_context_preallocated_create or + * secp256k1_context_clone, resp.), and you may call this repeatedly afterwards. + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_context_randomize( + secp256k1_context* ctx, + const unsigned char *seed32 +) SECP256K1_ARG_NONNULL(1); + +/** Add a number of public keys together. + * Returns: 1: the sum of the public keys is valid. + * 0: the sum of the public keys is not valid. + * Args: ctx: pointer to a context object + * Out: out: pointer to a public key object for placing the resulting public key + * (cannot be NULL) + * In: ins: pointer to array of pointers to public keys (cannot be NULL) + * n: the number of public keys to add together (must be at least 1) + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ec_pubkey_combine( + const secp256k1_context* ctx, + secp256k1_pubkey *out, + const secp256k1_pubkey * const * ins, + size_t n +) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +#ifdef __cplusplus +} +#endif + +#endif /* SECP256K1_H */ diff --git a/node_modules/secp256k1/src/secp256k1/include/secp256k1_ecdh.h b/node_modules/secp256k1/src/secp256k1/include/secp256k1_ecdh.h new file mode 100644 index 0000000..df5fde2 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/include/secp256k1_ecdh.h @@ -0,0 +1,55 @@ +#ifndef SECP256K1_ECDH_H +#define SECP256K1_ECDH_H + +#include "secp256k1.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** A pointer to a function that applies hash function to a point + * + * Returns: 1 if a point was successfully hashed. 0 will cause ecdh to fail + * Out: output: pointer to an array to be filled by the function + * In: x: pointer to a 32-byte x coordinate + * y: pointer to a 32-byte y coordinate + * data: Arbitrary data pointer that is passed through + */ +typedef int (*secp256k1_ecdh_hash_function)( + unsigned char *output, + const unsigned char *x, + const unsigned char *y, + void *data +); + +/** An implementation of SHA256 hash function that applies to compressed public key. */ +SECP256K1_API extern const secp256k1_ecdh_hash_function secp256k1_ecdh_hash_function_sha256; + +/** A default ecdh hash function (currently equal to secp256k1_ecdh_hash_function_sha256). */ +SECP256K1_API extern const secp256k1_ecdh_hash_function secp256k1_ecdh_hash_function_default; + +/** Compute an EC Diffie-Hellman secret in constant time + * Returns: 1: exponentiation was successful + * 0: scalar was invalid (zero or overflow) + * Args: ctx: pointer to a context object (cannot be NULL) + * Out: output: pointer to an array to be filled by the function + * In: pubkey: a pointer to a secp256k1_pubkey containing an + * initialized public key + * privkey: a 32-byte scalar with which to multiply the point + * hashfp: pointer to a hash function. If NULL, secp256k1_ecdh_hash_function_sha256 is used + * data: Arbitrary data pointer that is passed through + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ecdh( + const secp256k1_context* ctx, + unsigned char *output, + const secp256k1_pubkey *pubkey, + const unsigned char *privkey, + secp256k1_ecdh_hash_function hashfp, + void *data +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3) SECP256K1_ARG_NONNULL(4); + +#ifdef __cplusplus +} +#endif + +#endif /* SECP256K1_ECDH_H */ diff --git a/node_modules/secp256k1/src/secp256k1/include/secp256k1_preallocated.h b/node_modules/secp256k1/src/secp256k1/include/secp256k1_preallocated.h new file mode 100644 index 0000000..a9ae15d --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/include/secp256k1_preallocated.h @@ -0,0 +1,128 @@ +#ifndef SECP256K1_PREALLOCATED_H +#define SECP256K1_PREALLOCATED_H + +#include "secp256k1.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/* The module provided by this header file is intended for settings in which it + * is not possible or desirable to rely on dynamic memory allocation. It provides + * functions for creating, cloning, and destroying secp256k1 context objects in a + * contiguous fixed-size block of memory provided by the caller. + * + * Context objects created by functions in this module can be used like contexts + * objects created by functions in secp256k1.h, i.e., they can be passed to any + * API function that expects a context object (see secp256k1.h for details). The + * only exception is that context objects created by functions in this module + * must be destroyed using secp256k1_context_preallocated_destroy (in this + * module) instead of secp256k1_context_destroy (in secp256k1.h). + * + * It is guaranteed that functions in this module will not call malloc or its + * friends realloc, calloc, and free. + */ + +/** Determine the memory size of a secp256k1 context object to be created in + * caller-provided memory. + * + * The purpose of this function is to determine how much memory must be provided + * to secp256k1_context_preallocated_create. + * + * Returns: the required size of the caller-provided memory block + * In: flags: which parts of the context to initialize. + */ +SECP256K1_API size_t secp256k1_context_preallocated_size( + unsigned int flags +) SECP256K1_WARN_UNUSED_RESULT; + +/** Create a secp256k1 context object in caller-provided memory. + * + * The caller must provide a pointer to a rewritable contiguous block of memory + * of size at least secp256k1_context_preallocated_size(flags) bytes, suitably + * aligned to hold an object of any type. + * + * The block of memory is exclusively owned by the created context object during + * the lifetime of this context object, which begins with the call to this + * function and ends when a call to secp256k1_context_preallocated_destroy + * (which destroys the context object again) returns. During the lifetime of the + * context object, the caller is obligated not to access this block of memory, + * i.e., the caller may not read or write the memory, e.g., by copying the memory + * contents to a different location or trying to create a second context object + * in the memory. In simpler words, the prealloc pointer (or any pointer derived + * from it) should not be used during the lifetime of the context object. + * + * Returns: a newly created context object. + * In: prealloc: a pointer to a rewritable contiguous block of memory of + * size at least secp256k1_context_preallocated_size(flags) + * bytes, as detailed above (cannot be NULL) + * flags: which parts of the context to initialize. + * + * See also secp256k1_context_randomize (in secp256k1.h) + * and secp256k1_context_preallocated_destroy. + */ +SECP256K1_API secp256k1_context* secp256k1_context_preallocated_create( + void* prealloc, + unsigned int flags +) SECP256K1_ARG_NONNULL(1) SECP256K1_WARN_UNUSED_RESULT; + +/** Determine the memory size of a secp256k1 context object to be copied into + * caller-provided memory. + * + * Returns: the required size of the caller-provided memory block. + * In: ctx: an existing context to copy (cannot be NULL) + */ +SECP256K1_API size_t secp256k1_context_preallocated_clone_size( + const secp256k1_context* ctx +) SECP256K1_ARG_NONNULL(1) SECP256K1_WARN_UNUSED_RESULT; + +/** Copy a secp256k1 context object into caller-provided memory. + * + * The caller must provide a pointer to a rewritable contiguous block of memory + * of size at least secp256k1_context_preallocated_size(flags) bytes, suitably + * aligned to hold an object of any type. + * + * The block of memory is exclusively owned by the created context object during + * the lifetime of this context object, see the description of + * secp256k1_context_preallocated_create for details. + * + * Returns: a newly created context object. + * Args: ctx: an existing context to copy (cannot be NULL) + * In: prealloc: a pointer to a rewritable contiguous block of memory of + * size at least secp256k1_context_preallocated_size(flags) + * bytes, as detailed above (cannot be NULL) + */ +SECP256K1_API secp256k1_context* secp256k1_context_preallocated_clone( + const secp256k1_context* ctx, + void* prealloc +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_WARN_UNUSED_RESULT; + +/** Destroy a secp256k1 context object that has been created in + * caller-provided memory. + * + * The context pointer may not be used afterwards. + * + * The context to destroy must have been created using + * secp256k1_context_preallocated_create or secp256k1_context_preallocated_clone. + * If the context has instead been created using secp256k1_context_create or + * secp256k1_context_clone, the behaviour is undefined. In that case, + * secp256k1_context_destroy must be used instead. + * + * If required, it is the responsibility of the caller to deallocate the block + * of memory properly after this function returns, e.g., by calling free on the + * preallocated pointer given to secp256k1_context_preallocated_create or + * secp256k1_context_preallocated_clone. + * + * Args: ctx: an existing context to destroy, constructed using + * secp256k1_context_preallocated_create or + * secp256k1_context_preallocated_clone (cannot be NULL) + */ +SECP256K1_API void secp256k1_context_preallocated_destroy( + secp256k1_context* ctx +); + +#ifdef __cplusplus +} +#endif + +#endif /* SECP256K1_PREALLOCATED_H */ diff --git a/node_modules/secp256k1/src/secp256k1/include/secp256k1_recovery.h b/node_modules/secp256k1/src/secp256k1/include/secp256k1_recovery.h new file mode 100644 index 0000000..cf6c5ed --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/include/secp256k1_recovery.h @@ -0,0 +1,110 @@ +#ifndef SECP256K1_RECOVERY_H +#define SECP256K1_RECOVERY_H + +#include "secp256k1.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** Opaque data structured that holds a parsed ECDSA signature, + * supporting pubkey recovery. + * + * The exact representation of data inside is implementation defined and not + * guaranteed to be portable between different platforms or versions. It is + * however guaranteed to be 65 bytes in size, and can be safely copied/moved. + * If you need to convert to a format suitable for storage or transmission, use + * the secp256k1_ecdsa_signature_serialize_* and + * secp256k1_ecdsa_signature_parse_* functions. + * + * Furthermore, it is guaranteed that identical signatures (including their + * recoverability) will have identical representation, so they can be + * memcmp'ed. + */ +typedef struct { + unsigned char data[65]; +} secp256k1_ecdsa_recoverable_signature; + +/** Parse a compact ECDSA signature (64 bytes + recovery id). + * + * Returns: 1 when the signature could be parsed, 0 otherwise + * Args: ctx: a secp256k1 context object + * Out: sig: a pointer to a signature object + * In: input64: a pointer to a 64-byte compact signature + * recid: the recovery id (0, 1, 2 or 3) + */ +SECP256K1_API int secp256k1_ecdsa_recoverable_signature_parse_compact( + const secp256k1_context* ctx, + secp256k1_ecdsa_recoverable_signature* sig, + const unsigned char *input64, + int recid +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +/** Convert a recoverable signature into a normal signature. + * + * Returns: 1 + * Out: sig: a pointer to a normal signature (cannot be NULL). + * In: sigin: a pointer to a recoverable signature (cannot be NULL). + */ +SECP256K1_API int secp256k1_ecdsa_recoverable_signature_convert( + const secp256k1_context* ctx, + secp256k1_ecdsa_signature* sig, + const secp256k1_ecdsa_recoverable_signature* sigin +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3); + +/** Serialize an ECDSA signature in compact format (64 bytes + recovery id). + * + * Returns: 1 + * Args: ctx: a secp256k1 context object + * Out: output64: a pointer to a 64-byte array of the compact signature (cannot be NULL) + * recid: a pointer to an integer to hold the recovery id (can be NULL). + * In: sig: a pointer to an initialized signature object (cannot be NULL) + */ +SECP256K1_API int secp256k1_ecdsa_recoverable_signature_serialize_compact( + const secp256k1_context* ctx, + unsigned char *output64, + int *recid, + const secp256k1_ecdsa_recoverable_signature* sig +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3) SECP256K1_ARG_NONNULL(4); + +/** Create a recoverable ECDSA signature. + * + * Returns: 1: signature created + * 0: the nonce generation function failed, or the private key was invalid. + * Args: ctx: pointer to a context object, initialized for signing (cannot be NULL) + * Out: sig: pointer to an array where the signature will be placed (cannot be NULL) + * In: msg32: the 32-byte message hash being signed (cannot be NULL) + * seckey: pointer to a 32-byte secret key (cannot be NULL) + * noncefp:pointer to a nonce generation function. If NULL, secp256k1_nonce_function_default is used + * ndata: pointer to arbitrary data used by the nonce generation function (can be NULL) + */ +SECP256K1_API int secp256k1_ecdsa_sign_recoverable( + const secp256k1_context* ctx, + secp256k1_ecdsa_recoverable_signature *sig, + const unsigned char *msg32, + const unsigned char *seckey, + secp256k1_nonce_function noncefp, + const void *ndata +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3) SECP256K1_ARG_NONNULL(4); + +/** Recover an ECDSA public key from a signature. + * + * Returns: 1: public key successfully recovered (which guarantees a correct signature). + * 0: otherwise. + * Args: ctx: pointer to a context object, initialized for verification (cannot be NULL) + * Out: pubkey: pointer to the recovered public key (cannot be NULL) + * In: sig: pointer to initialized signature that supports pubkey recovery (cannot be NULL) + * msg32: the 32-byte message hash assumed to be signed (cannot be NULL) + */ +SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_ecdsa_recover( + const secp256k1_context* ctx, + secp256k1_pubkey *pubkey, + const secp256k1_ecdsa_recoverable_signature *sig, + const unsigned char *msg32 +) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3) SECP256K1_ARG_NONNULL(4); + +#ifdef __cplusplus +} +#endif + +#endif /* SECP256K1_RECOVERY_H */ diff --git a/node_modules/secp256k1/src/secp256k1/libsecp256k1.pc.in b/node_modules/secp256k1/src/secp256k1/libsecp256k1.pc.in new file mode 100644 index 0000000..694e98e --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/libsecp256k1.pc.in @@ -0,0 +1,13 @@ +prefix=@prefix@ +exec_prefix=@exec_prefix@ +libdir=@libdir@ +includedir=@includedir@ + +Name: libsecp256k1 +Description: Optimized C library for EC operations on curve secp256k1 +URL: https://github.com/bitcoin-core/secp256k1 +Version: @PACKAGE_VERSION@ +Cflags: -I${includedir} +Libs: -L${libdir} -lsecp256k1 +Libs.private: @SECP_LIBS@ + diff --git a/node_modules/secp256k1/src/secp256k1/obj/.npmignore b/node_modules/secp256k1/src/secp256k1/obj/.npmignore new file mode 100644 index 0000000..e69de29 diff --git a/node_modules/secp256k1/src/secp256k1/sage/group_prover.sage b/node_modules/secp256k1/src/secp256k1/sage/group_prover.sage new file mode 100644 index 0000000..8521f07 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/sage/group_prover.sage @@ -0,0 +1,322 @@ +# This code supports verifying group implementations which have branches +# or conditional statements (like cmovs), by allowing each execution path +# to independently set assumptions on input or intermediary variables. +# +# The general approach is: +# * A constraint is a tuple of two sets of symbolic expressions: +# the first of which are required to evaluate to zero, the second of which +# are required to evaluate to nonzero. +# - A constraint is said to be conflicting if any of its nonzero expressions +# is in the ideal with basis the zero expressions (in other words: when the +# zero expressions imply that one of the nonzero expressions are zero). +# * There is a list of laws that describe the intended behaviour, including +# laws for addition and doubling. Each law is called with the symbolic point +# coordinates as arguments, and returns: +# - A constraint describing the assumptions under which it is applicable, +# called "assumeLaw" +# - A constraint describing the requirements of the law, called "require" +# * Implementations are transliterated into functions that operate as well on +# algebraic input points, and are called once per combination of branches +# executed. Each execution returns: +# - A constraint describing the assumptions this implementation requires +# (such as Z1=1), called "assumeFormula" +# - A constraint describing the assumptions this specific branch requires, +# but which is by construction guaranteed to cover the entire space by +# merging the results from all branches, called "assumeBranch" +# - The result of the computation +# * All combinations of laws with implementation branches are tried, and: +# - If the combination of assumeLaw, assumeFormula, and assumeBranch results +# in a conflict, it means this law does not apply to this branch, and it is +# skipped. +# - For others, we try to prove the require constraints hold, assuming the +# information in assumeLaw + assumeFormula + assumeBranch, and if this does +# not succeed, we fail. +# + To prove an expression is zero, we check whether it belongs to the +# ideal with the assumed zero expressions as basis. This test is exact. +# + To prove an expression is nonzero, we check whether each of its +# factors is contained in the set of nonzero assumptions' factors. +# This test is not exact, so various combinations of original and +# reduced expressions' factors are tried. +# - If we succeed, we print out the assumptions from assumeFormula that +# weren't implied by assumeLaw already. Those from assumeBranch are skipped, +# as we assume that all constraints in it are complementary with each other. +# +# Based on the sage verification scripts used in the Explicit-Formulas Database +# by Tanja Lange and others, see http://hyperelliptic.org/EFD + +class fastfrac: + """Fractions over rings.""" + + def __init__(self,R,top,bot=1): + """Construct a fractional, given a ring, a numerator, and denominator.""" + self.R = R + if parent(top) == ZZ or parent(top) == R: + self.top = R(top) + self.bot = R(bot) + elif top.__class__ == fastfrac: + self.top = top.top + self.bot = top.bot * bot + else: + self.top = R(numerator(top)) + self.bot = R(denominator(top)) * bot + + def iszero(self,I): + """Return whether this fraction is zero given an ideal.""" + return self.top in I and self.bot not in I + + def reduce(self,assumeZero): + zero = self.R.ideal(map(numerator, assumeZero)) + return fastfrac(self.R, zero.reduce(self.top)) / fastfrac(self.R, zero.reduce(self.bot)) + + def __add__(self,other): + """Add two fractions.""" + if parent(other) == ZZ: + return fastfrac(self.R,self.top + self.bot * other,self.bot) + if other.__class__ == fastfrac: + return fastfrac(self.R,self.top * other.bot + self.bot * other.top,self.bot * other.bot) + return NotImplemented + + def __sub__(self,other): + """Subtract two fractions.""" + if parent(other) == ZZ: + return fastfrac(self.R,self.top - self.bot * other,self.bot) + if other.__class__ == fastfrac: + return fastfrac(self.R,self.top * other.bot - self.bot * other.top,self.bot * other.bot) + return NotImplemented + + def __neg__(self): + """Return the negation of a fraction.""" + return fastfrac(self.R,-self.top,self.bot) + + def __mul__(self,other): + """Multiply two fractions.""" + if parent(other) == ZZ: + return fastfrac(self.R,self.top * other,self.bot) + if other.__class__ == fastfrac: + return fastfrac(self.R,self.top * other.top,self.bot * other.bot) + return NotImplemented + + def __rmul__(self,other): + """Multiply something else with a fraction.""" + return self.__mul__(other) + + def __div__(self,other): + """Divide two fractions.""" + if parent(other) == ZZ: + return fastfrac(self.R,self.top,self.bot * other) + if other.__class__ == fastfrac: + return fastfrac(self.R,self.top * other.bot,self.bot * other.top) + return NotImplemented + + def __pow__(self,other): + """Compute a power of a fraction.""" + if parent(other) == ZZ: + if other < 0: + # Negative powers require flipping top and bottom + return fastfrac(self.R,self.bot ^ (-other),self.top ^ (-other)) + else: + return fastfrac(self.R,self.top ^ other,self.bot ^ other) + return NotImplemented + + def __str__(self): + return "fastfrac((" + str(self.top) + ") / (" + str(self.bot) + "))" + def __repr__(self): + return "%s" % self + + def numerator(self): + return self.top + +class constraints: + """A set of constraints, consisting of zero and nonzero expressions. + + Constraints can either be used to express knowledge or a requirement. + + Both the fields zero and nonzero are maps from expressions to description + strings. The expressions that are the keys in zero are required to be zero, + and the expressions that are the keys in nonzero are required to be nonzero. + + Note that (a != 0) and (b != 0) is the same as (a*b != 0), so all keys in + nonzero could be multiplied into a single key. This is often much less + efficient to work with though, so we keep them separate inside the + constraints. This allows higher-level code to do fast checks on the individual + nonzero elements, or combine them if needed for stronger checks. + + We can't multiply the different zero elements, as it would suffice for one of + the factors to be zero, instead of all of them. Instead, the zero elements are + typically combined into an ideal first. + """ + + def __init__(self, **kwargs): + if 'zero' in kwargs: + self.zero = dict(kwargs['zero']) + else: + self.zero = dict() + if 'nonzero' in kwargs: + self.nonzero = dict(kwargs['nonzero']) + else: + self.nonzero = dict() + + def negate(self): + return constraints(zero=self.nonzero, nonzero=self.zero) + + def __add__(self, other): + zero = self.zero.copy() + zero.update(other.zero) + nonzero = self.nonzero.copy() + nonzero.update(other.nonzero) + return constraints(zero=zero, nonzero=nonzero) + + def __str__(self): + return "constraints(zero=%s,nonzero=%s)" % (self.zero, self.nonzero) + + def __repr__(self): + return "%s" % self + + +def conflicts(R, con): + """Check whether any of the passed non-zero assumptions is implied by the zero assumptions""" + zero = R.ideal(map(numerator, con.zero)) + if 1 in zero: + return True + # First a cheap check whether any of the individual nonzero terms conflict on + # their own. + for nonzero in con.nonzero: + if nonzero.iszero(zero): + return True + # It can be the case that entries in the nonzero set do not individually + # conflict with the zero set, but their combination does. For example, knowing + # that either x or y is zero is equivalent to having x*y in the zero set. + # Having x or y individually in the nonzero set is not a conflict, but both + # simultaneously is, so that is the right thing to check for. + if reduce(lambda a,b: a * b, con.nonzero, fastfrac(R, 1)).iszero(zero): + return True + return False + + +def get_nonzero_set(R, assume): + """Calculate a simple set of nonzero expressions""" + zero = R.ideal(map(numerator, assume.zero)) + nonzero = set() + for nz in map(numerator, assume.nonzero): + for (f,n) in nz.factor(): + nonzero.add(f) + rnz = zero.reduce(nz) + for (f,n) in rnz.factor(): + nonzero.add(f) + return nonzero + + +def prove_nonzero(R, exprs, assume): + """Check whether an expression is provably nonzero, given assumptions""" + zero = R.ideal(map(numerator, assume.zero)) + nonzero = get_nonzero_set(R, assume) + expl = set() + ok = True + for expr in exprs: + if numerator(expr) in zero: + return (False, [exprs[expr]]) + allexprs = reduce(lambda a,b: numerator(a)*numerator(b), exprs, 1) + for (f, n) in allexprs.factor(): + if f not in nonzero: + ok = False + if ok: + return (True, None) + ok = True + for (f, n) in zero.reduce(numerator(allexprs)).factor(): + if f not in nonzero: + ok = False + if ok: + return (True, None) + ok = True + for expr in exprs: + for (f,n) in numerator(expr).factor(): + if f not in nonzero: + ok = False + if ok: + return (True, None) + ok = True + for expr in exprs: + for (f,n) in zero.reduce(numerator(expr)).factor(): + if f not in nonzero: + expl.add(exprs[expr]) + if expl: + return (False, list(expl)) + else: + return (True, None) + + +def prove_zero(R, exprs, assume): + """Check whether all of the passed expressions are provably zero, given assumptions""" + r, e = prove_nonzero(R, dict(map(lambda x: (fastfrac(R, x.bot, 1), exprs[x]), exprs)), assume) + if not r: + return (False, map(lambda x: "Possibly zero denominator: %s" % x, e)) + zero = R.ideal(map(numerator, assume.zero)) + nonzero = prod(x for x in assume.nonzero) + expl = [] + for expr in exprs: + if not expr.iszero(zero): + expl.append(exprs[expr]) + if not expl: + return (True, None) + return (False, expl) + + +def describe_extra(R, assume, assumeExtra): + """Describe what assumptions are added, given existing assumptions""" + zerox = assume.zero.copy() + zerox.update(assumeExtra.zero) + zero = R.ideal(map(numerator, assume.zero)) + zeroextra = R.ideal(map(numerator, zerox)) + nonzero = get_nonzero_set(R, assume) + ret = set() + # Iterate over the extra zero expressions + for base in assumeExtra.zero: + if base not in zero: + add = [] + for (f, n) in numerator(base).factor(): + if f not in nonzero: + add += ["%s" % f] + if add: + ret.add((" * ".join(add)) + " = 0 [%s]" % assumeExtra.zero[base]) + # Iterate over the extra nonzero expressions + for nz in assumeExtra.nonzero: + nzr = zeroextra.reduce(numerator(nz)) + if nzr not in zeroextra: + for (f,n) in nzr.factor(): + if zeroextra.reduce(f) not in nonzero: + ret.add("%s != 0" % zeroextra.reduce(f)) + return ", ".join(x for x in ret) + + +def check_symbolic(R, assumeLaw, assumeAssert, assumeBranch, require): + """Check a set of zero and nonzero requirements, given a set of zero and nonzero assumptions""" + assume = assumeLaw + assumeAssert + assumeBranch + + if conflicts(R, assume): + # This formula does not apply + return None + + describe = describe_extra(R, assumeLaw + assumeBranch, assumeAssert) + + ok, msg = prove_zero(R, require.zero, assume) + if not ok: + return "FAIL, %s fails (assuming %s)" % (str(msg), describe) + + res, expl = prove_nonzero(R, require.nonzero, assume) + if not res: + return "FAIL, %s fails (assuming %s)" % (str(expl), describe) + + if describe != "": + return "OK (assuming %s)" % describe + else: + return "OK" + + +def concrete_verify(c): + for k in c.zero: + if k != 0: + return (False, c.zero[k]) + for k in c.nonzero: + if k == 0: + return (False, c.nonzero[k]) + return (True, None) diff --git a/node_modules/secp256k1/src/secp256k1/sage/secp256k1.sage b/node_modules/secp256k1/src/secp256k1/sage/secp256k1.sage new file mode 100644 index 0000000..a97e732 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/sage/secp256k1.sage @@ -0,0 +1,306 @@ +# Test libsecp256k1' group operation implementations using prover.sage + +import sys + +load("group_prover.sage") +load("weierstrass_prover.sage") + +def formula_secp256k1_gej_double_var(a): + """libsecp256k1's secp256k1_gej_double_var, used by various addition functions""" + rz = a.Z * a.Y + rz = rz * 2 + t1 = a.X^2 + t1 = t1 * 3 + t2 = t1^2 + t3 = a.Y^2 + t3 = t3 * 2 + t4 = t3^2 + t4 = t4 * 2 + t3 = t3 * a.X + rx = t3 + rx = rx * 4 + rx = -rx + rx = rx + t2 + t2 = -t2 + t3 = t3 * 6 + t3 = t3 + t2 + ry = t1 * t3 + t2 = -t4 + ry = ry + t2 + return jacobianpoint(rx, ry, rz) + +def formula_secp256k1_gej_add_var(branch, a, b): + """libsecp256k1's secp256k1_gej_add_var""" + if branch == 0: + return (constraints(), constraints(nonzero={a.Infinity : 'a_infinite'}), b) + if branch == 1: + return (constraints(), constraints(zero={a.Infinity : 'a_finite'}, nonzero={b.Infinity : 'b_infinite'}), a) + z22 = b.Z^2 + z12 = a.Z^2 + u1 = a.X * z22 + u2 = b.X * z12 + s1 = a.Y * z22 + s1 = s1 * b.Z + s2 = b.Y * z12 + s2 = s2 * a.Z + h = -u1 + h = h + u2 + i = -s1 + i = i + s2 + if branch == 2: + r = formula_secp256k1_gej_double_var(a) + return (constraints(), constraints(zero={h : 'h=0', i : 'i=0', a.Infinity : 'a_finite', b.Infinity : 'b_finite'}), r) + if branch == 3: + return (constraints(), constraints(zero={h : 'h=0', a.Infinity : 'a_finite', b.Infinity : 'b_finite'}, nonzero={i : 'i!=0'}), point_at_infinity()) + i2 = i^2 + h2 = h^2 + h3 = h2 * h + h = h * b.Z + rz = a.Z * h + t = u1 * h2 + rx = t + rx = rx * 2 + rx = rx + h3 + rx = -rx + rx = rx + i2 + ry = -rx + ry = ry + t + ry = ry * i + h3 = h3 * s1 + h3 = -h3 + ry = ry + h3 + return (constraints(), constraints(zero={a.Infinity : 'a_finite', b.Infinity : 'b_finite'}, nonzero={h : 'h!=0'}), jacobianpoint(rx, ry, rz)) + +def formula_secp256k1_gej_add_ge_var(branch, a, b): + """libsecp256k1's secp256k1_gej_add_ge_var, which assume bz==1""" + if branch == 0: + return (constraints(zero={b.Z - 1 : 'b.z=1'}), constraints(nonzero={a.Infinity : 'a_infinite'}), b) + if branch == 1: + return (constraints(zero={b.Z - 1 : 'b.z=1'}), constraints(zero={a.Infinity : 'a_finite'}, nonzero={b.Infinity : 'b_infinite'}), a) + z12 = a.Z^2 + u1 = a.X + u2 = b.X * z12 + s1 = a.Y + s2 = b.Y * z12 + s2 = s2 * a.Z + h = -u1 + h = h + u2 + i = -s1 + i = i + s2 + if (branch == 2): + r = formula_secp256k1_gej_double_var(a) + return (constraints(zero={b.Z - 1 : 'b.z=1'}), constraints(zero={a.Infinity : 'a_finite', b.Infinity : 'b_finite', h : 'h=0', i : 'i=0'}), r) + if (branch == 3): + return (constraints(zero={b.Z - 1 : 'b.z=1'}), constraints(zero={a.Infinity : 'a_finite', b.Infinity : 'b_finite', h : 'h=0'}, nonzero={i : 'i!=0'}), point_at_infinity()) + i2 = i^2 + h2 = h^2 + h3 = h * h2 + rz = a.Z * h + t = u1 * h2 + rx = t + rx = rx * 2 + rx = rx + h3 + rx = -rx + rx = rx + i2 + ry = -rx + ry = ry + t + ry = ry * i + h3 = h3 * s1 + h3 = -h3 + ry = ry + h3 + return (constraints(zero={b.Z - 1 : 'b.z=1'}), constraints(zero={a.Infinity : 'a_finite', b.Infinity : 'b_finite'}, nonzero={h : 'h!=0'}), jacobianpoint(rx, ry, rz)) + +def formula_secp256k1_gej_add_zinv_var(branch, a, b): + """libsecp256k1's secp256k1_gej_add_zinv_var""" + bzinv = b.Z^(-1) + if branch == 0: + return (constraints(), constraints(nonzero={b.Infinity : 'b_infinite'}), a) + if branch == 1: + bzinv2 = bzinv^2 + bzinv3 = bzinv2 * bzinv + rx = b.X * bzinv2 + ry = b.Y * bzinv3 + rz = 1 + return (constraints(), constraints(zero={b.Infinity : 'b_finite'}, nonzero={a.Infinity : 'a_infinite'}), jacobianpoint(rx, ry, rz)) + azz = a.Z * bzinv + z12 = azz^2 + u1 = a.X + u2 = b.X * z12 + s1 = a.Y + s2 = b.Y * z12 + s2 = s2 * azz + h = -u1 + h = h + u2 + i = -s1 + i = i + s2 + if branch == 2: + r = formula_secp256k1_gej_double_var(a) + return (constraints(), constraints(zero={a.Infinity : 'a_finite', b.Infinity : 'b_finite', h : 'h=0', i : 'i=0'}), r) + if branch == 3: + return (constraints(), constraints(zero={a.Infinity : 'a_finite', b.Infinity : 'b_finite', h : 'h=0'}, nonzero={i : 'i!=0'}), point_at_infinity()) + i2 = i^2 + h2 = h^2 + h3 = h * h2 + rz = a.Z + rz = rz * h + t = u1 * h2 + rx = t + rx = rx * 2 + rx = rx + h3 + rx = -rx + rx = rx + i2 + ry = -rx + ry = ry + t + ry = ry * i + h3 = h3 * s1 + h3 = -h3 + ry = ry + h3 + return (constraints(), constraints(zero={a.Infinity : 'a_finite', b.Infinity : 'b_finite'}, nonzero={h : 'h!=0'}), jacobianpoint(rx, ry, rz)) + +def formula_secp256k1_gej_add_ge(branch, a, b): + """libsecp256k1's secp256k1_gej_add_ge""" + zeroes = {} + nonzeroes = {} + a_infinity = False + if (branch & 4) != 0: + nonzeroes.update({a.Infinity : 'a_infinite'}) + a_infinity = True + else: + zeroes.update({a.Infinity : 'a_finite'}) + zz = a.Z^2 + u1 = a.X + u2 = b.X * zz + s1 = a.Y + s2 = b.Y * zz + s2 = s2 * a.Z + t = u1 + t = t + u2 + m = s1 + m = m + s2 + rr = t^2 + m_alt = -u2 + tt = u1 * m_alt + rr = rr + tt + degenerate = (branch & 3) == 3 + if (branch & 1) != 0: + zeroes.update({m : 'm_zero'}) + else: + nonzeroes.update({m : 'm_nonzero'}) + if (branch & 2) != 0: + zeroes.update({rr : 'rr_zero'}) + else: + nonzeroes.update({rr : 'rr_nonzero'}) + rr_alt = s1 + rr_alt = rr_alt * 2 + m_alt = m_alt + u1 + if not degenerate: + rr_alt = rr + m_alt = m + n = m_alt^2 + q = n * t + n = n^2 + if degenerate: + n = m + t = rr_alt^2 + rz = a.Z * m_alt + infinity = False + if (branch & 8) != 0: + if not a_infinity: + infinity = True + zeroes.update({rz : 'r.z=0'}) + else: + nonzeroes.update({rz : 'r.z!=0'}) + rz = rz * 2 + q = -q + t = t + q + rx = t + t = t * 2 + t = t + q + t = t * rr_alt + t = t + n + ry = -t + rx = rx * 4 + ry = ry * 4 + if a_infinity: + rx = b.X + ry = b.Y + rz = 1 + if infinity: + return (constraints(zero={b.Z - 1 : 'b.z=1', b.Infinity : 'b_finite'}), constraints(zero=zeroes, nonzero=nonzeroes), point_at_infinity()) + return (constraints(zero={b.Z - 1 : 'b.z=1', b.Infinity : 'b_finite'}), constraints(zero=zeroes, nonzero=nonzeroes), jacobianpoint(rx, ry, rz)) + +def formula_secp256k1_gej_add_ge_old(branch, a, b): + """libsecp256k1's old secp256k1_gej_add_ge, which fails when ay+by=0 but ax!=bx""" + a_infinity = (branch & 1) != 0 + zero = {} + nonzero = {} + if a_infinity: + nonzero.update({a.Infinity : 'a_infinite'}) + else: + zero.update({a.Infinity : 'a_finite'}) + zz = a.Z^2 + u1 = a.X + u2 = b.X * zz + s1 = a.Y + s2 = b.Y * zz + s2 = s2 * a.Z + z = a.Z + t = u1 + t = t + u2 + m = s1 + m = m + s2 + n = m^2 + q = n * t + n = n^2 + rr = t^2 + t = u1 * u2 + t = -t + rr = rr + t + t = rr^2 + rz = m * z + infinity = False + if (branch & 2) != 0: + if not a_infinity: + infinity = True + else: + return (constraints(zero={b.Z - 1 : 'b.z=1', b.Infinity : 'b_finite'}), constraints(nonzero={z : 'conflict_a'}, zero={z : 'conflict_b'}), point_at_infinity()) + zero.update({rz : 'r.z=0'}) + else: + nonzero.update({rz : 'r.z!=0'}) + rz = rz * (0 if a_infinity else 2) + rx = t + q = -q + rx = rx + q + q = q * 3 + t = t * 2 + t = t + q + t = t * rr + t = t + n + ry = -t + rx = rx * (0 if a_infinity else 4) + ry = ry * (0 if a_infinity else 4) + t = b.X + t = t * (1 if a_infinity else 0) + rx = rx + t + t = b.Y + t = t * (1 if a_infinity else 0) + ry = ry + t + t = (1 if a_infinity else 0) + rz = rz + t + if infinity: + return (constraints(zero={b.Z - 1 : 'b.z=1', b.Infinity : 'b_finite'}), constraints(zero=zero, nonzero=nonzero), point_at_infinity()) + return (constraints(zero={b.Z - 1 : 'b.z=1', b.Infinity : 'b_finite'}), constraints(zero=zero, nonzero=nonzero), jacobianpoint(rx, ry, rz)) + +if __name__ == "__main__": + check_symbolic_jacobian_weierstrass("secp256k1_gej_add_var", 0, 7, 5, formula_secp256k1_gej_add_var) + check_symbolic_jacobian_weierstrass("secp256k1_gej_add_ge_var", 0, 7, 5, formula_secp256k1_gej_add_ge_var) + check_symbolic_jacobian_weierstrass("secp256k1_gej_add_zinv_var", 0, 7, 5, formula_secp256k1_gej_add_zinv_var) + check_symbolic_jacobian_weierstrass("secp256k1_gej_add_ge", 0, 7, 16, formula_secp256k1_gej_add_ge) + check_symbolic_jacobian_weierstrass("secp256k1_gej_add_ge_old [should fail]", 0, 7, 4, formula_secp256k1_gej_add_ge_old) + + if len(sys.argv) >= 2 and sys.argv[1] == "--exhaustive": + check_exhaustive_jacobian_weierstrass("secp256k1_gej_add_var", 0, 7, 5, formula_secp256k1_gej_add_var, 43) + check_exhaustive_jacobian_weierstrass("secp256k1_gej_add_ge_var", 0, 7, 5, formula_secp256k1_gej_add_ge_var, 43) + check_exhaustive_jacobian_weierstrass("secp256k1_gej_add_zinv_var", 0, 7, 5, formula_secp256k1_gej_add_zinv_var, 43) + check_exhaustive_jacobian_weierstrass("secp256k1_gej_add_ge", 0, 7, 16, formula_secp256k1_gej_add_ge, 43) + check_exhaustive_jacobian_weierstrass("secp256k1_gej_add_ge_old [should fail]", 0, 7, 4, formula_secp256k1_gej_add_ge_old, 43) diff --git a/node_modules/secp256k1/src/secp256k1/sage/weierstrass_prover.sage b/node_modules/secp256k1/src/secp256k1/sage/weierstrass_prover.sage new file mode 100644 index 0000000..03ef2ec --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/sage/weierstrass_prover.sage @@ -0,0 +1,264 @@ +# Prover implementation for Weierstrass curves of the form +# y^2 = x^3 + A * x + B, specifically with a = 0 and b = 7, with group laws +# operating on affine and Jacobian coordinates, including the point at infinity +# represented by a 4th variable in coordinates. + +load("group_prover.sage") + + +class affinepoint: + def __init__(self, x, y, infinity=0): + self.x = x + self.y = y + self.infinity = infinity + def __str__(self): + return "affinepoint(x=%s,y=%s,inf=%s)" % (self.x, self.y, self.infinity) + + +class jacobianpoint: + def __init__(self, x, y, z, infinity=0): + self.X = x + self.Y = y + self.Z = z + self.Infinity = infinity + def __str__(self): + return "jacobianpoint(X=%s,Y=%s,Z=%s,inf=%s)" % (self.X, self.Y, self.Z, self.Infinity) + + +def point_at_infinity(): + return jacobianpoint(1, 1, 1, 1) + + +def negate(p): + if p.__class__ == affinepoint: + return affinepoint(p.x, -p.y) + if p.__class__ == jacobianpoint: + return jacobianpoint(p.X, -p.Y, p.Z) + assert(False) + + +def on_weierstrass_curve(A, B, p): + """Return a set of zero-expressions for an affine point to be on the curve""" + return constraints(zero={p.x^3 + A*p.x + B - p.y^2: 'on_curve'}) + + +def tangential_to_weierstrass_curve(A, B, p12, p3): + """Return a set of zero-expressions for ((x12,y12),(x3,y3)) to be a line that is tangential to the curve at (x12,y12)""" + return constraints(zero={ + (p12.y - p3.y) * (p12.y * 2) - (p12.x^2 * 3 + A) * (p12.x - p3.x): 'tangential_to_curve' + }) + + +def colinear(p1, p2, p3): + """Return a set of zero-expressions for ((x1,y1),(x2,y2),(x3,y3)) to be collinear""" + return constraints(zero={ + (p1.y - p2.y) * (p1.x - p3.x) - (p1.y - p3.y) * (p1.x - p2.x): 'colinear_1', + (p2.y - p3.y) * (p2.x - p1.x) - (p2.y - p1.y) * (p2.x - p3.x): 'colinear_2', + (p3.y - p1.y) * (p3.x - p2.x) - (p3.y - p2.y) * (p3.x - p1.x): 'colinear_3' + }) + + +def good_affine_point(p): + return constraints(nonzero={p.x : 'nonzero_x', p.y : 'nonzero_y'}) + + +def good_jacobian_point(p): + return constraints(nonzero={p.X : 'nonzero_X', p.Y : 'nonzero_Y', p.Z^6 : 'nonzero_Z'}) + + +def good_point(p): + return constraints(nonzero={p.Z^6 : 'nonzero_X'}) + + +def finite(p, *affine_fns): + con = good_point(p) + constraints(zero={p.Infinity : 'finite_point'}) + if p.Z != 0: + return con + reduce(lambda a, b: a + b, (f(affinepoint(p.X / p.Z^2, p.Y / p.Z^3)) for f in affine_fns), con) + else: + return con + +def infinite(p): + return constraints(nonzero={p.Infinity : 'infinite_point'}) + + +def law_jacobian_weierstrass_add(A, B, pa, pb, pA, pB, pC): + """Check whether the passed set of coordinates is a valid Jacobian add, given assumptions""" + assumeLaw = (good_affine_point(pa) + + good_affine_point(pb) + + good_jacobian_point(pA) + + good_jacobian_point(pB) + + on_weierstrass_curve(A, B, pa) + + on_weierstrass_curve(A, B, pb) + + finite(pA) + + finite(pB) + + constraints(nonzero={pa.x - pb.x : 'different_x'})) + require = (finite(pC, lambda pc: on_weierstrass_curve(A, B, pc) + + colinear(pa, pb, negate(pc)))) + return (assumeLaw, require) + + +def law_jacobian_weierstrass_double(A, B, pa, pb, pA, pB, pC): + """Check whether the passed set of coordinates is a valid Jacobian doubling, given assumptions""" + assumeLaw = (good_affine_point(pa) + + good_affine_point(pb) + + good_jacobian_point(pA) + + good_jacobian_point(pB) + + on_weierstrass_curve(A, B, pa) + + on_weierstrass_curve(A, B, pb) + + finite(pA) + + finite(pB) + + constraints(zero={pa.x - pb.x : 'equal_x', pa.y - pb.y : 'equal_y'})) + require = (finite(pC, lambda pc: on_weierstrass_curve(A, B, pc) + + tangential_to_weierstrass_curve(A, B, pa, negate(pc)))) + return (assumeLaw, require) + + +def law_jacobian_weierstrass_add_opposites(A, B, pa, pb, pA, pB, pC): + assumeLaw = (good_affine_point(pa) + + good_affine_point(pb) + + good_jacobian_point(pA) + + good_jacobian_point(pB) + + on_weierstrass_curve(A, B, pa) + + on_weierstrass_curve(A, B, pb) + + finite(pA) + + finite(pB) + + constraints(zero={pa.x - pb.x : 'equal_x', pa.y + pb.y : 'opposite_y'})) + require = infinite(pC) + return (assumeLaw, require) + + +def law_jacobian_weierstrass_add_infinite_a(A, B, pa, pb, pA, pB, pC): + assumeLaw = (good_affine_point(pa) + + good_affine_point(pb) + + good_jacobian_point(pA) + + good_jacobian_point(pB) + + on_weierstrass_curve(A, B, pb) + + infinite(pA) + + finite(pB)) + require = finite(pC, lambda pc: constraints(zero={pc.x - pb.x : 'c.x=b.x', pc.y - pb.y : 'c.y=b.y'})) + return (assumeLaw, require) + + +def law_jacobian_weierstrass_add_infinite_b(A, B, pa, pb, pA, pB, pC): + assumeLaw = (good_affine_point(pa) + + good_affine_point(pb) + + good_jacobian_point(pA) + + good_jacobian_point(pB) + + on_weierstrass_curve(A, B, pa) + + infinite(pB) + + finite(pA)) + require = finite(pC, lambda pc: constraints(zero={pc.x - pa.x : 'c.x=a.x', pc.y - pa.y : 'c.y=a.y'})) + return (assumeLaw, require) + + +def law_jacobian_weierstrass_add_infinite_ab(A, B, pa, pb, pA, pB, pC): + assumeLaw = (good_affine_point(pa) + + good_affine_point(pb) + + good_jacobian_point(pA) + + good_jacobian_point(pB) + + infinite(pA) + + infinite(pB)) + require = infinite(pC) + return (assumeLaw, require) + + +laws_jacobian_weierstrass = { + 'add': law_jacobian_weierstrass_add, + 'double': law_jacobian_weierstrass_double, + 'add_opposite': law_jacobian_weierstrass_add_opposites, + 'add_infinite_a': law_jacobian_weierstrass_add_infinite_a, + 'add_infinite_b': law_jacobian_weierstrass_add_infinite_b, + 'add_infinite_ab': law_jacobian_weierstrass_add_infinite_ab +} + + +def check_exhaustive_jacobian_weierstrass(name, A, B, branches, formula, p): + """Verify an implementation of addition of Jacobian points on a Weierstrass curve, by executing and validating the result for every possible addition in a prime field""" + F = Integers(p) + print "Formula %s on Z%i:" % (name, p) + points = [] + for x in xrange(0, p): + for y in xrange(0, p): + point = affinepoint(F(x), F(y)) + r, e = concrete_verify(on_weierstrass_curve(A, B, point)) + if r: + points.append(point) + + for za in xrange(1, p): + for zb in xrange(1, p): + for pa in points: + for pb in points: + for ia in xrange(2): + for ib in xrange(2): + pA = jacobianpoint(pa.x * F(za)^2, pa.y * F(za)^3, F(za), ia) + pB = jacobianpoint(pb.x * F(zb)^2, pb.y * F(zb)^3, F(zb), ib) + for branch in xrange(0, branches): + assumeAssert, assumeBranch, pC = formula(branch, pA, pB) + pC.X = F(pC.X) + pC.Y = F(pC.Y) + pC.Z = F(pC.Z) + pC.Infinity = F(pC.Infinity) + r, e = concrete_verify(assumeAssert + assumeBranch) + if r: + match = False + for key in laws_jacobian_weierstrass: + assumeLaw, require = laws_jacobian_weierstrass[key](A, B, pa, pb, pA, pB, pC) + r, e = concrete_verify(assumeLaw) + if r: + if match: + print " multiple branches for (%s,%s,%s,%s) + (%s,%s,%s,%s)" % (pA.X, pA.Y, pA.Z, pA.Infinity, pB.X, pB.Y, pB.Z, pB.Infinity) + else: + match = True + r, e = concrete_verify(require) + if not r: + print " failure in branch %i for (%s,%s,%s,%s) + (%s,%s,%s,%s) = (%s,%s,%s,%s): %s" % (branch, pA.X, pA.Y, pA.Z, pA.Infinity, pB.X, pB.Y, pB.Z, pB.Infinity, pC.X, pC.Y, pC.Z, pC.Infinity, e) + print + + +def check_symbolic_function(R, assumeAssert, assumeBranch, f, A, B, pa, pb, pA, pB, pC): + assumeLaw, require = f(A, B, pa, pb, pA, pB, pC) + return check_symbolic(R, assumeLaw, assumeAssert, assumeBranch, require) + +def check_symbolic_jacobian_weierstrass(name, A, B, branches, formula): + """Verify an implementation of addition of Jacobian points on a Weierstrass curve symbolically""" + R. = PolynomialRing(QQ,8,order='invlex') + lift = lambda x: fastfrac(R,x) + ax = lift(ax) + ay = lift(ay) + Az = lift(Az) + bx = lift(bx) + by = lift(by) + Bz = lift(Bz) + Ai = lift(Ai) + Bi = lift(Bi) + + pa = affinepoint(ax, ay, Ai) + pb = affinepoint(bx, by, Bi) + pA = jacobianpoint(ax * Az^2, ay * Az^3, Az, Ai) + pB = jacobianpoint(bx * Bz^2, by * Bz^3, Bz, Bi) + + res = {} + + for key in laws_jacobian_weierstrass: + res[key] = [] + + print ("Formula " + name + ":") + count = 0 + for branch in xrange(branches): + assumeFormula, assumeBranch, pC = formula(branch, pA, pB) + pC.X = lift(pC.X) + pC.Y = lift(pC.Y) + pC.Z = lift(pC.Z) + pC.Infinity = lift(pC.Infinity) + + for key in laws_jacobian_weierstrass: + res[key].append((check_symbolic_function(R, assumeFormula, assumeBranch, laws_jacobian_weierstrass[key], A, B, pa, pb, pA, pB, pC), branch)) + + for key in res: + print " %s:" % key + val = res[key] + for x in val: + if x[0] is not None: + print " branch %i: %s" % (x[1], x[0]) + + print diff --git a/node_modules/secp256k1/src/secp256k1/src/asm/field_10x26_arm.s b/node_modules/secp256k1/src/secp256k1/src/asm/field_10x26_arm.s new file mode 100644 index 0000000..9a5bd06 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/asm/field_10x26_arm.s @@ -0,0 +1,913 @@ +@ vim: set tabstop=8 softtabstop=8 shiftwidth=8 noexpandtab syntax=armasm: +/********************************************************************** + * Copyright (c) 2014 Wladimir J. van der Laan * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ +/* +ARM implementation of field_10x26 inner loops. + +Note: + +- To avoid unnecessary loads and make use of available registers, two + 'passes' have every time been interleaved, with the odd passes accumulating c' and d' + which will be added to c and d respectively in the even passes + +*/ + + .syntax unified + @ eabi attributes - see readelf -A + .eabi_attribute 24, 1 @ Tag_ABI_align_needed = 8-byte + .eabi_attribute 25, 1 @ Tag_ABI_align_preserved = 8-byte, except leaf SP + .text + + @ Field constants + .set field_R0, 0x3d10 + .set field_R1, 0x400 + .set field_not_M, 0xfc000000 @ ~M = ~0x3ffffff + + .align 2 + .global secp256k1_fe_mul_inner + .type secp256k1_fe_mul_inner, %function + @ Arguments: + @ r0 r Restrict: can overlap with a, not with b + @ r1 a + @ r2 b + @ Stack (total 4+10*4 = 44) + @ sp + #0 saved 'r' pointer + @ sp + #4 + 4*X t0,t1,t2,t3,t4,t5,t6,t7,u8,t9 +secp256k1_fe_mul_inner: + stmfd sp!, {r4, r5, r6, r7, r8, r9, r10, r11, r14} + sub sp, sp, #48 @ frame=44 + alignment + str r0, [sp, #0] @ save result address, we need it only at the end + + /****************************************** + * Main computation code. + ****************************************** + + Allocation: + r0,r14,r7,r8 scratch + r1 a (pointer) + r2 b (pointer) + r3:r4 c + r5:r6 d + r11:r12 c' + r9:r10 d' + + Note: do not write to r[] here, it may overlap with a[] + */ + + /* A - interleaved with B */ + ldr r7, [r1, #0*4] @ a[0] + ldr r8, [r2, #9*4] @ b[9] + ldr r0, [r1, #1*4] @ a[1] + umull r5, r6, r7, r8 @ d = a[0] * b[9] + ldr r14, [r2, #8*4] @ b[8] + umull r9, r10, r0, r8 @ d' = a[1] * b[9] + ldr r7, [r1, #2*4] @ a[2] + umlal r5, r6, r0, r14 @ d += a[1] * b[8] + ldr r8, [r2, #7*4] @ b[7] + umlal r9, r10, r7, r14 @ d' += a[2] * b[8] + ldr r0, [r1, #3*4] @ a[3] + umlal r5, r6, r7, r8 @ d += a[2] * b[7] + ldr r14, [r2, #6*4] @ b[6] + umlal r9, r10, r0, r8 @ d' += a[3] * b[7] + ldr r7, [r1, #4*4] @ a[4] + umlal r5, r6, r0, r14 @ d += a[3] * b[6] + ldr r8, [r2, #5*4] @ b[5] + umlal r9, r10, r7, r14 @ d' += a[4] * b[6] + ldr r0, [r1, #5*4] @ a[5] + umlal r5, r6, r7, r8 @ d += a[4] * b[5] + ldr r14, [r2, #4*4] @ b[4] + umlal r9, r10, r0, r8 @ d' += a[5] * b[5] + ldr r7, [r1, #6*4] @ a[6] + umlal r5, r6, r0, r14 @ d += a[5] * b[4] + ldr r8, [r2, #3*4] @ b[3] + umlal r9, r10, r7, r14 @ d' += a[6] * b[4] + ldr r0, [r1, #7*4] @ a[7] + umlal r5, r6, r7, r8 @ d += a[6] * b[3] + ldr r14, [r2, #2*4] @ b[2] + umlal r9, r10, r0, r8 @ d' += a[7] * b[3] + ldr r7, [r1, #8*4] @ a[8] + umlal r5, r6, r0, r14 @ d += a[7] * b[2] + ldr r8, [r2, #1*4] @ b[1] + umlal r9, r10, r7, r14 @ d' += a[8] * b[2] + ldr r0, [r1, #9*4] @ a[9] + umlal r5, r6, r7, r8 @ d += a[8] * b[1] + ldr r14, [r2, #0*4] @ b[0] + umlal r9, r10, r0, r8 @ d' += a[9] * b[1] + ldr r7, [r1, #0*4] @ a[0] + umlal r5, r6, r0, r14 @ d += a[9] * b[0] + @ r7,r14 used in B + + bic r0, r5, field_not_M @ t9 = d & M + str r0, [sp, #4 + 4*9] + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + + /* B */ + umull r3, r4, r7, r14 @ c = a[0] * b[0] + adds r5, r5, r9 @ d += d' + adc r6, r6, r10 + + bic r0, r5, field_not_M @ u0 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u0 * R0 + umlal r3, r4, r0, r14 + + bic r14, r3, field_not_M @ t0 = c & M + str r14, [sp, #4 + 0*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u0 * R1 + umlal r3, r4, r0, r14 + + /* C - interleaved with D */ + ldr r7, [r1, #0*4] @ a[0] + ldr r8, [r2, #2*4] @ b[2] + ldr r14, [r2, #1*4] @ b[1] + umull r11, r12, r7, r8 @ c' = a[0] * b[2] + ldr r0, [r1, #1*4] @ a[1] + umlal r3, r4, r7, r14 @ c += a[0] * b[1] + ldr r8, [r2, #0*4] @ b[0] + umlal r11, r12, r0, r14 @ c' += a[1] * b[1] + ldr r7, [r1, #2*4] @ a[2] + umlal r3, r4, r0, r8 @ c += a[1] * b[0] + ldr r14, [r2, #9*4] @ b[9] + umlal r11, r12, r7, r8 @ c' += a[2] * b[0] + ldr r0, [r1, #3*4] @ a[3] + umlal r5, r6, r7, r14 @ d += a[2] * b[9] + ldr r8, [r2, #8*4] @ b[8] + umull r9, r10, r0, r14 @ d' = a[3] * b[9] + ldr r7, [r1, #4*4] @ a[4] + umlal r5, r6, r0, r8 @ d += a[3] * b[8] + ldr r14, [r2, #7*4] @ b[7] + umlal r9, r10, r7, r8 @ d' += a[4] * b[8] + ldr r0, [r1, #5*4] @ a[5] + umlal r5, r6, r7, r14 @ d += a[4] * b[7] + ldr r8, [r2, #6*4] @ b[6] + umlal r9, r10, r0, r14 @ d' += a[5] * b[7] + ldr r7, [r1, #6*4] @ a[6] + umlal r5, r6, r0, r8 @ d += a[5] * b[6] + ldr r14, [r2, #5*4] @ b[5] + umlal r9, r10, r7, r8 @ d' += a[6] * b[6] + ldr r0, [r1, #7*4] @ a[7] + umlal r5, r6, r7, r14 @ d += a[6] * b[5] + ldr r8, [r2, #4*4] @ b[4] + umlal r9, r10, r0, r14 @ d' += a[7] * b[5] + ldr r7, [r1, #8*4] @ a[8] + umlal r5, r6, r0, r8 @ d += a[7] * b[4] + ldr r14, [r2, #3*4] @ b[3] + umlal r9, r10, r7, r8 @ d' += a[8] * b[4] + ldr r0, [r1, #9*4] @ a[9] + umlal r5, r6, r7, r14 @ d += a[8] * b[3] + ldr r8, [r2, #2*4] @ b[2] + umlal r9, r10, r0, r14 @ d' += a[9] * b[3] + umlal r5, r6, r0, r8 @ d += a[9] * b[2] + + bic r0, r5, field_not_M @ u1 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u1 * R0 + umlal r3, r4, r0, r14 + + bic r14, r3, field_not_M @ t1 = c & M + str r14, [sp, #4 + 1*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u1 * R1 + umlal r3, r4, r0, r14 + + /* D */ + adds r3, r3, r11 @ c += c' + adc r4, r4, r12 + adds r5, r5, r9 @ d += d' + adc r6, r6, r10 + + bic r0, r5, field_not_M @ u2 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u2 * R0 + umlal r3, r4, r0, r14 + + bic r14, r3, field_not_M @ t2 = c & M + str r14, [sp, #4 + 2*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u2 * R1 + umlal r3, r4, r0, r14 + + /* E - interleaved with F */ + ldr r7, [r1, #0*4] @ a[0] + ldr r8, [r2, #4*4] @ b[4] + umull r11, r12, r7, r8 @ c' = a[0] * b[4] + ldr r8, [r2, #3*4] @ b[3] + umlal r3, r4, r7, r8 @ c += a[0] * b[3] + ldr r7, [r1, #1*4] @ a[1] + umlal r11, r12, r7, r8 @ c' += a[1] * b[3] + ldr r8, [r2, #2*4] @ b[2] + umlal r3, r4, r7, r8 @ c += a[1] * b[2] + ldr r7, [r1, #2*4] @ a[2] + umlal r11, r12, r7, r8 @ c' += a[2] * b[2] + ldr r8, [r2, #1*4] @ b[1] + umlal r3, r4, r7, r8 @ c += a[2] * b[1] + ldr r7, [r1, #3*4] @ a[3] + umlal r11, r12, r7, r8 @ c' += a[3] * b[1] + ldr r8, [r2, #0*4] @ b[0] + umlal r3, r4, r7, r8 @ c += a[3] * b[0] + ldr r7, [r1, #4*4] @ a[4] + umlal r11, r12, r7, r8 @ c' += a[4] * b[0] + ldr r8, [r2, #9*4] @ b[9] + umlal r5, r6, r7, r8 @ d += a[4] * b[9] + ldr r7, [r1, #5*4] @ a[5] + umull r9, r10, r7, r8 @ d' = a[5] * b[9] + ldr r8, [r2, #8*4] @ b[8] + umlal r5, r6, r7, r8 @ d += a[5] * b[8] + ldr r7, [r1, #6*4] @ a[6] + umlal r9, r10, r7, r8 @ d' += a[6] * b[8] + ldr r8, [r2, #7*4] @ b[7] + umlal r5, r6, r7, r8 @ d += a[6] * b[7] + ldr r7, [r1, #7*4] @ a[7] + umlal r9, r10, r7, r8 @ d' += a[7] * b[7] + ldr r8, [r2, #6*4] @ b[6] + umlal r5, r6, r7, r8 @ d += a[7] * b[6] + ldr r7, [r1, #8*4] @ a[8] + umlal r9, r10, r7, r8 @ d' += a[8] * b[6] + ldr r8, [r2, #5*4] @ b[5] + umlal r5, r6, r7, r8 @ d += a[8] * b[5] + ldr r7, [r1, #9*4] @ a[9] + umlal r9, r10, r7, r8 @ d' += a[9] * b[5] + ldr r8, [r2, #4*4] @ b[4] + umlal r5, r6, r7, r8 @ d += a[9] * b[4] + + bic r0, r5, field_not_M @ u3 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u3 * R0 + umlal r3, r4, r0, r14 + + bic r14, r3, field_not_M @ t3 = c & M + str r14, [sp, #4 + 3*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u3 * R1 + umlal r3, r4, r0, r14 + + /* F */ + adds r3, r3, r11 @ c += c' + adc r4, r4, r12 + adds r5, r5, r9 @ d += d' + adc r6, r6, r10 + + bic r0, r5, field_not_M @ u4 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u4 * R0 + umlal r3, r4, r0, r14 + + bic r14, r3, field_not_M @ t4 = c & M + str r14, [sp, #4 + 4*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u4 * R1 + umlal r3, r4, r0, r14 + + /* G - interleaved with H */ + ldr r7, [r1, #0*4] @ a[0] + ldr r8, [r2, #6*4] @ b[6] + ldr r14, [r2, #5*4] @ b[5] + umull r11, r12, r7, r8 @ c' = a[0] * b[6] + ldr r0, [r1, #1*4] @ a[1] + umlal r3, r4, r7, r14 @ c += a[0] * b[5] + ldr r8, [r2, #4*4] @ b[4] + umlal r11, r12, r0, r14 @ c' += a[1] * b[5] + ldr r7, [r1, #2*4] @ a[2] + umlal r3, r4, r0, r8 @ c += a[1] * b[4] + ldr r14, [r2, #3*4] @ b[3] + umlal r11, r12, r7, r8 @ c' += a[2] * b[4] + ldr r0, [r1, #3*4] @ a[3] + umlal r3, r4, r7, r14 @ c += a[2] * b[3] + ldr r8, [r2, #2*4] @ b[2] + umlal r11, r12, r0, r14 @ c' += a[3] * b[3] + ldr r7, [r1, #4*4] @ a[4] + umlal r3, r4, r0, r8 @ c += a[3] * b[2] + ldr r14, [r2, #1*4] @ b[1] + umlal r11, r12, r7, r8 @ c' += a[4] * b[2] + ldr r0, [r1, #5*4] @ a[5] + umlal r3, r4, r7, r14 @ c += a[4] * b[1] + ldr r8, [r2, #0*4] @ b[0] + umlal r11, r12, r0, r14 @ c' += a[5] * b[1] + ldr r7, [r1, #6*4] @ a[6] + umlal r3, r4, r0, r8 @ c += a[5] * b[0] + ldr r14, [r2, #9*4] @ b[9] + umlal r11, r12, r7, r8 @ c' += a[6] * b[0] + ldr r0, [r1, #7*4] @ a[7] + umlal r5, r6, r7, r14 @ d += a[6] * b[9] + ldr r8, [r2, #8*4] @ b[8] + umull r9, r10, r0, r14 @ d' = a[7] * b[9] + ldr r7, [r1, #8*4] @ a[8] + umlal r5, r6, r0, r8 @ d += a[7] * b[8] + ldr r14, [r2, #7*4] @ b[7] + umlal r9, r10, r7, r8 @ d' += a[8] * b[8] + ldr r0, [r1, #9*4] @ a[9] + umlal r5, r6, r7, r14 @ d += a[8] * b[7] + ldr r8, [r2, #6*4] @ b[6] + umlal r9, r10, r0, r14 @ d' += a[9] * b[7] + umlal r5, r6, r0, r8 @ d += a[9] * b[6] + + bic r0, r5, field_not_M @ u5 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u5 * R0 + umlal r3, r4, r0, r14 + + bic r14, r3, field_not_M @ t5 = c & M + str r14, [sp, #4 + 5*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u5 * R1 + umlal r3, r4, r0, r14 + + /* H */ + adds r3, r3, r11 @ c += c' + adc r4, r4, r12 + adds r5, r5, r9 @ d += d' + adc r6, r6, r10 + + bic r0, r5, field_not_M @ u6 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u6 * R0 + umlal r3, r4, r0, r14 + + bic r14, r3, field_not_M @ t6 = c & M + str r14, [sp, #4 + 6*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u6 * R1 + umlal r3, r4, r0, r14 + + /* I - interleaved with J */ + ldr r8, [r2, #8*4] @ b[8] + ldr r7, [r1, #0*4] @ a[0] + ldr r14, [r2, #7*4] @ b[7] + umull r11, r12, r7, r8 @ c' = a[0] * b[8] + ldr r0, [r1, #1*4] @ a[1] + umlal r3, r4, r7, r14 @ c += a[0] * b[7] + ldr r8, [r2, #6*4] @ b[6] + umlal r11, r12, r0, r14 @ c' += a[1] * b[7] + ldr r7, [r1, #2*4] @ a[2] + umlal r3, r4, r0, r8 @ c += a[1] * b[6] + ldr r14, [r2, #5*4] @ b[5] + umlal r11, r12, r7, r8 @ c' += a[2] * b[6] + ldr r0, [r1, #3*4] @ a[3] + umlal r3, r4, r7, r14 @ c += a[2] * b[5] + ldr r8, [r2, #4*4] @ b[4] + umlal r11, r12, r0, r14 @ c' += a[3] * b[5] + ldr r7, [r1, #4*4] @ a[4] + umlal r3, r4, r0, r8 @ c += a[3] * b[4] + ldr r14, [r2, #3*4] @ b[3] + umlal r11, r12, r7, r8 @ c' += a[4] * b[4] + ldr r0, [r1, #5*4] @ a[5] + umlal r3, r4, r7, r14 @ c += a[4] * b[3] + ldr r8, [r2, #2*4] @ b[2] + umlal r11, r12, r0, r14 @ c' += a[5] * b[3] + ldr r7, [r1, #6*4] @ a[6] + umlal r3, r4, r0, r8 @ c += a[5] * b[2] + ldr r14, [r2, #1*4] @ b[1] + umlal r11, r12, r7, r8 @ c' += a[6] * b[2] + ldr r0, [r1, #7*4] @ a[7] + umlal r3, r4, r7, r14 @ c += a[6] * b[1] + ldr r8, [r2, #0*4] @ b[0] + umlal r11, r12, r0, r14 @ c' += a[7] * b[1] + ldr r7, [r1, #8*4] @ a[8] + umlal r3, r4, r0, r8 @ c += a[7] * b[0] + ldr r14, [r2, #9*4] @ b[9] + umlal r11, r12, r7, r8 @ c' += a[8] * b[0] + ldr r0, [r1, #9*4] @ a[9] + umlal r5, r6, r7, r14 @ d += a[8] * b[9] + ldr r8, [r2, #8*4] @ b[8] + umull r9, r10, r0, r14 @ d' = a[9] * b[9] + umlal r5, r6, r0, r8 @ d += a[9] * b[8] + + bic r0, r5, field_not_M @ u7 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u7 * R0 + umlal r3, r4, r0, r14 + + bic r14, r3, field_not_M @ t7 = c & M + str r14, [sp, #4 + 7*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u7 * R1 + umlal r3, r4, r0, r14 + + /* J */ + adds r3, r3, r11 @ c += c' + adc r4, r4, r12 + adds r5, r5, r9 @ d += d' + adc r6, r6, r10 + + bic r0, r5, field_not_M @ u8 = d & M + str r0, [sp, #4 + 8*4] + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u8 * R0 + umlal r3, r4, r0, r14 + + /****************************************** + * compute and write back result + ****************************************** + Allocation: + r0 r + r3:r4 c + r5:r6 d + r7 t0 + r8 t1 + r9 t2 + r11 u8 + r12 t9 + r1,r2,r10,r14 scratch + + Note: do not read from a[] after here, it may overlap with r[] + */ + ldr r0, [sp, #0] + add r1, sp, #4 + 3*4 @ r[3..7] = t3..7, r11=u8, r12=t9 + ldmia r1, {r2,r7,r8,r9,r10,r11,r12} + add r1, r0, #3*4 + stmia r1, {r2,r7,r8,r9,r10} + + bic r2, r3, field_not_M @ r[8] = c & M + str r2, [r0, #8*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u8 * R1 + umlal r3, r4, r11, r14 + movw r14, field_R0 @ c += d * R0 + umlal r3, r4, r5, r14 + adds r3, r3, r12 @ c += t9 + adc r4, r4, #0 + + add r1, sp, #4 + 0*4 @ r7,r8,r9 = t0,t1,t2 + ldmia r1, {r7,r8,r9} + + ubfx r2, r3, #0, #22 @ r[9] = c & (M >> 4) + str r2, [r0, #9*4] + mov r3, r3, lsr #22 @ c >>= 22 + orr r3, r3, r4, asl #10 + mov r4, r4, lsr #22 + movw r14, field_R1 << 4 @ c += d * (R1 << 4) + umlal r3, r4, r5, r14 + + movw r14, field_R0 >> 4 @ d = c * (R0 >> 4) + t0 (64x64 multiply+add) + umull r5, r6, r3, r14 @ d = c.lo * (R0 >> 4) + adds r5, r5, r7 @ d.lo += t0 + mla r6, r14, r4, r6 @ d.hi += c.hi * (R0 >> 4) + adc r6, r6, 0 @ d.hi += carry + + bic r2, r5, field_not_M @ r[0] = d & M + str r2, [r0, #0*4] + + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + + movw r14, field_R1 >> 4 @ d += c * (R1 >> 4) + t1 (64x64 multiply+add) + umull r1, r2, r3, r14 @ tmp = c.lo * (R1 >> 4) + adds r5, r5, r8 @ d.lo += t1 + adc r6, r6, #0 @ d.hi += carry + adds r5, r5, r1 @ d.lo += tmp.lo + mla r2, r14, r4, r2 @ tmp.hi += c.hi * (R1 >> 4) + adc r6, r6, r2 @ d.hi += carry + tmp.hi + + bic r2, r5, field_not_M @ r[1] = d & M + str r2, [r0, #1*4] + mov r5, r5, lsr #26 @ d >>= 26 (ignore hi) + orr r5, r5, r6, asl #6 + + add r5, r5, r9 @ d += t2 + str r5, [r0, #2*4] @ r[2] = d + + add sp, sp, #48 + ldmfd sp!, {r4, r5, r6, r7, r8, r9, r10, r11, pc} + .size secp256k1_fe_mul_inner, .-secp256k1_fe_mul_inner + + .align 2 + .global secp256k1_fe_sqr_inner + .type secp256k1_fe_sqr_inner, %function + @ Arguments: + @ r0 r Can overlap with a + @ r1 a + @ Stack (total 4+10*4 = 44) + @ sp + #0 saved 'r' pointer + @ sp + #4 + 4*X t0,t1,t2,t3,t4,t5,t6,t7,u8,t9 +secp256k1_fe_sqr_inner: + stmfd sp!, {r4, r5, r6, r7, r8, r9, r10, r11, r14} + sub sp, sp, #48 @ frame=44 + alignment + str r0, [sp, #0] @ save result address, we need it only at the end + /****************************************** + * Main computation code. + ****************************************** + + Allocation: + r0,r14,r2,r7,r8 scratch + r1 a (pointer) + r3:r4 c + r5:r6 d + r11:r12 c' + r9:r10 d' + + Note: do not write to r[] here, it may overlap with a[] + */ + /* A interleaved with B */ + ldr r0, [r1, #1*4] @ a[1]*2 + ldr r7, [r1, #0*4] @ a[0] + mov r0, r0, asl #1 + ldr r14, [r1, #9*4] @ a[9] + umull r3, r4, r7, r7 @ c = a[0] * a[0] + ldr r8, [r1, #8*4] @ a[8] + mov r7, r7, asl #1 + umull r5, r6, r7, r14 @ d = a[0]*2 * a[9] + ldr r7, [r1, #2*4] @ a[2]*2 + umull r9, r10, r0, r14 @ d' = a[1]*2 * a[9] + ldr r14, [r1, #7*4] @ a[7] + umlal r5, r6, r0, r8 @ d += a[1]*2 * a[8] + mov r7, r7, asl #1 + ldr r0, [r1, #3*4] @ a[3]*2 + umlal r9, r10, r7, r8 @ d' += a[2]*2 * a[8] + ldr r8, [r1, #6*4] @ a[6] + umlal r5, r6, r7, r14 @ d += a[2]*2 * a[7] + mov r0, r0, asl #1 + ldr r7, [r1, #4*4] @ a[4]*2 + umlal r9, r10, r0, r14 @ d' += a[3]*2 * a[7] + ldr r14, [r1, #5*4] @ a[5] + mov r7, r7, asl #1 + umlal r5, r6, r0, r8 @ d += a[3]*2 * a[6] + umlal r9, r10, r7, r8 @ d' += a[4]*2 * a[6] + umlal r5, r6, r7, r14 @ d += a[4]*2 * a[5] + umlal r9, r10, r14, r14 @ d' += a[5] * a[5] + + bic r0, r5, field_not_M @ t9 = d & M + str r0, [sp, #4 + 9*4] + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + + /* B */ + adds r5, r5, r9 @ d += d' + adc r6, r6, r10 + + bic r0, r5, field_not_M @ u0 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u0 * R0 + umlal r3, r4, r0, r14 + bic r14, r3, field_not_M @ t0 = c & M + str r14, [sp, #4 + 0*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u0 * R1 + umlal r3, r4, r0, r14 + + /* C interleaved with D */ + ldr r0, [r1, #0*4] @ a[0]*2 + ldr r14, [r1, #1*4] @ a[1] + mov r0, r0, asl #1 + ldr r8, [r1, #2*4] @ a[2] + umlal r3, r4, r0, r14 @ c += a[0]*2 * a[1] + mov r7, r8, asl #1 @ a[2]*2 + umull r11, r12, r14, r14 @ c' = a[1] * a[1] + ldr r14, [r1, #9*4] @ a[9] + umlal r11, r12, r0, r8 @ c' += a[0]*2 * a[2] + ldr r0, [r1, #3*4] @ a[3]*2 + ldr r8, [r1, #8*4] @ a[8] + umlal r5, r6, r7, r14 @ d += a[2]*2 * a[9] + mov r0, r0, asl #1 + ldr r7, [r1, #4*4] @ a[4]*2 + umull r9, r10, r0, r14 @ d' = a[3]*2 * a[9] + ldr r14, [r1, #7*4] @ a[7] + umlal r5, r6, r0, r8 @ d += a[3]*2 * a[8] + mov r7, r7, asl #1 + ldr r0, [r1, #5*4] @ a[5]*2 + umlal r9, r10, r7, r8 @ d' += a[4]*2 * a[8] + ldr r8, [r1, #6*4] @ a[6] + mov r0, r0, asl #1 + umlal r5, r6, r7, r14 @ d += a[4]*2 * a[7] + umlal r9, r10, r0, r14 @ d' += a[5]*2 * a[7] + umlal r5, r6, r0, r8 @ d += a[5]*2 * a[6] + umlal r9, r10, r8, r8 @ d' += a[6] * a[6] + + bic r0, r5, field_not_M @ u1 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u1 * R0 + umlal r3, r4, r0, r14 + bic r14, r3, field_not_M @ t1 = c & M + str r14, [sp, #4 + 1*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u1 * R1 + umlal r3, r4, r0, r14 + + /* D */ + adds r3, r3, r11 @ c += c' + adc r4, r4, r12 + adds r5, r5, r9 @ d += d' + adc r6, r6, r10 + + bic r0, r5, field_not_M @ u2 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u2 * R0 + umlal r3, r4, r0, r14 + bic r14, r3, field_not_M @ t2 = c & M + str r14, [sp, #4 + 2*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u2 * R1 + umlal r3, r4, r0, r14 + + /* E interleaved with F */ + ldr r7, [r1, #0*4] @ a[0]*2 + ldr r0, [r1, #1*4] @ a[1]*2 + ldr r14, [r1, #2*4] @ a[2] + mov r7, r7, asl #1 + ldr r8, [r1, #3*4] @ a[3] + ldr r2, [r1, #4*4] + umlal r3, r4, r7, r8 @ c += a[0]*2 * a[3] + mov r0, r0, asl #1 + umull r11, r12, r7, r2 @ c' = a[0]*2 * a[4] + mov r2, r2, asl #1 @ a[4]*2 + umlal r11, r12, r0, r8 @ c' += a[1]*2 * a[3] + ldr r8, [r1, #9*4] @ a[9] + umlal r3, r4, r0, r14 @ c += a[1]*2 * a[2] + ldr r0, [r1, #5*4] @ a[5]*2 + umlal r11, r12, r14, r14 @ c' += a[2] * a[2] + ldr r14, [r1, #8*4] @ a[8] + mov r0, r0, asl #1 + umlal r5, r6, r2, r8 @ d += a[4]*2 * a[9] + ldr r7, [r1, #6*4] @ a[6]*2 + umull r9, r10, r0, r8 @ d' = a[5]*2 * a[9] + mov r7, r7, asl #1 + ldr r8, [r1, #7*4] @ a[7] + umlal r5, r6, r0, r14 @ d += a[5]*2 * a[8] + umlal r9, r10, r7, r14 @ d' += a[6]*2 * a[8] + umlal r5, r6, r7, r8 @ d += a[6]*2 * a[7] + umlal r9, r10, r8, r8 @ d' += a[7] * a[7] + + bic r0, r5, field_not_M @ u3 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u3 * R0 + umlal r3, r4, r0, r14 + bic r14, r3, field_not_M @ t3 = c & M + str r14, [sp, #4 + 3*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u3 * R1 + umlal r3, r4, r0, r14 + + /* F */ + adds r3, r3, r11 @ c += c' + adc r4, r4, r12 + adds r5, r5, r9 @ d += d' + adc r6, r6, r10 + + bic r0, r5, field_not_M @ u4 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u4 * R0 + umlal r3, r4, r0, r14 + bic r14, r3, field_not_M @ t4 = c & M + str r14, [sp, #4 + 4*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u4 * R1 + umlal r3, r4, r0, r14 + + /* G interleaved with H */ + ldr r7, [r1, #0*4] @ a[0]*2 + ldr r0, [r1, #1*4] @ a[1]*2 + mov r7, r7, asl #1 + ldr r8, [r1, #5*4] @ a[5] + ldr r2, [r1, #6*4] @ a[6] + umlal r3, r4, r7, r8 @ c += a[0]*2 * a[5] + ldr r14, [r1, #4*4] @ a[4] + mov r0, r0, asl #1 + umull r11, r12, r7, r2 @ c' = a[0]*2 * a[6] + ldr r7, [r1, #2*4] @ a[2]*2 + umlal r11, r12, r0, r8 @ c' += a[1]*2 * a[5] + mov r7, r7, asl #1 + ldr r8, [r1, #3*4] @ a[3] + umlal r3, r4, r0, r14 @ c += a[1]*2 * a[4] + mov r0, r2, asl #1 @ a[6]*2 + umlal r11, r12, r7, r14 @ c' += a[2]*2 * a[4] + ldr r14, [r1, #9*4] @ a[9] + umlal r3, r4, r7, r8 @ c += a[2]*2 * a[3] + ldr r7, [r1, #7*4] @ a[7]*2 + umlal r11, r12, r8, r8 @ c' += a[3] * a[3] + mov r7, r7, asl #1 + ldr r8, [r1, #8*4] @ a[8] + umlal r5, r6, r0, r14 @ d += a[6]*2 * a[9] + umull r9, r10, r7, r14 @ d' = a[7]*2 * a[9] + umlal r5, r6, r7, r8 @ d += a[7]*2 * a[8] + umlal r9, r10, r8, r8 @ d' += a[8] * a[8] + + bic r0, r5, field_not_M @ u5 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u5 * R0 + umlal r3, r4, r0, r14 + bic r14, r3, field_not_M @ t5 = c & M + str r14, [sp, #4 + 5*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u5 * R1 + umlal r3, r4, r0, r14 + + /* H */ + adds r3, r3, r11 @ c += c' + adc r4, r4, r12 + adds r5, r5, r9 @ d += d' + adc r6, r6, r10 + + bic r0, r5, field_not_M @ u6 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u6 * R0 + umlal r3, r4, r0, r14 + bic r14, r3, field_not_M @ t6 = c & M + str r14, [sp, #4 + 6*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u6 * R1 + umlal r3, r4, r0, r14 + + /* I interleaved with J */ + ldr r7, [r1, #0*4] @ a[0]*2 + ldr r0, [r1, #1*4] @ a[1]*2 + mov r7, r7, asl #1 + ldr r8, [r1, #7*4] @ a[7] + ldr r2, [r1, #8*4] @ a[8] + umlal r3, r4, r7, r8 @ c += a[0]*2 * a[7] + ldr r14, [r1, #6*4] @ a[6] + mov r0, r0, asl #1 + umull r11, r12, r7, r2 @ c' = a[0]*2 * a[8] + ldr r7, [r1, #2*4] @ a[2]*2 + umlal r11, r12, r0, r8 @ c' += a[1]*2 * a[7] + ldr r8, [r1, #5*4] @ a[5] + umlal r3, r4, r0, r14 @ c += a[1]*2 * a[6] + ldr r0, [r1, #3*4] @ a[3]*2 + mov r7, r7, asl #1 + umlal r11, r12, r7, r14 @ c' += a[2]*2 * a[6] + ldr r14, [r1, #4*4] @ a[4] + mov r0, r0, asl #1 + umlal r3, r4, r7, r8 @ c += a[2]*2 * a[5] + mov r2, r2, asl #1 @ a[8]*2 + umlal r11, r12, r0, r8 @ c' += a[3]*2 * a[5] + umlal r3, r4, r0, r14 @ c += a[3]*2 * a[4] + umlal r11, r12, r14, r14 @ c' += a[4] * a[4] + ldr r8, [r1, #9*4] @ a[9] + umlal r5, r6, r2, r8 @ d += a[8]*2 * a[9] + @ r8 will be used in J + + bic r0, r5, field_not_M @ u7 = d & M + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u7 * R0 + umlal r3, r4, r0, r14 + bic r14, r3, field_not_M @ t7 = c & M + str r14, [sp, #4 + 7*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u7 * R1 + umlal r3, r4, r0, r14 + + /* J */ + adds r3, r3, r11 @ c += c' + adc r4, r4, r12 + umlal r5, r6, r8, r8 @ d += a[9] * a[9] + + bic r0, r5, field_not_M @ u8 = d & M + str r0, [sp, #4 + 8*4] + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + movw r14, field_R0 @ c += u8 * R0 + umlal r3, r4, r0, r14 + + /****************************************** + * compute and write back result + ****************************************** + Allocation: + r0 r + r3:r4 c + r5:r6 d + r7 t0 + r8 t1 + r9 t2 + r11 u8 + r12 t9 + r1,r2,r10,r14 scratch + + Note: do not read from a[] after here, it may overlap with r[] + */ + ldr r0, [sp, #0] + add r1, sp, #4 + 3*4 @ r[3..7] = t3..7, r11=u8, r12=t9 + ldmia r1, {r2,r7,r8,r9,r10,r11,r12} + add r1, r0, #3*4 + stmia r1, {r2,r7,r8,r9,r10} + + bic r2, r3, field_not_M @ r[8] = c & M + str r2, [r0, #8*4] + mov r3, r3, lsr #26 @ c >>= 26 + orr r3, r3, r4, asl #6 + mov r4, r4, lsr #26 + mov r14, field_R1 @ c += u8 * R1 + umlal r3, r4, r11, r14 + movw r14, field_R0 @ c += d * R0 + umlal r3, r4, r5, r14 + adds r3, r3, r12 @ c += t9 + adc r4, r4, #0 + + add r1, sp, #4 + 0*4 @ r7,r8,r9 = t0,t1,t2 + ldmia r1, {r7,r8,r9} + + ubfx r2, r3, #0, #22 @ r[9] = c & (M >> 4) + str r2, [r0, #9*4] + mov r3, r3, lsr #22 @ c >>= 22 + orr r3, r3, r4, asl #10 + mov r4, r4, lsr #22 + movw r14, field_R1 << 4 @ c += d * (R1 << 4) + umlal r3, r4, r5, r14 + + movw r14, field_R0 >> 4 @ d = c * (R0 >> 4) + t0 (64x64 multiply+add) + umull r5, r6, r3, r14 @ d = c.lo * (R0 >> 4) + adds r5, r5, r7 @ d.lo += t0 + mla r6, r14, r4, r6 @ d.hi += c.hi * (R0 >> 4) + adc r6, r6, 0 @ d.hi += carry + + bic r2, r5, field_not_M @ r[0] = d & M + str r2, [r0, #0*4] + + mov r5, r5, lsr #26 @ d >>= 26 + orr r5, r5, r6, asl #6 + mov r6, r6, lsr #26 + + movw r14, field_R1 >> 4 @ d += c * (R1 >> 4) + t1 (64x64 multiply+add) + umull r1, r2, r3, r14 @ tmp = c.lo * (R1 >> 4) + adds r5, r5, r8 @ d.lo += t1 + adc r6, r6, #0 @ d.hi += carry + adds r5, r5, r1 @ d.lo += tmp.lo + mla r2, r14, r4, r2 @ tmp.hi += c.hi * (R1 >> 4) + adc r6, r6, r2 @ d.hi += carry + tmp.hi + + bic r2, r5, field_not_M @ r[1] = d & M + str r2, [r0, #1*4] + mov r5, r5, lsr #26 @ d >>= 26 (ignore hi) + orr r5, r5, r6, asl #6 + + add r5, r5, r9 @ d += t2 + str r5, [r0, #2*4] @ r[2] = d + + add sp, sp, #48 + ldmfd sp!, {r4, r5, r6, r7, r8, r9, r10, r11, pc} + .size secp256k1_fe_sqr_inner, .-secp256k1_fe_sqr_inner + diff --git a/node_modules/secp256k1/src/secp256k1/src/basic-config.h b/node_modules/secp256k1/src/secp256k1/src/basic-config.h new file mode 100644 index 0000000..e9be39d --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/basic-config.h @@ -0,0 +1,38 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_BASIC_CONFIG_H +#define SECP256K1_BASIC_CONFIG_H + +#ifdef USE_BASIC_CONFIG + +#undef USE_ASM_X86_64 +#undef USE_ECMULT_STATIC_PRECOMPUTATION +#undef USE_ENDOMORPHISM +#undef USE_EXTERNAL_ASM +#undef USE_EXTERNAL_DEFAULT_CALLBACKS +#undef USE_FIELD_10X26 +#undef USE_FIELD_5X52 +#undef USE_FIELD_INV_BUILTIN +#undef USE_FIELD_INV_NUM +#undef USE_NUM_GMP +#undef USE_NUM_NONE +#undef USE_SCALAR_4X64 +#undef USE_SCALAR_8X32 +#undef USE_SCALAR_INV_BUILTIN +#undef USE_SCALAR_INV_NUM +#undef ECMULT_WINDOW_SIZE + +#define USE_NUM_NONE 1 +#define USE_FIELD_INV_BUILTIN 1 +#define USE_SCALAR_INV_BUILTIN 1 +#define USE_FIELD_10X26 1 +#define USE_SCALAR_8X32 1 +#define ECMULT_WINDOW_SIZE 15 + +#endif /* USE_BASIC_CONFIG */ + +#endif /* SECP256K1_BASIC_CONFIG_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/bench.h b/node_modules/secp256k1/src/secp256k1/src/bench.h new file mode 100644 index 0000000..b8a1c2b --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/bench.h @@ -0,0 +1,124 @@ +/********************************************************************** + * Copyright (c) 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_BENCH_H +#define SECP256K1_BENCH_H + +#include +#include +#include +#include "sys/time.h" + +static int64_t gettime_i64(void) { + struct timeval tv; + gettimeofday(&tv, NULL); + return (int64_t)tv.tv_usec + (int64_t)tv.tv_sec * 1000000LL; +} + +#define FP_EXP (6) +#define FP_MULT (1000000LL) + +/* Format fixed point number. */ +void print_number(const int64_t x) { + int64_t x_abs, y; + int c, i, rounding; + size_t ptr; + char buffer[30]; + + if (x == INT64_MIN) { + /* Prevent UB. */ + printf("ERR"); + return; + } + x_abs = x < 0 ? -x : x; + + /* Determine how many decimals we want to show (more than FP_EXP makes no + * sense). */ + y = x_abs; + c = 0; + while (y > 0LL && y < 100LL * FP_MULT && c < FP_EXP) { + y *= 10LL; + c++; + } + + /* Round to 'c' decimals. */ + y = x_abs; + rounding = 0; + for (i = c; i < FP_EXP; ++i) { + rounding = (y % 10) >= 5; + y /= 10; + } + y += rounding; + + /* Format and print the number. */ + ptr = sizeof(buffer) - 1; + buffer[ptr] = 0; + if (c != 0) { + for (i = 0; i < c; ++i) { + buffer[--ptr] = '0' + (y % 10); + y /= 10; + } + buffer[--ptr] = '.'; + } + do { + buffer[--ptr] = '0' + (y % 10); + y /= 10; + } while (y != 0); + if (x < 0) { + buffer[--ptr] = '-'; + } + printf("%s", &buffer[ptr]); +} + +void run_benchmark(char *name, void (*benchmark)(void*), void (*setup)(void*), void (*teardown)(void*), void* data, int count, int iter) { + int i; + int64_t min = INT64_MAX; + int64_t sum = 0; + int64_t max = 0; + for (i = 0; i < count; i++) { + int64_t begin, total; + if (setup != NULL) { + setup(data); + } + begin = gettime_i64(); + benchmark(data); + total = gettime_i64() - begin; + if (teardown != NULL) { + teardown(data); + } + if (total < min) { + min = total; + } + if (total > max) { + max = total; + } + sum += total; + } + printf("%s: min ", name); + print_number(min * FP_MULT / iter); + printf("us / avg "); + print_number(((sum * FP_MULT) / count) / iter); + printf("us / max "); + print_number(max * FP_MULT / iter); + printf("us\n"); +} + +int have_flag(int argc, char** argv, char *flag) { + char** argm = argv + argc; + argv++; + if (argv == argm) { + return 1; + } + while (argv != NULL && argv != argm) { + if (strcmp(*argv, flag) == 0) { + return 1; + } + argv++; + } + return 0; +} + +#endif /* SECP256K1_BENCH_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/bench_ecdh.c b/node_modules/secp256k1/src/secp256k1/src/bench_ecdh.c new file mode 100644 index 0000000..c1dd5a6 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/bench_ecdh.c @@ -0,0 +1,54 @@ +/********************************************************************** + * Copyright (c) 2015 Pieter Wuille, Andrew Poelstra * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#include + +#include "include/secp256k1.h" +#include "include/secp256k1_ecdh.h" +#include "util.h" +#include "bench.h" + +typedef struct { + secp256k1_context *ctx; + secp256k1_pubkey point; + unsigned char scalar[32]; +} bench_ecdh_data; + +static void bench_ecdh_setup(void* arg) { + int i; + bench_ecdh_data *data = (bench_ecdh_data*)arg; + const unsigned char point[] = { + 0x03, + 0x54, 0x94, 0xc1, 0x5d, 0x32, 0x09, 0x97, 0x06, + 0xc2, 0x39, 0x5f, 0x94, 0x34, 0x87, 0x45, 0xfd, + 0x75, 0x7c, 0xe3, 0x0e, 0x4e, 0x8c, 0x90, 0xfb, + 0xa2, 0xba, 0xd1, 0x84, 0xf8, 0x83, 0xc6, 0x9f + }; + + /* create a context with no capabilities */ + data->ctx = secp256k1_context_create(SECP256K1_FLAGS_TYPE_CONTEXT); + for (i = 0; i < 32; i++) { + data->scalar[i] = i + 1; + } + CHECK(secp256k1_ec_pubkey_parse(data->ctx, &data->point, point, sizeof(point)) == 1); +} + +static void bench_ecdh(void* arg) { + int i; + unsigned char res[32]; + bench_ecdh_data *data = (bench_ecdh_data*)arg; + + for (i = 0; i < 20000; i++) { + CHECK(secp256k1_ecdh(data->ctx, res, &data->point, data->scalar, NULL, NULL) == 1); + } +} + +int main(void) { + bench_ecdh_data data; + + run_benchmark("ecdh", bench_ecdh, bench_ecdh_setup, NULL, &data, 10, 20000); + return 0; +} diff --git a/node_modules/secp256k1/src/secp256k1/src/bench_ecmult.c b/node_modules/secp256k1/src/secp256k1/src/bench_ecmult.c new file mode 100644 index 0000000..7b5d185 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/bench_ecmult.c @@ -0,0 +1,207 @@ +/********************************************************************** + * Copyright (c) 2017 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ +#include + +#include "include/secp256k1.h" + +#include "util.h" +#include "hash_impl.h" +#include "num_impl.h" +#include "field_impl.h" +#include "group_impl.h" +#include "scalar_impl.h" +#include "ecmult_impl.h" +#include "bench.h" +#include "secp256k1.c" + +#define POINTS 32768 +#define ITERS 10000 + +typedef struct { + /* Setup once in advance */ + secp256k1_context* ctx; + secp256k1_scratch_space* scratch; + secp256k1_scalar* scalars; + secp256k1_ge* pubkeys; + secp256k1_scalar* seckeys; + secp256k1_gej* expected_output; + secp256k1_ecmult_multi_func ecmult_multi; + + /* Changes per test */ + size_t count; + int includes_g; + + /* Changes per test iteration */ + size_t offset1; + size_t offset2; + + /* Test output. */ + secp256k1_gej* output; +} bench_data; + +static int bench_callback(secp256k1_scalar* sc, secp256k1_ge* ge, size_t idx, void* arg) { + bench_data* data = (bench_data*)arg; + if (data->includes_g) ++idx; + if (idx == 0) { + *sc = data->scalars[data->offset1]; + *ge = secp256k1_ge_const_g; + } else { + *sc = data->scalars[(data->offset1 + idx) % POINTS]; + *ge = data->pubkeys[(data->offset2 + idx - 1) % POINTS]; + } + return 1; +} + +static void bench_ecmult(void* arg) { + bench_data* data = (bench_data*)arg; + + size_t count = data->count; + int includes_g = data->includes_g; + size_t iters = 1 + ITERS / count; + size_t iter; + + for (iter = 0; iter < iters; ++iter) { + data->ecmult_multi(&data->ctx->error_callback, &data->ctx->ecmult_ctx, data->scratch, &data->output[iter], data->includes_g ? &data->scalars[data->offset1] : NULL, bench_callback, arg, count - includes_g); + data->offset1 = (data->offset1 + count) % POINTS; + data->offset2 = (data->offset2 + count - 1) % POINTS; + } +} + +static void bench_ecmult_setup(void* arg) { + bench_data* data = (bench_data*)arg; + data->offset1 = (data->count * 0x537b7f6f + 0x8f66a481) % POINTS; + data->offset2 = (data->count * 0x7f6f537b + 0x6a1a8f49) % POINTS; +} + +static void bench_ecmult_teardown(void* arg) { + bench_data* data = (bench_data*)arg; + size_t iters = 1 + ITERS / data->count; + size_t iter; + /* Verify the results in teardown, to avoid doing comparisons while benchmarking. */ + for (iter = 0; iter < iters; ++iter) { + secp256k1_gej tmp; + secp256k1_gej_add_var(&tmp, &data->output[iter], &data->expected_output[iter], NULL); + CHECK(secp256k1_gej_is_infinity(&tmp)); + } +} + +static void generate_scalar(uint32_t num, secp256k1_scalar* scalar) { + secp256k1_sha256 sha256; + unsigned char c[11] = {'e', 'c', 'm', 'u', 'l', 't', 0, 0, 0, 0}; + unsigned char buf[32]; + int overflow = 0; + c[6] = num; + c[7] = num >> 8; + c[8] = num >> 16; + c[9] = num >> 24; + secp256k1_sha256_initialize(&sha256); + secp256k1_sha256_write(&sha256, c, sizeof(c)); + secp256k1_sha256_finalize(&sha256, buf); + secp256k1_scalar_set_b32(scalar, buf, &overflow); + CHECK(!overflow); +} + +static void run_test(bench_data* data, size_t count, int includes_g) { + char str[32]; + static const secp256k1_scalar zero = SECP256K1_SCALAR_CONST(0, 0, 0, 0, 0, 0, 0, 0); + size_t iters = 1 + ITERS / count; + size_t iter; + + data->count = count; + data->includes_g = includes_g; + + /* Compute (the negation of) the expected results directly. */ + data->offset1 = (data->count * 0x537b7f6f + 0x8f66a481) % POINTS; + data->offset2 = (data->count * 0x7f6f537b + 0x6a1a8f49) % POINTS; + for (iter = 0; iter < iters; ++iter) { + secp256k1_scalar tmp; + secp256k1_scalar total = data->scalars[(data->offset1++) % POINTS]; + size_t i = 0; + for (i = 0; i + 1 < count; ++i) { + secp256k1_scalar_mul(&tmp, &data->seckeys[(data->offset2++) % POINTS], &data->scalars[(data->offset1++) % POINTS]); + secp256k1_scalar_add(&total, &total, &tmp); + } + secp256k1_scalar_negate(&total, &total); + secp256k1_ecmult(&data->ctx->ecmult_ctx, &data->expected_output[iter], NULL, &zero, &total); + } + + /* Run the benchmark. */ + sprintf(str, includes_g ? "ecmult_%ig" : "ecmult_%i", (int)count); + run_benchmark(str, bench_ecmult, bench_ecmult_setup, bench_ecmult_teardown, data, 10, count * (1 + ITERS / count)); +} + +int main(int argc, char **argv) { + bench_data data; + int i, p; + secp256k1_gej* pubkeys_gej; + size_t scratch_size; + + data.ctx = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY); + scratch_size = secp256k1_strauss_scratch_size(POINTS) + STRAUSS_SCRATCH_OBJECTS*16; + data.scratch = secp256k1_scratch_space_create(data.ctx, scratch_size); + data.ecmult_multi = secp256k1_ecmult_multi_var; + + if (argc > 1) { + if(have_flag(argc, argv, "pippenger_wnaf")) { + printf("Using pippenger_wnaf:\n"); + data.ecmult_multi = secp256k1_ecmult_pippenger_batch_single; + } else if(have_flag(argc, argv, "strauss_wnaf")) { + printf("Using strauss_wnaf:\n"); + data.ecmult_multi = secp256k1_ecmult_strauss_batch_single; + } else if(have_flag(argc, argv, "simple")) { + printf("Using simple algorithm:\n"); + data.ecmult_multi = secp256k1_ecmult_multi_var; + secp256k1_scratch_space_destroy(data.ctx, data.scratch); + data.scratch = NULL; + } else { + fprintf(stderr, "%s: unrecognized argument '%s'.\n", argv[0], argv[1]); + fprintf(stderr, "Use 'pippenger_wnaf', 'strauss_wnaf', 'simple' or no argument to benchmark a combined algorithm.\n"); + return 1; + } + } + + /* Allocate stuff */ + data.scalars = malloc(sizeof(secp256k1_scalar) * POINTS); + data.seckeys = malloc(sizeof(secp256k1_scalar) * POINTS); + data.pubkeys = malloc(sizeof(secp256k1_ge) * POINTS); + data.expected_output = malloc(sizeof(secp256k1_gej) * (ITERS + 1)); + data.output = malloc(sizeof(secp256k1_gej) * (ITERS + 1)); + + /* Generate a set of scalars, and private/public keypairs. */ + pubkeys_gej = malloc(sizeof(secp256k1_gej) * POINTS); + secp256k1_gej_set_ge(&pubkeys_gej[0], &secp256k1_ge_const_g); + secp256k1_scalar_set_int(&data.seckeys[0], 1); + for (i = 0; i < POINTS; ++i) { + generate_scalar(i, &data.scalars[i]); + if (i) { + secp256k1_gej_double_var(&pubkeys_gej[i], &pubkeys_gej[i - 1], NULL); + secp256k1_scalar_add(&data.seckeys[i], &data.seckeys[i - 1], &data.seckeys[i - 1]); + } + } + secp256k1_ge_set_all_gej_var(data.pubkeys, pubkeys_gej, POINTS); + free(pubkeys_gej); + + for (i = 1; i <= 8; ++i) { + run_test(&data, i, 1); + } + + for (p = 0; p <= 11; ++p) { + for (i = 9; i <= 16; ++i) { + run_test(&data, i << p, 1); + } + } + if (data.scratch != NULL) { + secp256k1_scratch_space_destroy(data.ctx, data.scratch); + } + secp256k1_context_destroy(data.ctx); + free(data.scalars); + free(data.pubkeys); + free(data.seckeys); + free(data.output); + free(data.expected_output); + + return(0); +} diff --git a/node_modules/secp256k1/src/secp256k1/src/bench_internal.c b/node_modules/secp256k1/src/secp256k1/src/bench_internal.c new file mode 100644 index 0000000..9159c0a --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/bench_internal.c @@ -0,0 +1,379 @@ +/********************************************************************** + * Copyright (c) 2014-2015 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ +#include + +#include "include/secp256k1.h" + +#include "util.h" +#include "hash_impl.h" +#include "num_impl.h" +#include "field_impl.h" +#include "group_impl.h" +#include "scalar_impl.h" +#include "ecmult_const_impl.h" +#include "ecmult_impl.h" +#include "bench.h" +#include "secp256k1.c" + +typedef struct { + secp256k1_scalar scalar_x, scalar_y; + secp256k1_fe fe_x, fe_y; + secp256k1_ge ge_x, ge_y; + secp256k1_gej gej_x, gej_y; + unsigned char data[64]; + int wnaf[256]; +} bench_inv; + +void bench_setup(void* arg) { + bench_inv *data = (bench_inv*)arg; + + static const unsigned char init_x[32] = { + 0x02, 0x03, 0x05, 0x07, 0x0b, 0x0d, 0x11, 0x13, + 0x17, 0x1d, 0x1f, 0x25, 0x29, 0x2b, 0x2f, 0x35, + 0x3b, 0x3d, 0x43, 0x47, 0x49, 0x4f, 0x53, 0x59, + 0x61, 0x65, 0x67, 0x6b, 0x6d, 0x71, 0x7f, 0x83 + }; + + static const unsigned char init_y[32] = { + 0x82, 0x83, 0x85, 0x87, 0x8b, 0x8d, 0x81, 0x83, + 0x97, 0xad, 0xaf, 0xb5, 0xb9, 0xbb, 0xbf, 0xc5, + 0xdb, 0xdd, 0xe3, 0xe7, 0xe9, 0xef, 0xf3, 0xf9, + 0x11, 0x15, 0x17, 0x1b, 0x1d, 0xb1, 0xbf, 0xd3 + }; + + secp256k1_scalar_set_b32(&data->scalar_x, init_x, NULL); + secp256k1_scalar_set_b32(&data->scalar_y, init_y, NULL); + secp256k1_fe_set_b32(&data->fe_x, init_x); + secp256k1_fe_set_b32(&data->fe_y, init_y); + CHECK(secp256k1_ge_set_xo_var(&data->ge_x, &data->fe_x, 0)); + CHECK(secp256k1_ge_set_xo_var(&data->ge_y, &data->fe_y, 1)); + secp256k1_gej_set_ge(&data->gej_x, &data->ge_x); + secp256k1_gej_set_ge(&data->gej_y, &data->ge_y); + memcpy(data->data, init_x, 32); + memcpy(data->data + 32, init_y, 32); +} + +void bench_scalar_add(void* arg) { + int i, j = 0; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 2000000; i++) { + j += secp256k1_scalar_add(&data->scalar_x, &data->scalar_x, &data->scalar_y); + } + CHECK(j <= 2000000); +} + +void bench_scalar_negate(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 2000000; i++) { + secp256k1_scalar_negate(&data->scalar_x, &data->scalar_x); + } +} + +void bench_scalar_sqr(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 200000; i++) { + secp256k1_scalar_sqr(&data->scalar_x, &data->scalar_x); + } +} + +void bench_scalar_mul(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 200000; i++) { + secp256k1_scalar_mul(&data->scalar_x, &data->scalar_x, &data->scalar_y); + } +} + +#ifdef USE_ENDOMORPHISM +void bench_scalar_split(void* arg) { + int i, j = 0; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 20000; i++) { + secp256k1_scalar_split_lambda(&data->scalar_x, &data->scalar_y, &data->scalar_x); + j += secp256k1_scalar_add(&data->scalar_x, &data->scalar_x, &data->scalar_y); + } + CHECK(j <= 20000); +} +#endif + +void bench_scalar_inverse(void* arg) { + int i, j = 0; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 2000; i++) { + secp256k1_scalar_inverse(&data->scalar_x, &data->scalar_x); + j += secp256k1_scalar_add(&data->scalar_x, &data->scalar_x, &data->scalar_y); + } + CHECK(j <= 2000); +} + +void bench_scalar_inverse_var(void* arg) { + int i, j = 0; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 2000; i++) { + secp256k1_scalar_inverse_var(&data->scalar_x, &data->scalar_x); + j += secp256k1_scalar_add(&data->scalar_x, &data->scalar_x, &data->scalar_y); + } + CHECK(j <= 2000); +} + +void bench_field_normalize(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 2000000; i++) { + secp256k1_fe_normalize(&data->fe_x); + } +} + +void bench_field_normalize_weak(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 2000000; i++) { + secp256k1_fe_normalize_weak(&data->fe_x); + } +} + +void bench_field_mul(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 200000; i++) { + secp256k1_fe_mul(&data->fe_x, &data->fe_x, &data->fe_y); + } +} + +void bench_field_sqr(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 200000; i++) { + secp256k1_fe_sqr(&data->fe_x, &data->fe_x); + } +} + +void bench_field_inverse(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 20000; i++) { + secp256k1_fe_inv(&data->fe_x, &data->fe_x); + secp256k1_fe_add(&data->fe_x, &data->fe_y); + } +} + +void bench_field_inverse_var(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 20000; i++) { + secp256k1_fe_inv_var(&data->fe_x, &data->fe_x); + secp256k1_fe_add(&data->fe_x, &data->fe_y); + } +} + +void bench_field_sqrt(void* arg) { + int i, j = 0; + bench_inv *data = (bench_inv*)arg; + secp256k1_fe t; + + for (i = 0; i < 20000; i++) { + t = data->fe_x; + j += secp256k1_fe_sqrt(&data->fe_x, &t); + secp256k1_fe_add(&data->fe_x, &data->fe_y); + } + CHECK(j <= 20000); +} + +void bench_group_double_var(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 200000; i++) { + secp256k1_gej_double_var(&data->gej_x, &data->gej_x, NULL); + } +} + +void bench_group_add_var(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 200000; i++) { + secp256k1_gej_add_var(&data->gej_x, &data->gej_x, &data->gej_y, NULL); + } +} + +void bench_group_add_affine(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 200000; i++) { + secp256k1_gej_add_ge(&data->gej_x, &data->gej_x, &data->ge_y); + } +} + +void bench_group_add_affine_var(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 200000; i++) { + secp256k1_gej_add_ge_var(&data->gej_x, &data->gej_x, &data->ge_y, NULL); + } +} + +void bench_group_jacobi_var(void* arg) { + int i, j = 0; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 20000; i++) { + j += secp256k1_gej_has_quad_y_var(&data->gej_x); + } + CHECK(j == 20000); +} + +void bench_ecmult_wnaf(void* arg) { + int i, bits = 0, overflow = 0; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 20000; i++) { + bits += secp256k1_ecmult_wnaf(data->wnaf, 256, &data->scalar_x, WINDOW_A); + overflow += secp256k1_scalar_add(&data->scalar_x, &data->scalar_x, &data->scalar_y); + } + CHECK(overflow >= 0); + CHECK(bits <= 256*20000); +} + +void bench_wnaf_const(void* arg) { + int i, bits = 0, overflow = 0; + bench_inv *data = (bench_inv*)arg; + + for (i = 0; i < 20000; i++) { + bits += secp256k1_wnaf_const(data->wnaf, &data->scalar_x, WINDOW_A, 256); + overflow += secp256k1_scalar_add(&data->scalar_x, &data->scalar_x, &data->scalar_y); + } + CHECK(overflow >= 0); + CHECK(bits <= 256*20000); +} + + +void bench_sha256(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + secp256k1_sha256 sha; + + for (i = 0; i < 20000; i++) { + secp256k1_sha256_initialize(&sha); + secp256k1_sha256_write(&sha, data->data, 32); + secp256k1_sha256_finalize(&sha, data->data); + } +} + +void bench_hmac_sha256(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + secp256k1_hmac_sha256 hmac; + + for (i = 0; i < 20000; i++) { + secp256k1_hmac_sha256_initialize(&hmac, data->data, 32); + secp256k1_hmac_sha256_write(&hmac, data->data, 32); + secp256k1_hmac_sha256_finalize(&hmac, data->data); + } +} + +void bench_rfc6979_hmac_sha256(void* arg) { + int i; + bench_inv *data = (bench_inv*)arg; + secp256k1_rfc6979_hmac_sha256 rng; + + for (i = 0; i < 20000; i++) { + secp256k1_rfc6979_hmac_sha256_initialize(&rng, data->data, 64); + secp256k1_rfc6979_hmac_sha256_generate(&rng, data->data, 32); + } +} + +void bench_context_verify(void* arg) { + int i; + (void)arg; + for (i = 0; i < 20; i++) { + secp256k1_context_destroy(secp256k1_context_create(SECP256K1_CONTEXT_VERIFY)); + } +} + +void bench_context_sign(void* arg) { + int i; + (void)arg; + for (i = 0; i < 200; i++) { + secp256k1_context_destroy(secp256k1_context_create(SECP256K1_CONTEXT_SIGN)); + } +} + +#ifndef USE_NUM_NONE +void bench_num_jacobi(void* arg) { + int i, j = 0; + bench_inv *data = (bench_inv*)arg; + secp256k1_num nx, norder; + + secp256k1_scalar_get_num(&nx, &data->scalar_x); + secp256k1_scalar_order_get_num(&norder); + secp256k1_scalar_get_num(&norder, &data->scalar_y); + + for (i = 0; i < 200000; i++) { + j += secp256k1_num_jacobi(&nx, &norder); + } + CHECK(j <= 200000); +} +#endif + +int main(int argc, char **argv) { + bench_inv data; + if (have_flag(argc, argv, "scalar") || have_flag(argc, argv, "add")) run_benchmark("scalar_add", bench_scalar_add, bench_setup, NULL, &data, 10, 2000000); + if (have_flag(argc, argv, "scalar") || have_flag(argc, argv, "negate")) run_benchmark("scalar_negate", bench_scalar_negate, bench_setup, NULL, &data, 10, 2000000); + if (have_flag(argc, argv, "scalar") || have_flag(argc, argv, "sqr")) run_benchmark("scalar_sqr", bench_scalar_sqr, bench_setup, NULL, &data, 10, 200000); + if (have_flag(argc, argv, "scalar") || have_flag(argc, argv, "mul")) run_benchmark("scalar_mul", bench_scalar_mul, bench_setup, NULL, &data, 10, 200000); +#ifdef USE_ENDOMORPHISM + if (have_flag(argc, argv, "scalar") || have_flag(argc, argv, "split")) run_benchmark("scalar_split", bench_scalar_split, bench_setup, NULL, &data, 10, 20000); +#endif + if (have_flag(argc, argv, "scalar") || have_flag(argc, argv, "inverse")) run_benchmark("scalar_inverse", bench_scalar_inverse, bench_setup, NULL, &data, 10, 2000); + if (have_flag(argc, argv, "scalar") || have_flag(argc, argv, "inverse")) run_benchmark("scalar_inverse_var", bench_scalar_inverse_var, bench_setup, NULL, &data, 10, 2000); + + if (have_flag(argc, argv, "field") || have_flag(argc, argv, "normalize")) run_benchmark("field_normalize", bench_field_normalize, bench_setup, NULL, &data, 10, 2000000); + if (have_flag(argc, argv, "field") || have_flag(argc, argv, "normalize")) run_benchmark("field_normalize_weak", bench_field_normalize_weak, bench_setup, NULL, &data, 10, 2000000); + if (have_flag(argc, argv, "field") || have_flag(argc, argv, "sqr")) run_benchmark("field_sqr", bench_field_sqr, bench_setup, NULL, &data, 10, 200000); + if (have_flag(argc, argv, "field") || have_flag(argc, argv, "mul")) run_benchmark("field_mul", bench_field_mul, bench_setup, NULL, &data, 10, 200000); + if (have_flag(argc, argv, "field") || have_flag(argc, argv, "inverse")) run_benchmark("field_inverse", bench_field_inverse, bench_setup, NULL, &data, 10, 20000); + if (have_flag(argc, argv, "field") || have_flag(argc, argv, "inverse")) run_benchmark("field_inverse_var", bench_field_inverse_var, bench_setup, NULL, &data, 10, 20000); + if (have_flag(argc, argv, "field") || have_flag(argc, argv, "sqrt")) run_benchmark("field_sqrt", bench_field_sqrt, bench_setup, NULL, &data, 10, 20000); + + if (have_flag(argc, argv, "group") || have_flag(argc, argv, "double")) run_benchmark("group_double_var", bench_group_double_var, bench_setup, NULL, &data, 10, 200000); + if (have_flag(argc, argv, "group") || have_flag(argc, argv, "add")) run_benchmark("group_add_var", bench_group_add_var, bench_setup, NULL, &data, 10, 200000); + if (have_flag(argc, argv, "group") || have_flag(argc, argv, "add")) run_benchmark("group_add_affine", bench_group_add_affine, bench_setup, NULL, &data, 10, 200000); + if (have_flag(argc, argv, "group") || have_flag(argc, argv, "add")) run_benchmark("group_add_affine_var", bench_group_add_affine_var, bench_setup, NULL, &data, 10, 200000); + if (have_flag(argc, argv, "group") || have_flag(argc, argv, "jacobi")) run_benchmark("group_jacobi_var", bench_group_jacobi_var, bench_setup, NULL, &data, 10, 20000); + + if (have_flag(argc, argv, "ecmult") || have_flag(argc, argv, "wnaf")) run_benchmark("wnaf_const", bench_wnaf_const, bench_setup, NULL, &data, 10, 20000); + if (have_flag(argc, argv, "ecmult") || have_flag(argc, argv, "wnaf")) run_benchmark("ecmult_wnaf", bench_ecmult_wnaf, bench_setup, NULL, &data, 10, 20000); + + if (have_flag(argc, argv, "hash") || have_flag(argc, argv, "sha256")) run_benchmark("hash_sha256", bench_sha256, bench_setup, NULL, &data, 10, 20000); + if (have_flag(argc, argv, "hash") || have_flag(argc, argv, "hmac")) run_benchmark("hash_hmac_sha256", bench_hmac_sha256, bench_setup, NULL, &data, 10, 20000); + if (have_flag(argc, argv, "hash") || have_flag(argc, argv, "rng6979")) run_benchmark("hash_rfc6979_hmac_sha256", bench_rfc6979_hmac_sha256, bench_setup, NULL, &data, 10, 20000); + + if (have_flag(argc, argv, "context") || have_flag(argc, argv, "verify")) run_benchmark("context_verify", bench_context_verify, bench_setup, NULL, &data, 10, 20); + if (have_flag(argc, argv, "context") || have_flag(argc, argv, "sign")) run_benchmark("context_sign", bench_context_sign, bench_setup, NULL, &data, 10, 200); + +#ifndef USE_NUM_NONE + if (have_flag(argc, argv, "num") || have_flag(argc, argv, "jacobi")) run_benchmark("num_jacobi", bench_num_jacobi, bench_setup, NULL, &data, 10, 200000); +#endif + return 0; +} diff --git a/node_modules/secp256k1/src/secp256k1/src/bench_recover.c b/node_modules/secp256k1/src/secp256k1/src/bench_recover.c new file mode 100644 index 0000000..b806eed --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/bench_recover.c @@ -0,0 +1,60 @@ +/********************************************************************** + * Copyright (c) 2014-2015 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#include "include/secp256k1.h" +#include "include/secp256k1_recovery.h" +#include "util.h" +#include "bench.h" + +typedef struct { + secp256k1_context *ctx; + unsigned char msg[32]; + unsigned char sig[64]; +} bench_recover_data; + +void bench_recover(void* arg) { + int i; + bench_recover_data *data = (bench_recover_data*)arg; + secp256k1_pubkey pubkey; + unsigned char pubkeyc[33]; + + for (i = 0; i < 20000; i++) { + int j; + size_t pubkeylen = 33; + secp256k1_ecdsa_recoverable_signature sig; + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(data->ctx, &sig, data->sig, i % 2)); + CHECK(secp256k1_ecdsa_recover(data->ctx, &pubkey, &sig, data->msg)); + CHECK(secp256k1_ec_pubkey_serialize(data->ctx, pubkeyc, &pubkeylen, &pubkey, SECP256K1_EC_COMPRESSED)); + for (j = 0; j < 32; j++) { + data->sig[j + 32] = data->msg[j]; /* Move former message to S. */ + data->msg[j] = data->sig[j]; /* Move former R to message. */ + data->sig[j] = pubkeyc[j + 1]; /* Move recovered pubkey X coordinate to R (which must be a valid X coordinate). */ + } + } +} + +void bench_recover_setup(void* arg) { + int i; + bench_recover_data *data = (bench_recover_data*)arg; + + for (i = 0; i < 32; i++) { + data->msg[i] = 1 + i; + } + for (i = 0; i < 64; i++) { + data->sig[i] = 65 + i; + } +} + +int main(void) { + bench_recover_data data; + + data.ctx = secp256k1_context_create(SECP256K1_CONTEXT_VERIFY); + + run_benchmark("ecdsa_recover", bench_recover, bench_recover_setup, NULL, &data, 10, 20000); + + secp256k1_context_destroy(data.ctx); + return 0; +} diff --git a/node_modules/secp256k1/src/secp256k1/src/bench_sign.c b/node_modules/secp256k1/src/secp256k1/src/bench_sign.c new file mode 100644 index 0000000..544b439 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/bench_sign.c @@ -0,0 +1,56 @@ +/********************************************************************** + * Copyright (c) 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#include "include/secp256k1.h" +#include "util.h" +#include "bench.h" + +typedef struct { + secp256k1_context* ctx; + unsigned char msg[32]; + unsigned char key[32]; +} bench_sign; + +static void bench_sign_setup(void* arg) { + int i; + bench_sign *data = (bench_sign*)arg; + + for (i = 0; i < 32; i++) { + data->msg[i] = i + 1; + } + for (i = 0; i < 32; i++) { + data->key[i] = i + 65; + } +} + +static void bench_sign_run(void* arg) { + int i; + bench_sign *data = (bench_sign*)arg; + + unsigned char sig[74]; + for (i = 0; i < 20000; i++) { + size_t siglen = 74; + int j; + secp256k1_ecdsa_signature signature; + CHECK(secp256k1_ecdsa_sign(data->ctx, &signature, data->msg, data->key, NULL, NULL)); + CHECK(secp256k1_ecdsa_signature_serialize_der(data->ctx, sig, &siglen, &signature)); + for (j = 0; j < 32; j++) { + data->msg[j] = sig[j]; + data->key[j] = sig[j + 32]; + } + } +} + +int main(void) { + bench_sign data; + + data.ctx = secp256k1_context_create(SECP256K1_CONTEXT_SIGN); + + run_benchmark("ecdsa_sign", bench_sign_run, bench_sign_setup, NULL, &data, 10, 20000); + + secp256k1_context_destroy(data.ctx); + return 0; +} diff --git a/node_modules/secp256k1/src/secp256k1/src/bench_verify.c b/node_modules/secp256k1/src/secp256k1/src/bench_verify.c new file mode 100644 index 0000000..418defa --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/bench_verify.c @@ -0,0 +1,112 @@ +/********************************************************************** + * Copyright (c) 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#include +#include + +#include "include/secp256k1.h" +#include "util.h" +#include "bench.h" + +#ifdef ENABLE_OPENSSL_TESTS +#include +#include +#include +#endif + +typedef struct { + secp256k1_context *ctx; + unsigned char msg[32]; + unsigned char key[32]; + unsigned char sig[72]; + size_t siglen; + unsigned char pubkey[33]; + size_t pubkeylen; +#ifdef ENABLE_OPENSSL_TESTS + EC_GROUP* ec_group; +#endif +} benchmark_verify_t; + +static void benchmark_verify(void* arg) { + int i; + benchmark_verify_t* data = (benchmark_verify_t*)arg; + + for (i = 0; i < 20000; i++) { + secp256k1_pubkey pubkey; + secp256k1_ecdsa_signature sig; + data->sig[data->siglen - 1] ^= (i & 0xFF); + data->sig[data->siglen - 2] ^= ((i >> 8) & 0xFF); + data->sig[data->siglen - 3] ^= ((i >> 16) & 0xFF); + CHECK(secp256k1_ec_pubkey_parse(data->ctx, &pubkey, data->pubkey, data->pubkeylen) == 1); + CHECK(secp256k1_ecdsa_signature_parse_der(data->ctx, &sig, data->sig, data->siglen) == 1); + CHECK(secp256k1_ecdsa_verify(data->ctx, &sig, data->msg, &pubkey) == (i == 0)); + data->sig[data->siglen - 1] ^= (i & 0xFF); + data->sig[data->siglen - 2] ^= ((i >> 8) & 0xFF); + data->sig[data->siglen - 3] ^= ((i >> 16) & 0xFF); + } +} + +#ifdef ENABLE_OPENSSL_TESTS +static void benchmark_verify_openssl(void* arg) { + int i; + benchmark_verify_t* data = (benchmark_verify_t*)arg; + + for (i = 0; i < 20000; i++) { + data->sig[data->siglen - 1] ^= (i & 0xFF); + data->sig[data->siglen - 2] ^= ((i >> 8) & 0xFF); + data->sig[data->siglen - 3] ^= ((i >> 16) & 0xFF); + { + EC_KEY *pkey = EC_KEY_new(); + const unsigned char *pubkey = &data->pubkey[0]; + int result; + + CHECK(pkey != NULL); + result = EC_KEY_set_group(pkey, data->ec_group); + CHECK(result); + result = (o2i_ECPublicKey(&pkey, &pubkey, data->pubkeylen)) != NULL; + CHECK(result); + result = ECDSA_verify(0, &data->msg[0], sizeof(data->msg), &data->sig[0], data->siglen, pkey) == (i == 0); + CHECK(result); + EC_KEY_free(pkey); + } + data->sig[data->siglen - 1] ^= (i & 0xFF); + data->sig[data->siglen - 2] ^= ((i >> 8) & 0xFF); + data->sig[data->siglen - 3] ^= ((i >> 16) & 0xFF); + } +} +#endif + +int main(void) { + int i; + secp256k1_pubkey pubkey; + secp256k1_ecdsa_signature sig; + benchmark_verify_t data; + + data.ctx = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY); + + for (i = 0; i < 32; i++) { + data.msg[i] = 1 + i; + } + for (i = 0; i < 32; i++) { + data.key[i] = 33 + i; + } + data.siglen = 72; + CHECK(secp256k1_ecdsa_sign(data.ctx, &sig, data.msg, data.key, NULL, NULL)); + CHECK(secp256k1_ecdsa_signature_serialize_der(data.ctx, data.sig, &data.siglen, &sig)); + CHECK(secp256k1_ec_pubkey_create(data.ctx, &pubkey, data.key)); + data.pubkeylen = 33; + CHECK(secp256k1_ec_pubkey_serialize(data.ctx, data.pubkey, &data.pubkeylen, &pubkey, SECP256K1_EC_COMPRESSED) == 1); + + run_benchmark("ecdsa_verify", benchmark_verify, NULL, NULL, &data, 10, 20000); +#ifdef ENABLE_OPENSSL_TESTS + data.ec_group = EC_GROUP_new_by_curve_name(NID_secp256k1); + run_benchmark("ecdsa_verify_openssl", benchmark_verify_openssl, NULL, NULL, &data, 10, 20000); + EC_GROUP_free(data.ec_group); +#endif + + secp256k1_context_destroy(data.ctx); + return 0; +} diff --git a/node_modules/secp256k1/src/secp256k1/src/ecdsa.h b/node_modules/secp256k1/src/secp256k1/src/ecdsa.h new file mode 100644 index 0000000..80590c7 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/ecdsa.h @@ -0,0 +1,21 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_ECDSA_H +#define SECP256K1_ECDSA_H + +#include + +#include "scalar.h" +#include "group.h" +#include "ecmult.h" + +static int secp256k1_ecdsa_sig_parse(secp256k1_scalar *r, secp256k1_scalar *s, const unsigned char *sig, size_t size); +static int secp256k1_ecdsa_sig_serialize(unsigned char *sig, size_t *size, const secp256k1_scalar *r, const secp256k1_scalar *s); +static int secp256k1_ecdsa_sig_verify(const secp256k1_ecmult_context *ctx, const secp256k1_scalar* r, const secp256k1_scalar* s, const secp256k1_ge *pubkey, const secp256k1_scalar *message); +static int secp256k1_ecdsa_sig_sign(const secp256k1_ecmult_gen_context *ctx, secp256k1_scalar* r, secp256k1_scalar* s, const secp256k1_scalar *seckey, const secp256k1_scalar *message, const secp256k1_scalar *nonce, int *recid); + +#endif /* SECP256K1_ECDSA_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/ecdsa_impl.h b/node_modules/secp256k1/src/secp256k1/src/ecdsa_impl.h new file mode 100644 index 0000000..eb099c8 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/ecdsa_impl.h @@ -0,0 +1,319 @@ +/********************************************************************** + * Copyright (c) 2013-2015 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + + +#ifndef SECP256K1_ECDSA_IMPL_H +#define SECP256K1_ECDSA_IMPL_H + +#include "scalar.h" +#include "field.h" +#include "group.h" +#include "ecmult.h" +#include "ecmult_gen.h" +#include "ecdsa.h" + +/** Group order for secp256k1 defined as 'n' in "Standards for Efficient Cryptography" (SEC2) 2.7.1 + * sage: for t in xrange(1023, -1, -1): + * .. p = 2**256 - 2**32 - t + * .. if p.is_prime(): + * .. print '%x'%p + * .. break + * 'fffffffffffffffffffffffffffffffffffffffffffffffffffffffefffffc2f' + * sage: a = 0 + * sage: b = 7 + * sage: F = FiniteField (p) + * sage: '%x' % (EllipticCurve ([F (a), F (b)]).order()) + * 'fffffffffffffffffffffffffffffffebaaedce6af48a03bbfd25e8cd0364141' + */ +static const secp256k1_fe secp256k1_ecdsa_const_order_as_fe = SECP256K1_FE_CONST( + 0xFFFFFFFFUL, 0xFFFFFFFFUL, 0xFFFFFFFFUL, 0xFFFFFFFEUL, + 0xBAAEDCE6UL, 0xAF48A03BUL, 0xBFD25E8CUL, 0xD0364141UL +); + +/** Difference between field and order, values 'p' and 'n' values defined in + * "Standards for Efficient Cryptography" (SEC2) 2.7.1. + * sage: p = 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFC2F + * sage: a = 0 + * sage: b = 7 + * sage: F = FiniteField (p) + * sage: '%x' % (p - EllipticCurve ([F (a), F (b)]).order()) + * '14551231950b75fc4402da1722fc9baee' + */ +static const secp256k1_fe secp256k1_ecdsa_const_p_minus_order = SECP256K1_FE_CONST( + 0, 0, 0, 1, 0x45512319UL, 0x50B75FC4UL, 0x402DA172UL, 0x2FC9BAEEUL +); + +static int secp256k1_der_read_len(size_t *len, const unsigned char **sigp, const unsigned char *sigend) { + size_t lenleft; + unsigned char b1; + VERIFY_CHECK(len != NULL); + *len = 0; + if (*sigp >= sigend) { + return 0; + } + b1 = *((*sigp)++); + if (b1 == 0xFF) { + /* X.690-0207 8.1.3.5.c the value 0xFF shall not be used. */ + return 0; + } + if ((b1 & 0x80) == 0) { + /* X.690-0207 8.1.3.4 short form length octets */ + *len = b1; + return 1; + } + if (b1 == 0x80) { + /* Indefinite length is not allowed in DER. */ + return 0; + } + /* X.690-207 8.1.3.5 long form length octets */ + lenleft = b1 & 0x7F; /* lenleft is at least 1 */ + if (lenleft > (size_t)(sigend - *sigp)) { + return 0; + } + if (**sigp == 0) { + /* Not the shortest possible length encoding. */ + return 0; + } + if (lenleft > sizeof(size_t)) { + /* The resulting length would exceed the range of a size_t, so + * certainly longer than the passed array size. + */ + return 0; + } + while (lenleft > 0) { + *len = (*len << 8) | **sigp; + (*sigp)++; + lenleft--; + } + if (*len > (size_t)(sigend - *sigp)) { + /* Result exceeds the length of the passed array. */ + return 0; + } + if (*len < 128) { + /* Not the shortest possible length encoding. */ + return 0; + } + return 1; +} + +static int secp256k1_der_parse_integer(secp256k1_scalar *r, const unsigned char **sig, const unsigned char *sigend) { + int overflow = 0; + unsigned char ra[32] = {0}; + size_t rlen; + + if (*sig == sigend || **sig != 0x02) { + /* Not a primitive integer (X.690-0207 8.3.1). */ + return 0; + } + (*sig)++; + if (secp256k1_der_read_len(&rlen, sig, sigend) == 0) { + return 0; + } + if (rlen == 0 || *sig + rlen > sigend) { + /* Exceeds bounds or not at least length 1 (X.690-0207 8.3.1). */ + return 0; + } + if (**sig == 0x00 && rlen > 1 && (((*sig)[1]) & 0x80) == 0x00) { + /* Excessive 0x00 padding. */ + return 0; + } + if (**sig == 0xFF && rlen > 1 && (((*sig)[1]) & 0x80) == 0x80) { + /* Excessive 0xFF padding. */ + return 0; + } + if ((**sig & 0x80) == 0x80) { + /* Negative. */ + overflow = 1; + } + /* There is at most one leading zero byte: + * if there were two leading zero bytes, we would have failed and returned 0 + * because of excessive 0x00 padding already. */ + if (rlen > 0 && **sig == 0) { + /* Skip leading zero byte */ + rlen--; + (*sig)++; + } + if (rlen > 32) { + overflow = 1; + } + if (!overflow) { + memcpy(ra + 32 - rlen, *sig, rlen); + secp256k1_scalar_set_b32(r, ra, &overflow); + } + if (overflow) { + secp256k1_scalar_set_int(r, 0); + } + (*sig) += rlen; + return 1; +} + +static int secp256k1_ecdsa_sig_parse(secp256k1_scalar *rr, secp256k1_scalar *rs, const unsigned char *sig, size_t size) { + const unsigned char *sigend = sig + size; + size_t rlen; + if (sig == sigend || *(sig++) != 0x30) { + /* The encoding doesn't start with a constructed sequence (X.690-0207 8.9.1). */ + return 0; + } + if (secp256k1_der_read_len(&rlen, &sig, sigend) == 0) { + return 0; + } + if (rlen != (size_t)(sigend - sig)) { + /* Tuple exceeds bounds or garage after tuple. */ + return 0; + } + + if (!secp256k1_der_parse_integer(rr, &sig, sigend)) { + return 0; + } + if (!secp256k1_der_parse_integer(rs, &sig, sigend)) { + return 0; + } + + if (sig != sigend) { + /* Trailing garbage inside tuple. */ + return 0; + } + + return 1; +} + +static int secp256k1_ecdsa_sig_serialize(unsigned char *sig, size_t *size, const secp256k1_scalar* ar, const secp256k1_scalar* as) { + unsigned char r[33] = {0}, s[33] = {0}; + unsigned char *rp = r, *sp = s; + size_t lenR = 33, lenS = 33; + secp256k1_scalar_get_b32(&r[1], ar); + secp256k1_scalar_get_b32(&s[1], as); + while (lenR > 1 && rp[0] == 0 && rp[1] < 0x80) { lenR--; rp++; } + while (lenS > 1 && sp[0] == 0 && sp[1] < 0x80) { lenS--; sp++; } + if (*size < 6+lenS+lenR) { + *size = 6 + lenS + lenR; + return 0; + } + *size = 6 + lenS + lenR; + sig[0] = 0x30; + sig[1] = 4 + lenS + lenR; + sig[2] = 0x02; + sig[3] = lenR; + memcpy(sig+4, rp, lenR); + sig[4+lenR] = 0x02; + sig[5+lenR] = lenS; + memcpy(sig+lenR+6, sp, lenS); + return 1; +} + +static int secp256k1_ecdsa_sig_verify(const secp256k1_ecmult_context *ctx, const secp256k1_scalar *sigr, const secp256k1_scalar *sigs, const secp256k1_ge *pubkey, const secp256k1_scalar *message) { + unsigned char c[32]; + secp256k1_scalar sn, u1, u2; +#if !defined(EXHAUSTIVE_TEST_ORDER) + secp256k1_fe xr; +#endif + secp256k1_gej pubkeyj; + secp256k1_gej pr; + + if (secp256k1_scalar_is_zero(sigr) || secp256k1_scalar_is_zero(sigs)) { + return 0; + } + + secp256k1_scalar_inverse_var(&sn, sigs); + secp256k1_scalar_mul(&u1, &sn, message); + secp256k1_scalar_mul(&u2, &sn, sigr); + secp256k1_gej_set_ge(&pubkeyj, pubkey); + secp256k1_ecmult(ctx, &pr, &pubkeyj, &u2, &u1); + if (secp256k1_gej_is_infinity(&pr)) { + return 0; + } + +#if defined(EXHAUSTIVE_TEST_ORDER) +{ + secp256k1_scalar computed_r; + secp256k1_ge pr_ge; + secp256k1_ge_set_gej(&pr_ge, &pr); + secp256k1_fe_normalize(&pr_ge.x); + + secp256k1_fe_get_b32(c, &pr_ge.x); + secp256k1_scalar_set_b32(&computed_r, c, NULL); + return secp256k1_scalar_eq(sigr, &computed_r); +} +#else + secp256k1_scalar_get_b32(c, sigr); + secp256k1_fe_set_b32(&xr, c); + + /** We now have the recomputed R point in pr, and its claimed x coordinate (modulo n) + * in xr. Naively, we would extract the x coordinate from pr (requiring a inversion modulo p), + * compute the remainder modulo n, and compare it to xr. However: + * + * xr == X(pr) mod n + * <=> exists h. (xr + h * n < p && xr + h * n == X(pr)) + * [Since 2 * n > p, h can only be 0 or 1] + * <=> (xr == X(pr)) || (xr + n < p && xr + n == X(pr)) + * [In Jacobian coordinates, X(pr) is pr.x / pr.z^2 mod p] + * <=> (xr == pr.x / pr.z^2 mod p) || (xr + n < p && xr + n == pr.x / pr.z^2 mod p) + * [Multiplying both sides of the equations by pr.z^2 mod p] + * <=> (xr * pr.z^2 mod p == pr.x) || (xr + n < p && (xr + n) * pr.z^2 mod p == pr.x) + * + * Thus, we can avoid the inversion, but we have to check both cases separately. + * secp256k1_gej_eq_x implements the (xr * pr.z^2 mod p == pr.x) test. + */ + if (secp256k1_gej_eq_x_var(&xr, &pr)) { + /* xr * pr.z^2 mod p == pr.x, so the signature is valid. */ + return 1; + } + if (secp256k1_fe_cmp_var(&xr, &secp256k1_ecdsa_const_p_minus_order) >= 0) { + /* xr + n >= p, so we can skip testing the second case. */ + return 0; + } + secp256k1_fe_add(&xr, &secp256k1_ecdsa_const_order_as_fe); + if (secp256k1_gej_eq_x_var(&xr, &pr)) { + /* (xr + n) * pr.z^2 mod p == pr.x, so the signature is valid. */ + return 1; + } + return 0; +#endif +} + +static int secp256k1_ecdsa_sig_sign(const secp256k1_ecmult_gen_context *ctx, secp256k1_scalar *sigr, secp256k1_scalar *sigs, const secp256k1_scalar *seckey, const secp256k1_scalar *message, const secp256k1_scalar *nonce, int *recid) { + unsigned char b[32]; + secp256k1_gej rp; + secp256k1_ge r; + secp256k1_scalar n; + int overflow = 0; + + secp256k1_ecmult_gen(ctx, &rp, nonce); + secp256k1_ge_set_gej(&r, &rp); + secp256k1_fe_normalize(&r.x); + secp256k1_fe_normalize(&r.y); + secp256k1_fe_get_b32(b, &r.x); + secp256k1_scalar_set_b32(sigr, b, &overflow); + /* These two conditions should be checked before calling */ + VERIFY_CHECK(!secp256k1_scalar_is_zero(sigr)); + VERIFY_CHECK(overflow == 0); + + if (recid) { + /* The overflow condition is cryptographically unreachable as hitting it requires finding the discrete log + * of some P where P.x >= order, and only 1 in about 2^127 points meet this criteria. + */ + *recid = (overflow ? 2 : 0) | (secp256k1_fe_is_odd(&r.y) ? 1 : 0); + } + secp256k1_scalar_mul(&n, sigr, seckey); + secp256k1_scalar_add(&n, &n, message); + secp256k1_scalar_inverse(sigs, nonce); + secp256k1_scalar_mul(sigs, sigs, &n); + secp256k1_scalar_clear(&n); + secp256k1_gej_clear(&rp); + secp256k1_ge_clear(&r); + if (secp256k1_scalar_is_zero(sigs)) { + return 0; + } + if (secp256k1_scalar_is_high(sigs)) { + secp256k1_scalar_negate(sigs, sigs); + if (recid) { + *recid ^= 1; + } + } + return 1; +} + +#endif /* SECP256K1_ECDSA_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/eckey.h b/node_modules/secp256k1/src/secp256k1/src/eckey.h new file mode 100644 index 0000000..b621f1e --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/eckey.h @@ -0,0 +1,25 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_ECKEY_H +#define SECP256K1_ECKEY_H + +#include + +#include "group.h" +#include "scalar.h" +#include "ecmult.h" +#include "ecmult_gen.h" + +static int secp256k1_eckey_pubkey_parse(secp256k1_ge *elem, const unsigned char *pub, size_t size); +static int secp256k1_eckey_pubkey_serialize(secp256k1_ge *elem, unsigned char *pub, size_t *size, int compressed); + +static int secp256k1_eckey_privkey_tweak_add(secp256k1_scalar *key, const secp256k1_scalar *tweak); +static int secp256k1_eckey_pubkey_tweak_add(const secp256k1_ecmult_context *ctx, secp256k1_ge *key, const secp256k1_scalar *tweak); +static int secp256k1_eckey_privkey_tweak_mul(secp256k1_scalar *key, const secp256k1_scalar *tweak); +static int secp256k1_eckey_pubkey_tweak_mul(const secp256k1_ecmult_context *ctx, secp256k1_ge *key, const secp256k1_scalar *tweak); + +#endif /* SECP256K1_ECKEY_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/eckey_impl.h b/node_modules/secp256k1/src/secp256k1/src/eckey_impl.h new file mode 100644 index 0000000..7c5b789 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/eckey_impl.h @@ -0,0 +1,100 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_ECKEY_IMPL_H +#define SECP256K1_ECKEY_IMPL_H + +#include "eckey.h" + +#include "scalar.h" +#include "field.h" +#include "group.h" +#include "ecmult_gen.h" + +static int secp256k1_eckey_pubkey_parse(secp256k1_ge *elem, const unsigned char *pub, size_t size) { + if (size == 33 && (pub[0] == SECP256K1_TAG_PUBKEY_EVEN || pub[0] == SECP256K1_TAG_PUBKEY_ODD)) { + secp256k1_fe x; + return secp256k1_fe_set_b32(&x, pub+1) && secp256k1_ge_set_xo_var(elem, &x, pub[0] == SECP256K1_TAG_PUBKEY_ODD); + } else if (size == 65 && (pub[0] == SECP256K1_TAG_PUBKEY_UNCOMPRESSED || pub[0] == SECP256K1_TAG_PUBKEY_HYBRID_EVEN || pub[0] == SECP256K1_TAG_PUBKEY_HYBRID_ODD)) { + secp256k1_fe x, y; + if (!secp256k1_fe_set_b32(&x, pub+1) || !secp256k1_fe_set_b32(&y, pub+33)) { + return 0; + } + secp256k1_ge_set_xy(elem, &x, &y); + if ((pub[0] == SECP256K1_TAG_PUBKEY_HYBRID_EVEN || pub[0] == SECP256K1_TAG_PUBKEY_HYBRID_ODD) && + secp256k1_fe_is_odd(&y) != (pub[0] == SECP256K1_TAG_PUBKEY_HYBRID_ODD)) { + return 0; + } + return secp256k1_ge_is_valid_var(elem); + } else { + return 0; + } +} + +static int secp256k1_eckey_pubkey_serialize(secp256k1_ge *elem, unsigned char *pub, size_t *size, int compressed) { + if (secp256k1_ge_is_infinity(elem)) { + return 0; + } + secp256k1_fe_normalize_var(&elem->x); + secp256k1_fe_normalize_var(&elem->y); + secp256k1_fe_get_b32(&pub[1], &elem->x); + if (compressed) { + *size = 33; + pub[0] = secp256k1_fe_is_odd(&elem->y) ? SECP256K1_TAG_PUBKEY_ODD : SECP256K1_TAG_PUBKEY_EVEN; + } else { + *size = 65; + pub[0] = SECP256K1_TAG_PUBKEY_UNCOMPRESSED; + secp256k1_fe_get_b32(&pub[33], &elem->y); + } + return 1; +} + +static int secp256k1_eckey_privkey_tweak_add(secp256k1_scalar *key, const secp256k1_scalar *tweak) { + secp256k1_scalar_add(key, key, tweak); + if (secp256k1_scalar_is_zero(key)) { + return 0; + } + return 1; +} + +static int secp256k1_eckey_pubkey_tweak_add(const secp256k1_ecmult_context *ctx, secp256k1_ge *key, const secp256k1_scalar *tweak) { + secp256k1_gej pt; + secp256k1_scalar one; + secp256k1_gej_set_ge(&pt, key); + secp256k1_scalar_set_int(&one, 1); + secp256k1_ecmult(ctx, &pt, &pt, &one, tweak); + + if (secp256k1_gej_is_infinity(&pt)) { + return 0; + } + secp256k1_ge_set_gej(key, &pt); + return 1; +} + +static int secp256k1_eckey_privkey_tweak_mul(secp256k1_scalar *key, const secp256k1_scalar *tweak) { + if (secp256k1_scalar_is_zero(tweak)) { + return 0; + } + + secp256k1_scalar_mul(key, key, tweak); + return 1; +} + +static int secp256k1_eckey_pubkey_tweak_mul(const secp256k1_ecmult_context *ctx, secp256k1_ge *key, const secp256k1_scalar *tweak) { + secp256k1_scalar zero; + secp256k1_gej pt; + if (secp256k1_scalar_is_zero(tweak)) { + return 0; + } + + secp256k1_scalar_set_int(&zero, 0); + secp256k1_gej_set_ge(&pt, key); + secp256k1_ecmult(ctx, &pt, &pt, tweak, &zero); + secp256k1_ge_set_gej(key, &pt); + return 1; +} + +#endif /* SECP256K1_ECKEY_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/ecmult.h b/node_modules/secp256k1/src/secp256k1/src/ecmult.h new file mode 100644 index 0000000..c9b1982 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/ecmult.h @@ -0,0 +1,48 @@ +/********************************************************************** + * Copyright (c) 2013, 2014, 2017 Pieter Wuille, Andrew Poelstra * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_ECMULT_H +#define SECP256K1_ECMULT_H + +#include "num.h" +#include "group.h" +#include "scalar.h" +#include "scratch.h" + +typedef struct { + /* For accelerating the computation of a*P + b*G: */ + secp256k1_ge_storage (*pre_g)[]; /* odd multiples of the generator */ +#ifdef USE_ENDOMORPHISM + secp256k1_ge_storage (*pre_g_128)[]; /* odd multiples of 2^128*generator */ +#endif +} secp256k1_ecmult_context; + +static const size_t SECP256K1_ECMULT_CONTEXT_PREALLOCATED_SIZE; +static void secp256k1_ecmult_context_init(secp256k1_ecmult_context *ctx); +static void secp256k1_ecmult_context_build(secp256k1_ecmult_context *ctx, void **prealloc); +static void secp256k1_ecmult_context_finalize_memcpy(secp256k1_ecmult_context *dst, const secp256k1_ecmult_context *src); +static void secp256k1_ecmult_context_clear(secp256k1_ecmult_context *ctx); +static int secp256k1_ecmult_context_is_built(const secp256k1_ecmult_context *ctx); + +/** Double multiply: R = na*A + ng*G */ +static void secp256k1_ecmult(const secp256k1_ecmult_context *ctx, secp256k1_gej *r, const secp256k1_gej *a, const secp256k1_scalar *na, const secp256k1_scalar *ng); + +typedef int (secp256k1_ecmult_multi_callback)(secp256k1_scalar *sc, secp256k1_ge *pt, size_t idx, void *data); + +/** + * Multi-multiply: R = inp_g_sc * G + sum_i ni * Ai. + * Chooses the right algorithm for a given number of points and scratch space + * size. Resets and overwrites the given scratch space. If the points do not + * fit in the scratch space the algorithm is repeatedly run with batches of + * points. If no scratch space is given then a simple algorithm is used that + * simply multiplies the points with the corresponding scalars and adds them up. + * Returns: 1 on success (including when inp_g_sc is NULL and n is 0) + * 0 if there is not enough scratch space for a single point or + * callback returns 0 + */ +static int secp256k1_ecmult_multi_var(const secp256k1_callback* error_callback, const secp256k1_ecmult_context *ctx, secp256k1_scratch *scratch, secp256k1_gej *r, const secp256k1_scalar *inp_g_sc, secp256k1_ecmult_multi_callback cb, void *cbdata, size_t n); + +#endif /* SECP256K1_ECMULT_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/ecmult_const.h b/node_modules/secp256k1/src/secp256k1/src/ecmult_const.h new file mode 100644 index 0000000..03bb332 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/ecmult_const.h @@ -0,0 +1,20 @@ +/********************************************************************** + * Copyright (c) 2015 Andrew Poelstra * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_ECMULT_CONST_H +#define SECP256K1_ECMULT_CONST_H + +#include "scalar.h" +#include "group.h" + +/** + * Multiply: R = q*A (in constant-time) + * Here `bits` should be set to the maximum bitlength of the _absolute value_ of `q`, plus + * one because we internally sometimes add 2 to the number during the WNAF conversion. + */ +static void secp256k1_ecmult_const(secp256k1_gej *r, const secp256k1_ge *a, const secp256k1_scalar *q, int bits); + +#endif /* SECP256K1_ECMULT_CONST_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/ecmult_const_impl.h b/node_modules/secp256k1/src/secp256k1/src/ecmult_const_impl.h new file mode 100644 index 0000000..aaa576a --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/ecmult_const_impl.h @@ -0,0 +1,261 @@ +/********************************************************************** + * Copyright (c) 2015 Pieter Wuille, Andrew Poelstra * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_ECMULT_CONST_IMPL_H +#define SECP256K1_ECMULT_CONST_IMPL_H + +#include "scalar.h" +#include "group.h" +#include "ecmult_const.h" +#include "ecmult_impl.h" + +/* This is like `ECMULT_TABLE_GET_GE` but is constant time */ +#define ECMULT_CONST_TABLE_GET_GE(r,pre,n,w) do { \ + int m; \ + int abs_n = (n) * (((n) > 0) * 2 - 1); \ + int idx_n = abs_n / 2; \ + secp256k1_fe neg_y; \ + VERIFY_CHECK(((n) & 1) == 1); \ + VERIFY_CHECK((n) >= -((1 << ((w)-1)) - 1)); \ + VERIFY_CHECK((n) <= ((1 << ((w)-1)) - 1)); \ + VERIFY_SETUP(secp256k1_fe_clear(&(r)->x)); \ + VERIFY_SETUP(secp256k1_fe_clear(&(r)->y)); \ + for (m = 0; m < ECMULT_TABLE_SIZE(w); m++) { \ + /* This loop is used to avoid secret data in array indices. See + * the comment in ecmult_gen_impl.h for rationale. */ \ + secp256k1_fe_cmov(&(r)->x, &(pre)[m].x, m == idx_n); \ + secp256k1_fe_cmov(&(r)->y, &(pre)[m].y, m == idx_n); \ + } \ + (r)->infinity = 0; \ + secp256k1_fe_negate(&neg_y, &(r)->y, 1); \ + secp256k1_fe_cmov(&(r)->y, &neg_y, (n) != abs_n); \ +} while(0) + + +/** Convert a number to WNAF notation. + * The number becomes represented by sum(2^{wi} * wnaf[i], i=0..WNAF_SIZE(w)+1) - return_val. + * It has the following guarantees: + * - each wnaf[i] an odd integer between -(1 << w) and (1 << w) + * - each wnaf[i] is nonzero + * - the number of words set is always WNAF_SIZE(w) + 1 + * + * Adapted from `The Width-w NAF Method Provides Small Memory and Fast Elliptic Scalar + * Multiplications Secure against Side Channel Attacks`, Okeya and Tagaki. M. Joye (Ed.) + * CT-RSA 2003, LNCS 2612, pp. 328-443, 2003. Springer-Verlagy Berlin Heidelberg 2003 + * + * Numbers reference steps of `Algorithm SPA-resistant Width-w NAF with Odd Scalar` on pp. 335 + */ +static int secp256k1_wnaf_const(int *wnaf, const secp256k1_scalar *scalar, int w, int size) { + int global_sign; + int skew = 0; + int word = 0; + + /* 1 2 3 */ + int u_last; + int u; + + int flip; + int bit; + secp256k1_scalar s; + int not_neg_one; + + VERIFY_CHECK(w > 0); + VERIFY_CHECK(size > 0); + + /* Note that we cannot handle even numbers by negating them to be odd, as is + * done in other implementations, since if our scalars were specified to have + * width < 256 for performance reasons, their negations would have width 256 + * and we'd lose any performance benefit. Instead, we use a technique from + * Section 4.2 of the Okeya/Tagaki paper, which is to add either 1 (for even) + * or 2 (for odd) to the number we are encoding, returning a skew value indicating + * this, and having the caller compensate after doing the multiplication. + * + * In fact, we _do_ want to negate numbers to minimize their bit-lengths (and in + * particular, to ensure that the outputs from the endomorphism-split fit into + * 128 bits). If we negate, the parity of our number flips, inverting which of + * {1, 2} we want to add to the scalar when ensuring that it's odd. Further + * complicating things, -1 interacts badly with `secp256k1_scalar_cadd_bit` and + * we need to special-case it in this logic. */ + flip = secp256k1_scalar_is_high(scalar); + /* We add 1 to even numbers, 2 to odd ones, noting that negation flips parity */ + bit = flip ^ !secp256k1_scalar_is_even(scalar); + /* We check for negative one, since adding 2 to it will cause an overflow */ + secp256k1_scalar_negate(&s, scalar); + not_neg_one = !secp256k1_scalar_is_one(&s); + s = *scalar; + secp256k1_scalar_cadd_bit(&s, bit, not_neg_one); + /* If we had negative one, flip == 1, s.d[0] == 0, bit == 1, so caller expects + * that we added two to it and flipped it. In fact for -1 these operations are + * identical. We only flipped, but since skewing is required (in the sense that + * the skew must be 1 or 2, never zero) and flipping is not, we need to change + * our flags to claim that we only skewed. */ + global_sign = secp256k1_scalar_cond_negate(&s, flip); + global_sign *= not_neg_one * 2 - 1; + skew = 1 << bit; + + /* 4 */ + u_last = secp256k1_scalar_shr_int(&s, w); + do { + int sign; + int even; + + /* 4.1 4.4 */ + u = secp256k1_scalar_shr_int(&s, w); + /* 4.2 */ + even = ((u & 1) == 0); + sign = 2 * (u_last > 0) - 1; + u += sign * even; + u_last -= sign * even * (1 << w); + + /* 4.3, adapted for global sign change */ + wnaf[word++] = u_last * global_sign; + + u_last = u; + } while (word * w < size); + wnaf[word] = u * global_sign; + + VERIFY_CHECK(secp256k1_scalar_is_zero(&s)); + VERIFY_CHECK(word == WNAF_SIZE_BITS(size, w)); + return skew; +} + +static void secp256k1_ecmult_const(secp256k1_gej *r, const secp256k1_ge *a, const secp256k1_scalar *scalar, int size) { + secp256k1_ge pre_a[ECMULT_TABLE_SIZE(WINDOW_A)]; + secp256k1_ge tmpa; + secp256k1_fe Z; + + int skew_1; +#ifdef USE_ENDOMORPHISM + secp256k1_ge pre_a_lam[ECMULT_TABLE_SIZE(WINDOW_A)]; + int wnaf_lam[1 + WNAF_SIZE(WINDOW_A - 1)]; + int skew_lam; + secp256k1_scalar q_1, q_lam; +#endif + int wnaf_1[1 + WNAF_SIZE(WINDOW_A - 1)]; + + int i; + + /* build wnaf representation for q. */ + int rsize = size; +#ifdef USE_ENDOMORPHISM + if (size > 128) { + rsize = 128; + /* split q into q_1 and q_lam (where q = q_1 + q_lam*lambda, and q_1 and q_lam are ~128 bit) */ + secp256k1_scalar_split_lambda(&q_1, &q_lam, scalar); + skew_1 = secp256k1_wnaf_const(wnaf_1, &q_1, WINDOW_A - 1, 128); + skew_lam = secp256k1_wnaf_const(wnaf_lam, &q_lam, WINDOW_A - 1, 128); + } else +#endif + { + skew_1 = secp256k1_wnaf_const(wnaf_1, scalar, WINDOW_A - 1, size); +#ifdef USE_ENDOMORPHISM + skew_lam = 0; +#endif + } + + /* Calculate odd multiples of a. + * All multiples are brought to the same Z 'denominator', which is stored + * in Z. Due to secp256k1' isomorphism we can do all operations pretending + * that the Z coordinate was 1, use affine addition formulae, and correct + * the Z coordinate of the result once at the end. + */ + secp256k1_gej_set_ge(r, a); + secp256k1_ecmult_odd_multiples_table_globalz_windowa(pre_a, &Z, r); + for (i = 0; i < ECMULT_TABLE_SIZE(WINDOW_A); i++) { + secp256k1_fe_normalize_weak(&pre_a[i].y); + } +#ifdef USE_ENDOMORPHISM + if (size > 128) { + for (i = 0; i < ECMULT_TABLE_SIZE(WINDOW_A); i++) { + secp256k1_ge_mul_lambda(&pre_a_lam[i], &pre_a[i]); + } + } +#endif + + /* first loop iteration (separated out so we can directly set r, rather + * than having it start at infinity, get doubled several times, then have + * its new value added to it) */ + i = wnaf_1[WNAF_SIZE_BITS(rsize, WINDOW_A - 1)]; + VERIFY_CHECK(i != 0); + ECMULT_CONST_TABLE_GET_GE(&tmpa, pre_a, i, WINDOW_A); + secp256k1_gej_set_ge(r, &tmpa); +#ifdef USE_ENDOMORPHISM + if (size > 128) { + i = wnaf_lam[WNAF_SIZE_BITS(rsize, WINDOW_A - 1)]; + VERIFY_CHECK(i != 0); + ECMULT_CONST_TABLE_GET_GE(&tmpa, pre_a_lam, i, WINDOW_A); + secp256k1_gej_add_ge(r, r, &tmpa); + } +#endif + /* remaining loop iterations */ + for (i = WNAF_SIZE_BITS(rsize, WINDOW_A - 1) - 1; i >= 0; i--) { + int n; + int j; + for (j = 0; j < WINDOW_A - 1; ++j) { + secp256k1_gej_double_nonzero(r, r, NULL); + } + + n = wnaf_1[i]; + ECMULT_CONST_TABLE_GET_GE(&tmpa, pre_a, n, WINDOW_A); + VERIFY_CHECK(n != 0); + secp256k1_gej_add_ge(r, r, &tmpa); +#ifdef USE_ENDOMORPHISM + if (size > 128) { + n = wnaf_lam[i]; + ECMULT_CONST_TABLE_GET_GE(&tmpa, pre_a_lam, n, WINDOW_A); + VERIFY_CHECK(n != 0); + secp256k1_gej_add_ge(r, r, &tmpa); + } +#endif + } + + secp256k1_fe_mul(&r->z, &r->z, &Z); + + { + /* Correct for wNAF skew */ + secp256k1_ge correction = *a; + secp256k1_ge_storage correction_1_stor; +#ifdef USE_ENDOMORPHISM + secp256k1_ge_storage correction_lam_stor; +#endif + secp256k1_ge_storage a2_stor; + secp256k1_gej tmpj; + secp256k1_gej_set_ge(&tmpj, &correction); + secp256k1_gej_double_var(&tmpj, &tmpj, NULL); + secp256k1_ge_set_gej(&correction, &tmpj); + secp256k1_ge_to_storage(&correction_1_stor, a); +#ifdef USE_ENDOMORPHISM + if (size > 128) { + secp256k1_ge_to_storage(&correction_lam_stor, a); + } +#endif + secp256k1_ge_to_storage(&a2_stor, &correction); + + /* For odd numbers this is 2a (so replace it), for even ones a (so no-op) */ + secp256k1_ge_storage_cmov(&correction_1_stor, &a2_stor, skew_1 == 2); +#ifdef USE_ENDOMORPHISM + if (size > 128) { + secp256k1_ge_storage_cmov(&correction_lam_stor, &a2_stor, skew_lam == 2); + } +#endif + + /* Apply the correction */ + secp256k1_ge_from_storage(&correction, &correction_1_stor); + secp256k1_ge_neg(&correction, &correction); + secp256k1_gej_add_ge(r, r, &correction); + +#ifdef USE_ENDOMORPHISM + if (size > 128) { + secp256k1_ge_from_storage(&correction, &correction_lam_stor); + secp256k1_ge_neg(&correction, &correction); + secp256k1_ge_mul_lambda(&correction, &correction); + secp256k1_gej_add_ge(r, r, &correction); + } +#endif + } +} + +#endif /* SECP256K1_ECMULT_CONST_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/ecmult_gen.h b/node_modules/secp256k1/src/secp256k1/src/ecmult_gen.h new file mode 100644 index 0000000..30815e5 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/ecmult_gen.h @@ -0,0 +1,50 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_ECMULT_GEN_H +#define SECP256K1_ECMULT_GEN_H + +#include "scalar.h" +#include "group.h" + +#if ECMULT_GEN_PREC_BITS != 2 && ECMULT_GEN_PREC_BITS != 4 && ECMULT_GEN_PREC_BITS != 8 +# error "Set ECMULT_GEN_PREC_BITS to 2, 4 or 8." +#endif +#define ECMULT_GEN_PREC_B ECMULT_GEN_PREC_BITS +#define ECMULT_GEN_PREC_G (1 << ECMULT_GEN_PREC_B) +#define ECMULT_GEN_PREC_N (256 / ECMULT_GEN_PREC_B) + +typedef struct { + /* For accelerating the computation of a*G: + * To harden against timing attacks, use the following mechanism: + * * Break up the multiplicand into groups of PREC_B bits, called n_0, n_1, n_2, ..., n_(PREC_N-1). + * * Compute sum(n_i * (PREC_G)^i * G + U_i, i=0 ... PREC_N-1), where: + * * U_i = U * 2^i, for i=0 ... PREC_N-2 + * * U_i = U * (1-2^(PREC_N-1)), for i=PREC_N-1 + * where U is a point with no known corresponding scalar. Note that sum(U_i, i=0 ... PREC_N-1) = 0. + * For each i, and each of the PREC_G possible values of n_i, (n_i * (PREC_G)^i * G + U_i) is + * precomputed (call it prec(i, n_i)). The formula now becomes sum(prec(i, n_i), i=0 ... PREC_N-1). + * None of the resulting prec group elements have a known scalar, and neither do any of + * the intermediate sums while computing a*G. + */ + secp256k1_ge_storage (*prec)[ECMULT_GEN_PREC_N][ECMULT_GEN_PREC_G]; /* prec[j][i] = (PREC_G)^j * i * G + U_i */ + secp256k1_scalar blind; + secp256k1_gej initial; +} secp256k1_ecmult_gen_context; + +static const size_t SECP256K1_ECMULT_GEN_CONTEXT_PREALLOCATED_SIZE; +static void secp256k1_ecmult_gen_context_init(secp256k1_ecmult_gen_context* ctx); +static void secp256k1_ecmult_gen_context_build(secp256k1_ecmult_gen_context* ctx, void **prealloc); +static void secp256k1_ecmult_gen_context_finalize_memcpy(secp256k1_ecmult_gen_context *dst, const secp256k1_ecmult_gen_context* src); +static void secp256k1_ecmult_gen_context_clear(secp256k1_ecmult_gen_context* ctx); +static int secp256k1_ecmult_gen_context_is_built(const secp256k1_ecmult_gen_context* ctx); + +/** Multiply with the generator: R = a*G */ +static void secp256k1_ecmult_gen(const secp256k1_ecmult_gen_context* ctx, secp256k1_gej *r, const secp256k1_scalar *a); + +static void secp256k1_ecmult_gen_blind(secp256k1_ecmult_gen_context *ctx, const unsigned char *seed32); + +#endif /* SECP256K1_ECMULT_GEN_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/ecmult_gen_impl.h b/node_modules/secp256k1/src/secp256k1/src/ecmult_gen_impl.h new file mode 100644 index 0000000..a1b9639 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/ecmult_gen_impl.h @@ -0,0 +1,211 @@ +/********************************************************************** + * Copyright (c) 2013, 2014, 2015 Pieter Wuille, Gregory Maxwell * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_ECMULT_GEN_IMPL_H +#define SECP256K1_ECMULT_GEN_IMPL_H + +#include "util.h" +#include "scalar.h" +#include "group.h" +#include "ecmult_gen.h" +#include "hash_impl.h" +#ifdef USE_ECMULT_STATIC_PRECOMPUTATION +#include "ecmult_static_context.h" +#endif + +#ifndef USE_ECMULT_STATIC_PRECOMPUTATION + static const size_t SECP256K1_ECMULT_GEN_CONTEXT_PREALLOCATED_SIZE = ROUND_TO_ALIGN(sizeof(*((secp256k1_ecmult_gen_context*) NULL)->prec)); +#else + static const size_t SECP256K1_ECMULT_GEN_CONTEXT_PREALLOCATED_SIZE = 0; +#endif + +static void secp256k1_ecmult_gen_context_init(secp256k1_ecmult_gen_context *ctx) { + ctx->prec = NULL; +} + +static void secp256k1_ecmult_gen_context_build(secp256k1_ecmult_gen_context *ctx, void **prealloc) { +#ifndef USE_ECMULT_STATIC_PRECOMPUTATION + secp256k1_ge prec[ECMULT_GEN_PREC_N * ECMULT_GEN_PREC_G]; + secp256k1_gej gj; + secp256k1_gej nums_gej; + int i, j; + size_t const prealloc_size = SECP256K1_ECMULT_GEN_CONTEXT_PREALLOCATED_SIZE; + void* const base = *prealloc; +#endif + + if (ctx->prec != NULL) { + return; + } +#ifndef USE_ECMULT_STATIC_PRECOMPUTATION + ctx->prec = (secp256k1_ge_storage (*)[ECMULT_GEN_PREC_N][ECMULT_GEN_PREC_G])manual_alloc(prealloc, prealloc_size, base, prealloc_size); + + /* get the generator */ + secp256k1_gej_set_ge(&gj, &secp256k1_ge_const_g); + + /* Construct a group element with no known corresponding scalar (nothing up my sleeve). */ + { + static const unsigned char nums_b32[33] = "The scalar for this x is unknown"; + secp256k1_fe nums_x; + secp256k1_ge nums_ge; + int r; + r = secp256k1_fe_set_b32(&nums_x, nums_b32); + (void)r; + VERIFY_CHECK(r); + r = secp256k1_ge_set_xo_var(&nums_ge, &nums_x, 0); + (void)r; + VERIFY_CHECK(r); + secp256k1_gej_set_ge(&nums_gej, &nums_ge); + /* Add G to make the bits in x uniformly distributed. */ + secp256k1_gej_add_ge_var(&nums_gej, &nums_gej, &secp256k1_ge_const_g, NULL); + } + + /* compute prec. */ + { + secp256k1_gej precj[ECMULT_GEN_PREC_N * ECMULT_GEN_PREC_G]; /* Jacobian versions of prec. */ + secp256k1_gej gbase; + secp256k1_gej numsbase; + gbase = gj; /* PREC_G^j * G */ + numsbase = nums_gej; /* 2^j * nums. */ + for (j = 0; j < ECMULT_GEN_PREC_N; j++) { + /* Set precj[j*PREC_G .. j*PREC_G+(PREC_G-1)] to (numsbase, numsbase + gbase, ..., numsbase + (PREC_G-1)*gbase). */ + precj[j*ECMULT_GEN_PREC_G] = numsbase; + for (i = 1; i < ECMULT_GEN_PREC_G; i++) { + secp256k1_gej_add_var(&precj[j*ECMULT_GEN_PREC_G + i], &precj[j*ECMULT_GEN_PREC_G + i - 1], &gbase, NULL); + } + /* Multiply gbase by PREC_G. */ + for (i = 0; i < ECMULT_GEN_PREC_B; i++) { + secp256k1_gej_double_var(&gbase, &gbase, NULL); + } + /* Multiply numbase by 2. */ + secp256k1_gej_double_var(&numsbase, &numsbase, NULL); + if (j == ECMULT_GEN_PREC_N - 2) { + /* In the last iteration, numsbase is (1 - 2^j) * nums instead. */ + secp256k1_gej_neg(&numsbase, &numsbase); + secp256k1_gej_add_var(&numsbase, &numsbase, &nums_gej, NULL); + } + } + secp256k1_ge_set_all_gej_var(prec, precj, ECMULT_GEN_PREC_N * ECMULT_GEN_PREC_G); + } + for (j = 0; j < ECMULT_GEN_PREC_N; j++) { + for (i = 0; i < ECMULT_GEN_PREC_G; i++) { + secp256k1_ge_to_storage(&(*ctx->prec)[j][i], &prec[j*ECMULT_GEN_PREC_G + i]); + } + } +#else + (void)prealloc; + ctx->prec = (secp256k1_ge_storage (*)[ECMULT_GEN_PREC_N][ECMULT_GEN_PREC_G])secp256k1_ecmult_static_context; +#endif + secp256k1_ecmult_gen_blind(ctx, NULL); +} + +static int secp256k1_ecmult_gen_context_is_built(const secp256k1_ecmult_gen_context* ctx) { + return ctx->prec != NULL; +} + +static void secp256k1_ecmult_gen_context_finalize_memcpy(secp256k1_ecmult_gen_context *dst, const secp256k1_ecmult_gen_context *src) { +#ifndef USE_ECMULT_STATIC_PRECOMPUTATION + if (src->prec != NULL) { + /* We cast to void* first to suppress a -Wcast-align warning. */ + dst->prec = (secp256k1_ge_storage (*)[ECMULT_GEN_PREC_N][ECMULT_GEN_PREC_G])(void*)((unsigned char*)dst + ((unsigned char*)src->prec - (unsigned char*)src)); + } +#else + (void)dst, (void)src; +#endif +} + +static void secp256k1_ecmult_gen_context_clear(secp256k1_ecmult_gen_context *ctx) { + secp256k1_scalar_clear(&ctx->blind); + secp256k1_gej_clear(&ctx->initial); + ctx->prec = NULL; +} + +static void secp256k1_ecmult_gen(const secp256k1_ecmult_gen_context *ctx, secp256k1_gej *r, const secp256k1_scalar *gn) { + secp256k1_ge add; + secp256k1_ge_storage adds; + secp256k1_scalar gnb; + int bits; + int i, j; + memset(&adds, 0, sizeof(adds)); + *r = ctx->initial; + /* Blind scalar/point multiplication by computing (n-b)G + bG instead of nG. */ + secp256k1_scalar_add(&gnb, gn, &ctx->blind); + add.infinity = 0; + for (j = 0; j < ECMULT_GEN_PREC_N; j++) { + bits = secp256k1_scalar_get_bits(&gnb, j * ECMULT_GEN_PREC_B, ECMULT_GEN_PREC_B); + for (i = 0; i < ECMULT_GEN_PREC_G; i++) { + /** This uses a conditional move to avoid any secret data in array indexes. + * _Any_ use of secret indexes has been demonstrated to result in timing + * sidechannels, even when the cache-line access patterns are uniform. + * See also: + * "A word of warning", CHES 2013 Rump Session, by Daniel J. Bernstein and Peter Schwabe + * (https://cryptojedi.org/peter/data/chesrump-20130822.pdf) and + * "Cache Attacks and Countermeasures: the Case of AES", RSA 2006, + * by Dag Arne Osvik, Adi Shamir, and Eran Tromer + * (http://www.tau.ac.il/~tromer/papers/cache.pdf) + */ + secp256k1_ge_storage_cmov(&adds, &(*ctx->prec)[j][i], i == bits); + } + secp256k1_ge_from_storage(&add, &adds); + secp256k1_gej_add_ge(r, r, &add); + } + bits = 0; + secp256k1_ge_clear(&add); + secp256k1_scalar_clear(&gnb); +} + +/* Setup blinding values for secp256k1_ecmult_gen. */ +static void secp256k1_ecmult_gen_blind(secp256k1_ecmult_gen_context *ctx, const unsigned char *seed32) { + secp256k1_scalar b; + secp256k1_gej gb; + secp256k1_fe s; + unsigned char nonce32[32]; + secp256k1_rfc6979_hmac_sha256 rng; + int retry; + unsigned char keydata[64] = {0}; + if (seed32 == NULL) { + /* When seed is NULL, reset the initial point and blinding value. */ + secp256k1_gej_set_ge(&ctx->initial, &secp256k1_ge_const_g); + secp256k1_gej_neg(&ctx->initial, &ctx->initial); + secp256k1_scalar_set_int(&ctx->blind, 1); + } + /* The prior blinding value (if not reset) is chained forward by including it in the hash. */ + secp256k1_scalar_get_b32(nonce32, &ctx->blind); + /** Using a CSPRNG allows a failure free interface, avoids needing large amounts of random data, + * and guards against weak or adversarial seeds. This is a simpler and safer interface than + * asking the caller for blinding values directly and expecting them to retry on failure. + */ + memcpy(keydata, nonce32, 32); + if (seed32 != NULL) { + memcpy(keydata + 32, seed32, 32); + } + secp256k1_rfc6979_hmac_sha256_initialize(&rng, keydata, seed32 ? 64 : 32); + memset(keydata, 0, sizeof(keydata)); + /* Retry for out of range results to achieve uniformity. */ + do { + secp256k1_rfc6979_hmac_sha256_generate(&rng, nonce32, 32); + retry = !secp256k1_fe_set_b32(&s, nonce32); + retry = retry || secp256k1_fe_is_zero(&s); + } while (retry); /* This branch true is cryptographically unreachable. Requires sha256_hmac output > Fp. */ + /* Randomize the projection to defend against multiplier sidechannels. */ + secp256k1_gej_rescale(&ctx->initial, &s); + secp256k1_fe_clear(&s); + do { + secp256k1_rfc6979_hmac_sha256_generate(&rng, nonce32, 32); + secp256k1_scalar_set_b32(&b, nonce32, &retry); + /* A blinding value of 0 works, but would undermine the projection hardening. */ + retry = retry || secp256k1_scalar_is_zero(&b); + } while (retry); /* This branch true is cryptographically unreachable. Requires sha256_hmac output > order. */ + secp256k1_rfc6979_hmac_sha256_finalize(&rng); + memset(nonce32, 0, 32); + secp256k1_ecmult_gen(ctx, &gb, &b); + secp256k1_scalar_negate(&b, &b); + ctx->blind = b; + ctx->initial = gb; + secp256k1_scalar_clear(&b); + secp256k1_gej_clear(&gb); +} + +#endif /* SECP256K1_ECMULT_GEN_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/ecmult_impl.h b/node_modules/secp256k1/src/secp256k1/src/ecmult_impl.h new file mode 100644 index 0000000..f03fa94 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/ecmult_impl.h @@ -0,0 +1,1216 @@ +/***************************************************************************** + * Copyright (c) 2013, 2014, 2017 Pieter Wuille, Andrew Poelstra, Jonas Nick * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php. * + *****************************************************************************/ + +#ifndef SECP256K1_ECMULT_IMPL_H +#define SECP256K1_ECMULT_IMPL_H + +#include +#include + +#include "util.h" +#include "group.h" +#include "scalar.h" +#include "ecmult.h" + +#if defined(EXHAUSTIVE_TEST_ORDER) +/* We need to lower these values for exhaustive tests because + * the tables cannot have infinities in them (this breaks the + * affine-isomorphism stuff which tracks z-ratios) */ +# if EXHAUSTIVE_TEST_ORDER > 128 +# define WINDOW_A 5 +# define WINDOW_G 8 +# elif EXHAUSTIVE_TEST_ORDER > 8 +# define WINDOW_A 4 +# define WINDOW_G 4 +# else +# define WINDOW_A 2 +# define WINDOW_G 2 +# endif +#else +/* optimal for 128-bit and 256-bit exponents. */ +# define WINDOW_A 5 +/** Larger values for ECMULT_WINDOW_SIZE result in possibly better + * performance at the cost of an exponentially larger precomputed + * table. The exact table size is + * (1 << (WINDOW_G - 2)) * sizeof(secp256k1_ge_storage) bytes, + * where sizeof(secp256k1_ge_storage) is typically 64 bytes but can + * be larger due to platform-specific padding and alignment. + * If the endomorphism optimization is enabled (USE_ENDOMORMPHSIM) + * two tables of this size are used instead of only one. + */ +# define WINDOW_G ECMULT_WINDOW_SIZE +#endif + +/* Noone will ever need more than a window size of 24. The code might + * be correct for larger values of ECMULT_WINDOW_SIZE but this is not + * not tested. + * + * The following limitations are known, and there are probably more: + * If WINDOW_G > 27 and size_t has 32 bits, then the code is incorrect + * because the size of the memory object that we allocate (in bytes) + * will not fit in a size_t. + * If WINDOW_G > 31 and int has 32 bits, then the code is incorrect + * because certain expressions will overflow. + */ +#if ECMULT_WINDOW_SIZE < 2 || ECMULT_WINDOW_SIZE > 24 +# error Set ECMULT_WINDOW_SIZE to an integer in range [2..24]. +#endif + +#ifdef USE_ENDOMORPHISM + #define WNAF_BITS 128 +#else + #define WNAF_BITS 256 +#endif +#define WNAF_SIZE_BITS(bits, w) (((bits) + (w) - 1) / (w)) +#define WNAF_SIZE(w) WNAF_SIZE_BITS(WNAF_BITS, w) + +/** The number of entries a table with precomputed multiples needs to have. */ +#define ECMULT_TABLE_SIZE(w) (1 << ((w)-2)) + +/* The number of objects allocated on the scratch space for ecmult_multi algorithms */ +#define PIPPENGER_SCRATCH_OBJECTS 6 +#define STRAUSS_SCRATCH_OBJECTS 6 + +#define PIPPENGER_MAX_BUCKET_WINDOW 12 + +/* Minimum number of points for which pippenger_wnaf is faster than strauss wnaf */ +#ifdef USE_ENDOMORPHISM + #define ECMULT_PIPPENGER_THRESHOLD 88 +#else + #define ECMULT_PIPPENGER_THRESHOLD 160 +#endif + +#ifdef USE_ENDOMORPHISM + #define ECMULT_MAX_POINTS_PER_BATCH 5000000 +#else + #define ECMULT_MAX_POINTS_PER_BATCH 10000000 +#endif + +/** Fill a table 'prej' with precomputed odd multiples of a. Prej will contain + * the values [1*a,3*a,...,(2*n-1)*a], so it space for n values. zr[0] will + * contain prej[0].z / a.z. The other zr[i] values = prej[i].z / prej[i-1].z. + * Prej's Z values are undefined, except for the last value. + */ +static void secp256k1_ecmult_odd_multiples_table(int n, secp256k1_gej *prej, secp256k1_fe *zr, const secp256k1_gej *a) { + secp256k1_gej d; + secp256k1_ge a_ge, d_ge; + int i; + + VERIFY_CHECK(!a->infinity); + + secp256k1_gej_double_var(&d, a, NULL); + + /* + * Perform the additions on an isomorphism where 'd' is affine: drop the z coordinate + * of 'd', and scale the 1P starting value's x/y coordinates without changing its z. + */ + d_ge.x = d.x; + d_ge.y = d.y; + d_ge.infinity = 0; + + secp256k1_ge_set_gej_zinv(&a_ge, a, &d.z); + prej[0].x = a_ge.x; + prej[0].y = a_ge.y; + prej[0].z = a->z; + prej[0].infinity = 0; + + zr[0] = d.z; + for (i = 1; i < n; i++) { + secp256k1_gej_add_ge_var(&prej[i], &prej[i-1], &d_ge, &zr[i]); + } + + /* + * Each point in 'prej' has a z coordinate too small by a factor of 'd.z'. Only + * the final point's z coordinate is actually used though, so just update that. + */ + secp256k1_fe_mul(&prej[n-1].z, &prej[n-1].z, &d.z); +} + +/** Fill a table 'pre' with precomputed odd multiples of a. + * + * There are two versions of this function: + * - secp256k1_ecmult_odd_multiples_table_globalz_windowa which brings its + * resulting point set to a single constant Z denominator, stores the X and Y + * coordinates as ge_storage points in pre, and stores the global Z in rz. + * It only operates on tables sized for WINDOW_A wnaf multiples. + * - secp256k1_ecmult_odd_multiples_table_storage_var, which converts its + * resulting point set to actually affine points, and stores those in pre. + * It operates on tables of any size. + * + * To compute a*P + b*G, we compute a table for P using the first function, + * and for G using the second (which requires an inverse, but it only needs to + * happen once). + */ +static void secp256k1_ecmult_odd_multiples_table_globalz_windowa(secp256k1_ge *pre, secp256k1_fe *globalz, const secp256k1_gej *a) { + secp256k1_gej prej[ECMULT_TABLE_SIZE(WINDOW_A)]; + secp256k1_fe zr[ECMULT_TABLE_SIZE(WINDOW_A)]; + + /* Compute the odd multiples in Jacobian form. */ + secp256k1_ecmult_odd_multiples_table(ECMULT_TABLE_SIZE(WINDOW_A), prej, zr, a); + /* Bring them to the same Z denominator. */ + secp256k1_ge_globalz_set_table_gej(ECMULT_TABLE_SIZE(WINDOW_A), pre, globalz, prej, zr); +} + +static void secp256k1_ecmult_odd_multiples_table_storage_var(const int n, secp256k1_ge_storage *pre, const secp256k1_gej *a) { + secp256k1_gej d; + secp256k1_ge d_ge, p_ge; + secp256k1_gej pj; + secp256k1_fe zi; + secp256k1_fe zr; + secp256k1_fe dx_over_dz_squared; + int i; + + VERIFY_CHECK(!a->infinity); + + secp256k1_gej_double_var(&d, a, NULL); + + /* First, we perform all the additions in an isomorphic curve obtained by multiplying + * all `z` coordinates by 1/`d.z`. In these coordinates `d` is affine so we can use + * `secp256k1_gej_add_ge_var` to perform the additions. For each addition, we store + * the resulting y-coordinate and the z-ratio, since we only have enough memory to + * store two field elements. These are sufficient to efficiently undo the isomorphism + * and recompute all the `x`s. + */ + d_ge.x = d.x; + d_ge.y = d.y; + d_ge.infinity = 0; + + secp256k1_ge_set_gej_zinv(&p_ge, a, &d.z); + pj.x = p_ge.x; + pj.y = p_ge.y; + pj.z = a->z; + pj.infinity = 0; + + for (i = 0; i < (n - 1); i++) { + secp256k1_fe_normalize_var(&pj.y); + secp256k1_fe_to_storage(&pre[i].y, &pj.y); + secp256k1_gej_add_ge_var(&pj, &pj, &d_ge, &zr); + secp256k1_fe_normalize_var(&zr); + secp256k1_fe_to_storage(&pre[i].x, &zr); + } + + /* Invert d.z in the same batch, preserving pj.z so we can extract 1/d.z */ + secp256k1_fe_mul(&zi, &pj.z, &d.z); + secp256k1_fe_inv_var(&zi, &zi); + + /* Directly set `pre[n - 1]` to `pj`, saving the inverted z-coordinate so + * that we can combine it with the saved z-ratios to compute the other zs + * without any more inversions. */ + secp256k1_ge_set_gej_zinv(&p_ge, &pj, &zi); + secp256k1_ge_to_storage(&pre[n - 1], &p_ge); + + /* Compute the actual x-coordinate of D, which will be needed below. */ + secp256k1_fe_mul(&d.z, &zi, &pj.z); /* d.z = 1/d.z */ + secp256k1_fe_sqr(&dx_over_dz_squared, &d.z); + secp256k1_fe_mul(&dx_over_dz_squared, &dx_over_dz_squared, &d.x); + + /* Going into the second loop, we have set `pre[n-1]` to its final affine + * form, but still need to set `pre[i]` for `i` in 0 through `n-2`. We + * have `zi = (p.z * d.z)^-1`, where + * + * `p.z` is the z-coordinate of the point on the isomorphic curve + * which was ultimately assigned to `pre[n-1]`. + * `d.z` is the multiplier that must be applied to all z-coordinates + * to move from our isomorphic curve back to secp256k1; so the + * product `p.z * d.z` is the z-coordinate of the secp256k1 + * point assigned to `pre[n-1]`. + * + * All subsequent inverse-z-coordinates can be obtained by multiplying this + * factor by successive z-ratios, which is much more efficient than directly + * computing each one. + * + * Importantly, these inverse-zs will be coordinates of points on secp256k1, + * while our other stored values come from computations on the isomorphic + * curve. So in the below loop, we will take care not to actually use `zi` + * or any derived values until we're back on secp256k1. + */ + i = n - 1; + while (i > 0) { + secp256k1_fe zi2, zi3; + const secp256k1_fe *rzr; + i--; + + secp256k1_ge_from_storage(&p_ge, &pre[i]); + + /* For each remaining point, we extract the z-ratio from the stored + * x-coordinate, compute its z^-1 from that, and compute the full + * point from that. */ + rzr = &p_ge.x; + secp256k1_fe_mul(&zi, &zi, rzr); + secp256k1_fe_sqr(&zi2, &zi); + secp256k1_fe_mul(&zi3, &zi2, &zi); + /* To compute the actual x-coordinate, we use the stored z ratio and + * y-coordinate, which we obtained from `secp256k1_gej_add_ge_var` + * in the loop above, as well as the inverse of the square of its + * z-coordinate. We store the latter in the `zi2` variable, which is + * computed iteratively starting from the overall Z inverse then + * multiplying by each z-ratio in turn. + * + * Denoting the z-ratio as `rzr`, we observe that it is equal to `h` + * from the inside of the above `gej_add_ge_var` call. This satisfies + * + * rzr = d_x * z^2 - x * d_z^2 + * + * where (`d_x`, `d_z`) are Jacobian coordinates of `D` and `(x, z)` + * are Jacobian coordinates of our desired point -- except both are on + * the isomorphic curve that we were using when we called `gej_add_ge_var`. + * To get back to secp256k1, we must multiply both `z`s by `d_z`, or + * equivalently divide both `x`s by `d_z^2`. Our equation then becomes + * + * rzr = d_x * z^2 / d_z^2 - x + * + * (The left-hand-side, being a ratio of z-coordinates, is unaffected + * by the isomorphism.) + * + * Rearranging to solve for `x`, we have + * + * x = d_x * z^2 / d_z^2 - rzr + * + * But what we actually want is the affine coordinate `X = x/z^2`, + * which will satisfy + * + * X = d_x / d_z^2 - rzr / z^2 + * = dx_over_dz_squared - rzr * zi2 + */ + secp256k1_fe_mul(&p_ge.x, rzr, &zi2); + secp256k1_fe_negate(&p_ge.x, &p_ge.x, 1); + secp256k1_fe_add(&p_ge.x, &dx_over_dz_squared); + /* y is stored_y/z^3, as we expect */ + secp256k1_fe_mul(&p_ge.y, &p_ge.y, &zi3); + /* Store */ + secp256k1_ge_to_storage(&pre[i], &p_ge); + } +} + +/** The following two macro retrieves a particular odd multiple from a table + * of precomputed multiples. */ +#define ECMULT_TABLE_GET_GE(r,pre,n,w) do { \ + VERIFY_CHECK(((n) & 1) == 1); \ + VERIFY_CHECK((n) >= -((1 << ((w)-1)) - 1)); \ + VERIFY_CHECK((n) <= ((1 << ((w)-1)) - 1)); \ + if ((n) > 0) { \ + *(r) = (pre)[((n)-1)/2]; \ + } else { \ + *(r) = (pre)[(-(n)-1)/2]; \ + secp256k1_fe_negate(&((r)->y), &((r)->y), 1); \ + } \ +} while(0) + +#define ECMULT_TABLE_GET_GE_STORAGE(r,pre,n,w) do { \ + VERIFY_CHECK(((n) & 1) == 1); \ + VERIFY_CHECK((n) >= -((1 << ((w)-1)) - 1)); \ + VERIFY_CHECK((n) <= ((1 << ((w)-1)) - 1)); \ + if ((n) > 0) { \ + secp256k1_ge_from_storage((r), &(pre)[((n)-1)/2]); \ + } else { \ + secp256k1_ge_from_storage((r), &(pre)[(-(n)-1)/2]); \ + secp256k1_fe_negate(&((r)->y), &((r)->y), 1); \ + } \ +} while(0) + +static const size_t SECP256K1_ECMULT_CONTEXT_PREALLOCATED_SIZE = + ROUND_TO_ALIGN(sizeof((*((secp256k1_ecmult_context*) NULL)->pre_g)[0]) * ECMULT_TABLE_SIZE(WINDOW_G)) +#ifdef USE_ENDOMORPHISM + + ROUND_TO_ALIGN(sizeof((*((secp256k1_ecmult_context*) NULL)->pre_g_128)[0]) * ECMULT_TABLE_SIZE(WINDOW_G)) +#endif + ; + +static void secp256k1_ecmult_context_init(secp256k1_ecmult_context *ctx) { + ctx->pre_g = NULL; +#ifdef USE_ENDOMORPHISM + ctx->pre_g_128 = NULL; +#endif +} + +static void secp256k1_ecmult_context_build(secp256k1_ecmult_context *ctx, void **prealloc) { + secp256k1_gej gj; + void* const base = *prealloc; + size_t const prealloc_size = SECP256K1_ECMULT_CONTEXT_PREALLOCATED_SIZE; + + if (ctx->pre_g != NULL) { + return; + } + + /* get the generator */ + secp256k1_gej_set_ge(&gj, &secp256k1_ge_const_g); + + { + size_t size = sizeof((*ctx->pre_g)[0]) * ((size_t)ECMULT_TABLE_SIZE(WINDOW_G)); + /* check for overflow */ + VERIFY_CHECK(size / sizeof((*ctx->pre_g)[0]) == ((size_t)ECMULT_TABLE_SIZE(WINDOW_G))); + ctx->pre_g = (secp256k1_ge_storage (*)[])manual_alloc(prealloc, sizeof((*ctx->pre_g)[0]) * ECMULT_TABLE_SIZE(WINDOW_G), base, prealloc_size); + } + + /* precompute the tables with odd multiples */ + secp256k1_ecmult_odd_multiples_table_storage_var(ECMULT_TABLE_SIZE(WINDOW_G), *ctx->pre_g, &gj); + +#ifdef USE_ENDOMORPHISM + { + secp256k1_gej g_128j; + int i; + + size_t size = sizeof((*ctx->pre_g_128)[0]) * ((size_t) ECMULT_TABLE_SIZE(WINDOW_G)); + /* check for overflow */ + VERIFY_CHECK(size / sizeof((*ctx->pre_g_128)[0]) == ((size_t)ECMULT_TABLE_SIZE(WINDOW_G))); + ctx->pre_g_128 = (secp256k1_ge_storage (*)[])manual_alloc(prealloc, sizeof((*ctx->pre_g_128)[0]) * ECMULT_TABLE_SIZE(WINDOW_G), base, prealloc_size); + + /* calculate 2^128*generator */ + g_128j = gj; + for (i = 0; i < 128; i++) { + secp256k1_gej_double_var(&g_128j, &g_128j, NULL); + } + secp256k1_ecmult_odd_multiples_table_storage_var(ECMULT_TABLE_SIZE(WINDOW_G), *ctx->pre_g_128, &g_128j); + } +#endif +} + +static void secp256k1_ecmult_context_finalize_memcpy(secp256k1_ecmult_context *dst, const secp256k1_ecmult_context *src) { + if (src->pre_g != NULL) { + /* We cast to void* first to suppress a -Wcast-align warning. */ + dst->pre_g = (secp256k1_ge_storage (*)[])(void*)((unsigned char*)dst + ((unsigned char*)(src->pre_g) - (unsigned char*)src)); + } +#ifdef USE_ENDOMORPHISM + if (src->pre_g_128 != NULL) { + dst->pre_g_128 = (secp256k1_ge_storage (*)[])(void*)((unsigned char*)dst + ((unsigned char*)(src->pre_g_128) - (unsigned char*)src)); + } +#endif +} + +static int secp256k1_ecmult_context_is_built(const secp256k1_ecmult_context *ctx) { + return ctx->pre_g != NULL; +} + +static void secp256k1_ecmult_context_clear(secp256k1_ecmult_context *ctx) { + secp256k1_ecmult_context_init(ctx); +} + +/** Convert a number to WNAF notation. The number becomes represented by sum(2^i * wnaf[i], i=0..bits), + * with the following guarantees: + * - each wnaf[i] is either 0, or an odd integer between -(1<<(w-1) - 1) and (1<<(w-1) - 1) + * - two non-zero entries in wnaf are separated by at least w-1 zeroes. + * - the number of set values in wnaf is returned. This number is at most 256, and at most one more + * than the number of bits in the (absolute value) of the input. + */ +static int secp256k1_ecmult_wnaf(int *wnaf, int len, const secp256k1_scalar *a, int w) { + secp256k1_scalar s; + int last_set_bit = -1; + int bit = 0; + int sign = 1; + int carry = 0; + + VERIFY_CHECK(wnaf != NULL); + VERIFY_CHECK(0 <= len && len <= 256); + VERIFY_CHECK(a != NULL); + VERIFY_CHECK(2 <= w && w <= 31); + + memset(wnaf, 0, len * sizeof(wnaf[0])); + + s = *a; + if (secp256k1_scalar_get_bits(&s, 255, 1)) { + secp256k1_scalar_negate(&s, &s); + sign = -1; + } + + while (bit < len) { + int now; + int word; + if (secp256k1_scalar_get_bits(&s, bit, 1) == (unsigned int)carry) { + bit++; + continue; + } + + now = w; + if (now > len - bit) { + now = len - bit; + } + + word = secp256k1_scalar_get_bits_var(&s, bit, now) + carry; + + carry = (word >> (w-1)) & 1; + word -= carry << w; + + wnaf[bit] = sign * word; + last_set_bit = bit; + + bit += now; + } +#ifdef VERIFY + CHECK(carry == 0); + while (bit < 256) { + CHECK(secp256k1_scalar_get_bits(&s, bit++, 1) == 0); + } +#endif + return last_set_bit + 1; +} + +struct secp256k1_strauss_point_state { +#ifdef USE_ENDOMORPHISM + secp256k1_scalar na_1, na_lam; + int wnaf_na_1[130]; + int wnaf_na_lam[130]; + int bits_na_1; + int bits_na_lam; +#else + int wnaf_na[256]; + int bits_na; +#endif + size_t input_pos; +}; + +struct secp256k1_strauss_state { + secp256k1_gej* prej; + secp256k1_fe* zr; + secp256k1_ge* pre_a; +#ifdef USE_ENDOMORPHISM + secp256k1_ge* pre_a_lam; +#endif + struct secp256k1_strauss_point_state* ps; +}; + +static void secp256k1_ecmult_strauss_wnaf(const secp256k1_ecmult_context *ctx, const struct secp256k1_strauss_state *state, secp256k1_gej *r, int num, const secp256k1_gej *a, const secp256k1_scalar *na, const secp256k1_scalar *ng) { + secp256k1_ge tmpa; + secp256k1_fe Z; +#ifdef USE_ENDOMORPHISM + /* Splitted G factors. */ + secp256k1_scalar ng_1, ng_128; + int wnaf_ng_1[129]; + int bits_ng_1 = 0; + int wnaf_ng_128[129]; + int bits_ng_128 = 0; +#else + int wnaf_ng[256]; + int bits_ng = 0; +#endif + int i; + int bits = 0; + int np; + int no = 0; + + for (np = 0; np < num; ++np) { + if (secp256k1_scalar_is_zero(&na[np]) || secp256k1_gej_is_infinity(&a[np])) { + continue; + } + state->ps[no].input_pos = np; +#ifdef USE_ENDOMORPHISM + /* split na into na_1 and na_lam (where na = na_1 + na_lam*lambda, and na_1 and na_lam are ~128 bit) */ + secp256k1_scalar_split_lambda(&state->ps[no].na_1, &state->ps[no].na_lam, &na[np]); + + /* build wnaf representation for na_1 and na_lam. */ + state->ps[no].bits_na_1 = secp256k1_ecmult_wnaf(state->ps[no].wnaf_na_1, 130, &state->ps[no].na_1, WINDOW_A); + state->ps[no].bits_na_lam = secp256k1_ecmult_wnaf(state->ps[no].wnaf_na_lam, 130, &state->ps[no].na_lam, WINDOW_A); + VERIFY_CHECK(state->ps[no].bits_na_1 <= 130); + VERIFY_CHECK(state->ps[no].bits_na_lam <= 130); + if (state->ps[no].bits_na_1 > bits) { + bits = state->ps[no].bits_na_1; + } + if (state->ps[no].bits_na_lam > bits) { + bits = state->ps[no].bits_na_lam; + } +#else + /* build wnaf representation for na. */ + state->ps[no].bits_na = secp256k1_ecmult_wnaf(state->ps[no].wnaf_na, 256, &na[np], WINDOW_A); + if (state->ps[no].bits_na > bits) { + bits = state->ps[no].bits_na; + } +#endif + ++no; + } + + /* Calculate odd multiples of a. + * All multiples are brought to the same Z 'denominator', which is stored + * in Z. Due to secp256k1' isomorphism we can do all operations pretending + * that the Z coordinate was 1, use affine addition formulae, and correct + * the Z coordinate of the result once at the end. + * The exception is the precomputed G table points, which are actually + * affine. Compared to the base used for other points, they have a Z ratio + * of 1/Z, so we can use secp256k1_gej_add_zinv_var, which uses the same + * isomorphism to efficiently add with a known Z inverse. + */ + if (no > 0) { + /* Compute the odd multiples in Jacobian form. */ + secp256k1_ecmult_odd_multiples_table(ECMULT_TABLE_SIZE(WINDOW_A), state->prej, state->zr, &a[state->ps[0].input_pos]); + for (np = 1; np < no; ++np) { + secp256k1_gej tmp = a[state->ps[np].input_pos]; +#ifdef VERIFY + secp256k1_fe_normalize_var(&(state->prej[(np - 1) * ECMULT_TABLE_SIZE(WINDOW_A) + ECMULT_TABLE_SIZE(WINDOW_A) - 1].z)); +#endif + secp256k1_gej_rescale(&tmp, &(state->prej[(np - 1) * ECMULT_TABLE_SIZE(WINDOW_A) + ECMULT_TABLE_SIZE(WINDOW_A) - 1].z)); + secp256k1_ecmult_odd_multiples_table(ECMULT_TABLE_SIZE(WINDOW_A), state->prej + np * ECMULT_TABLE_SIZE(WINDOW_A), state->zr + np * ECMULT_TABLE_SIZE(WINDOW_A), &tmp); + secp256k1_fe_mul(state->zr + np * ECMULT_TABLE_SIZE(WINDOW_A), state->zr + np * ECMULT_TABLE_SIZE(WINDOW_A), &(a[state->ps[np].input_pos].z)); + } + /* Bring them to the same Z denominator. */ + secp256k1_ge_globalz_set_table_gej(ECMULT_TABLE_SIZE(WINDOW_A) * no, state->pre_a, &Z, state->prej, state->zr); + } else { + secp256k1_fe_set_int(&Z, 1); + } + +#ifdef USE_ENDOMORPHISM + for (np = 0; np < no; ++np) { + for (i = 0; i < ECMULT_TABLE_SIZE(WINDOW_A); i++) { + secp256k1_ge_mul_lambda(&state->pre_a_lam[np * ECMULT_TABLE_SIZE(WINDOW_A) + i], &state->pre_a[np * ECMULT_TABLE_SIZE(WINDOW_A) + i]); + } + } + + if (ng) { + /* split ng into ng_1 and ng_128 (where gn = gn_1 + gn_128*2^128, and gn_1 and gn_128 are ~128 bit) */ + secp256k1_scalar_split_128(&ng_1, &ng_128, ng); + + /* Build wnaf representation for ng_1 and ng_128 */ + bits_ng_1 = secp256k1_ecmult_wnaf(wnaf_ng_1, 129, &ng_1, WINDOW_G); + bits_ng_128 = secp256k1_ecmult_wnaf(wnaf_ng_128, 129, &ng_128, WINDOW_G); + if (bits_ng_1 > bits) { + bits = bits_ng_1; + } + if (bits_ng_128 > bits) { + bits = bits_ng_128; + } + } +#else + if (ng) { + bits_ng = secp256k1_ecmult_wnaf(wnaf_ng, 256, ng, WINDOW_G); + if (bits_ng > bits) { + bits = bits_ng; + } + } +#endif + + secp256k1_gej_set_infinity(r); + + for (i = bits - 1; i >= 0; i--) { + int n; + secp256k1_gej_double_var(r, r, NULL); +#ifdef USE_ENDOMORPHISM + for (np = 0; np < no; ++np) { + if (i < state->ps[np].bits_na_1 && (n = state->ps[np].wnaf_na_1[i])) { + ECMULT_TABLE_GET_GE(&tmpa, state->pre_a + np * ECMULT_TABLE_SIZE(WINDOW_A), n, WINDOW_A); + secp256k1_gej_add_ge_var(r, r, &tmpa, NULL); + } + if (i < state->ps[np].bits_na_lam && (n = state->ps[np].wnaf_na_lam[i])) { + ECMULT_TABLE_GET_GE(&tmpa, state->pre_a_lam + np * ECMULT_TABLE_SIZE(WINDOW_A), n, WINDOW_A); + secp256k1_gej_add_ge_var(r, r, &tmpa, NULL); + } + } + if (i < bits_ng_1 && (n = wnaf_ng_1[i])) { + ECMULT_TABLE_GET_GE_STORAGE(&tmpa, *ctx->pre_g, n, WINDOW_G); + secp256k1_gej_add_zinv_var(r, r, &tmpa, &Z); + } + if (i < bits_ng_128 && (n = wnaf_ng_128[i])) { + ECMULT_TABLE_GET_GE_STORAGE(&tmpa, *ctx->pre_g_128, n, WINDOW_G); + secp256k1_gej_add_zinv_var(r, r, &tmpa, &Z); + } +#else + for (np = 0; np < no; ++np) { + if (i < state->ps[np].bits_na && (n = state->ps[np].wnaf_na[i])) { + ECMULT_TABLE_GET_GE(&tmpa, state->pre_a + np * ECMULT_TABLE_SIZE(WINDOW_A), n, WINDOW_A); + secp256k1_gej_add_ge_var(r, r, &tmpa, NULL); + } + } + if (i < bits_ng && (n = wnaf_ng[i])) { + ECMULT_TABLE_GET_GE_STORAGE(&tmpa, *ctx->pre_g, n, WINDOW_G); + secp256k1_gej_add_zinv_var(r, r, &tmpa, &Z); + } +#endif + } + + if (!r->infinity) { + secp256k1_fe_mul(&r->z, &r->z, &Z); + } +} + +static void secp256k1_ecmult(const secp256k1_ecmult_context *ctx, secp256k1_gej *r, const secp256k1_gej *a, const secp256k1_scalar *na, const secp256k1_scalar *ng) { + secp256k1_gej prej[ECMULT_TABLE_SIZE(WINDOW_A)]; + secp256k1_fe zr[ECMULT_TABLE_SIZE(WINDOW_A)]; + secp256k1_ge pre_a[ECMULT_TABLE_SIZE(WINDOW_A)]; + struct secp256k1_strauss_point_state ps[1]; +#ifdef USE_ENDOMORPHISM + secp256k1_ge pre_a_lam[ECMULT_TABLE_SIZE(WINDOW_A)]; +#endif + struct secp256k1_strauss_state state; + + state.prej = prej; + state.zr = zr; + state.pre_a = pre_a; +#ifdef USE_ENDOMORPHISM + state.pre_a_lam = pre_a_lam; +#endif + state.ps = ps; + secp256k1_ecmult_strauss_wnaf(ctx, &state, r, 1, a, na, ng); +} + +static size_t secp256k1_strauss_scratch_size(size_t n_points) { +#ifdef USE_ENDOMORPHISM + static const size_t point_size = (2 * sizeof(secp256k1_ge) + sizeof(secp256k1_gej) + sizeof(secp256k1_fe)) * ECMULT_TABLE_SIZE(WINDOW_A) + sizeof(struct secp256k1_strauss_point_state) + sizeof(secp256k1_gej) + sizeof(secp256k1_scalar); +#else + static const size_t point_size = (sizeof(secp256k1_ge) + sizeof(secp256k1_gej) + sizeof(secp256k1_fe)) * ECMULT_TABLE_SIZE(WINDOW_A) + sizeof(struct secp256k1_strauss_point_state) + sizeof(secp256k1_gej) + sizeof(secp256k1_scalar); +#endif + return n_points*point_size; +} + +static int secp256k1_ecmult_strauss_batch(const secp256k1_callback* error_callback, const secp256k1_ecmult_context *ctx, secp256k1_scratch *scratch, secp256k1_gej *r, const secp256k1_scalar *inp_g_sc, secp256k1_ecmult_multi_callback cb, void *cbdata, size_t n_points, size_t cb_offset) { + secp256k1_gej* points; + secp256k1_scalar* scalars; + struct secp256k1_strauss_state state; + size_t i; + const size_t scratch_checkpoint = secp256k1_scratch_checkpoint(error_callback, scratch); + + secp256k1_gej_set_infinity(r); + if (inp_g_sc == NULL && n_points == 0) { + return 1; + } + + points = (secp256k1_gej*)secp256k1_scratch_alloc(error_callback, scratch, n_points * sizeof(secp256k1_gej)); + scalars = (secp256k1_scalar*)secp256k1_scratch_alloc(error_callback, scratch, n_points * sizeof(secp256k1_scalar)); + state.prej = (secp256k1_gej*)secp256k1_scratch_alloc(error_callback, scratch, n_points * ECMULT_TABLE_SIZE(WINDOW_A) * sizeof(secp256k1_gej)); + state.zr = (secp256k1_fe*)secp256k1_scratch_alloc(error_callback, scratch, n_points * ECMULT_TABLE_SIZE(WINDOW_A) * sizeof(secp256k1_fe)); +#ifdef USE_ENDOMORPHISM + state.pre_a = (secp256k1_ge*)secp256k1_scratch_alloc(error_callback, scratch, n_points * 2 * ECMULT_TABLE_SIZE(WINDOW_A) * sizeof(secp256k1_ge)); + state.pre_a_lam = state.pre_a + n_points * ECMULT_TABLE_SIZE(WINDOW_A); +#else + state.pre_a = (secp256k1_ge*)secp256k1_scratch_alloc(error_callback, scratch, n_points * ECMULT_TABLE_SIZE(WINDOW_A) * sizeof(secp256k1_ge)); +#endif + state.ps = (struct secp256k1_strauss_point_state*)secp256k1_scratch_alloc(error_callback, scratch, n_points * sizeof(struct secp256k1_strauss_point_state)); + + if (points == NULL || scalars == NULL || state.prej == NULL || state.zr == NULL || state.pre_a == NULL) { + secp256k1_scratch_apply_checkpoint(error_callback, scratch, scratch_checkpoint); + return 0; + } + + for (i = 0; i < n_points; i++) { + secp256k1_ge point; + if (!cb(&scalars[i], &point, i+cb_offset, cbdata)) { + secp256k1_scratch_apply_checkpoint(error_callback, scratch, scratch_checkpoint); + return 0; + } + secp256k1_gej_set_ge(&points[i], &point); + } + secp256k1_ecmult_strauss_wnaf(ctx, &state, r, n_points, points, scalars, inp_g_sc); + secp256k1_scratch_apply_checkpoint(error_callback, scratch, scratch_checkpoint); + return 1; +} + +/* Wrapper for secp256k1_ecmult_multi_func interface */ +static int secp256k1_ecmult_strauss_batch_single(const secp256k1_callback* error_callback, const secp256k1_ecmult_context *actx, secp256k1_scratch *scratch, secp256k1_gej *r, const secp256k1_scalar *inp_g_sc, secp256k1_ecmult_multi_callback cb, void *cbdata, size_t n) { + return secp256k1_ecmult_strauss_batch(error_callback, actx, scratch, r, inp_g_sc, cb, cbdata, n, 0); +} + +static size_t secp256k1_strauss_max_points(const secp256k1_callback* error_callback, secp256k1_scratch *scratch) { + return secp256k1_scratch_max_allocation(error_callback, scratch, STRAUSS_SCRATCH_OBJECTS) / secp256k1_strauss_scratch_size(1); +} + +/** Convert a number to WNAF notation. + * The number becomes represented by sum(2^{wi} * wnaf[i], i=0..WNAF_SIZE(w)+1) - return_val. + * It has the following guarantees: + * - each wnaf[i] is either 0 or an odd integer between -(1 << w) and (1 << w) + * - the number of words set is always WNAF_SIZE(w) + * - the returned skew is 0 or 1 + */ +static int secp256k1_wnaf_fixed(int *wnaf, const secp256k1_scalar *s, int w) { + int skew = 0; + int pos; + int max_pos; + int last_w; + const secp256k1_scalar *work = s; + + if (secp256k1_scalar_is_zero(s)) { + for (pos = 0; pos < WNAF_SIZE(w); pos++) { + wnaf[pos] = 0; + } + return 0; + } + + if (secp256k1_scalar_is_even(s)) { + skew = 1; + } + + wnaf[0] = secp256k1_scalar_get_bits_var(work, 0, w) + skew; + /* Compute last window size. Relevant when window size doesn't divide the + * number of bits in the scalar */ + last_w = WNAF_BITS - (WNAF_SIZE(w) - 1) * w; + + /* Store the position of the first nonzero word in max_pos to allow + * skipping leading zeros when calculating the wnaf. */ + for (pos = WNAF_SIZE(w) - 1; pos > 0; pos--) { + int val = secp256k1_scalar_get_bits_var(work, pos * w, pos == WNAF_SIZE(w)-1 ? last_w : w); + if(val != 0) { + break; + } + wnaf[pos] = 0; + } + max_pos = pos; + pos = 1; + + while (pos <= max_pos) { + int val = secp256k1_scalar_get_bits_var(work, pos * w, pos == WNAF_SIZE(w)-1 ? last_w : w); + if ((val & 1) == 0) { + wnaf[pos - 1] -= (1 << w); + wnaf[pos] = (val + 1); + } else { + wnaf[pos] = val; + } + /* Set a coefficient to zero if it is 1 or -1 and the proceeding digit + * is strictly negative or strictly positive respectively. Only change + * coefficients at previous positions because above code assumes that + * wnaf[pos - 1] is odd. + */ + if (pos >= 2 && ((wnaf[pos - 1] == 1 && wnaf[pos - 2] < 0) || (wnaf[pos - 1] == -1 && wnaf[pos - 2] > 0))) { + if (wnaf[pos - 1] == 1) { + wnaf[pos - 2] += 1 << w; + } else { + wnaf[pos - 2] -= 1 << w; + } + wnaf[pos - 1] = 0; + } + ++pos; + } + + return skew; +} + +struct secp256k1_pippenger_point_state { + int skew_na; + size_t input_pos; +}; + +struct secp256k1_pippenger_state { + int *wnaf_na; + struct secp256k1_pippenger_point_state* ps; +}; + +/* + * pippenger_wnaf computes the result of a multi-point multiplication as + * follows: The scalars are brought into wnaf with n_wnaf elements each. Then + * for every i < n_wnaf, first each point is added to a "bucket" corresponding + * to the point's wnaf[i]. Second, the buckets are added together such that + * r += 1*bucket[0] + 3*bucket[1] + 5*bucket[2] + ... + */ +static int secp256k1_ecmult_pippenger_wnaf(secp256k1_gej *buckets, int bucket_window, struct secp256k1_pippenger_state *state, secp256k1_gej *r, const secp256k1_scalar *sc, const secp256k1_ge *pt, size_t num) { + size_t n_wnaf = WNAF_SIZE(bucket_window+1); + size_t np; + size_t no = 0; + int i; + int j; + + for (np = 0; np < num; ++np) { + if (secp256k1_scalar_is_zero(&sc[np]) || secp256k1_ge_is_infinity(&pt[np])) { + continue; + } + state->ps[no].input_pos = np; + state->ps[no].skew_na = secp256k1_wnaf_fixed(&state->wnaf_na[no*n_wnaf], &sc[np], bucket_window+1); + no++; + } + secp256k1_gej_set_infinity(r); + + if (no == 0) { + return 1; + } + + for (i = n_wnaf - 1; i >= 0; i--) { + secp256k1_gej running_sum; + + for(j = 0; j < ECMULT_TABLE_SIZE(bucket_window+2); j++) { + secp256k1_gej_set_infinity(&buckets[j]); + } + + for (np = 0; np < no; ++np) { + int n = state->wnaf_na[np*n_wnaf + i]; + struct secp256k1_pippenger_point_state point_state = state->ps[np]; + secp256k1_ge tmp; + int idx; + + if (i == 0) { + /* correct for wnaf skew */ + int skew = point_state.skew_na; + if (skew) { + secp256k1_ge_neg(&tmp, &pt[point_state.input_pos]); + secp256k1_gej_add_ge_var(&buckets[0], &buckets[0], &tmp, NULL); + } + } + if (n > 0) { + idx = (n - 1)/2; + secp256k1_gej_add_ge_var(&buckets[idx], &buckets[idx], &pt[point_state.input_pos], NULL); + } else if (n < 0) { + idx = -(n + 1)/2; + secp256k1_ge_neg(&tmp, &pt[point_state.input_pos]); + secp256k1_gej_add_ge_var(&buckets[idx], &buckets[idx], &tmp, NULL); + } + } + + for(j = 0; j < bucket_window; j++) { + secp256k1_gej_double_var(r, r, NULL); + } + + secp256k1_gej_set_infinity(&running_sum); + /* Accumulate the sum: bucket[0] + 3*bucket[1] + 5*bucket[2] + 7*bucket[3] + ... + * = bucket[0] + bucket[1] + bucket[2] + bucket[3] + ... + * + 2 * (bucket[1] + 2*bucket[2] + 3*bucket[3] + ...) + * using an intermediate running sum: + * running_sum = bucket[0] + bucket[1] + bucket[2] + ... + * + * The doubling is done implicitly by deferring the final window doubling (of 'r'). + */ + for(j = ECMULT_TABLE_SIZE(bucket_window+2) - 1; j > 0; j--) { + secp256k1_gej_add_var(&running_sum, &running_sum, &buckets[j], NULL); + secp256k1_gej_add_var(r, r, &running_sum, NULL); + } + + secp256k1_gej_add_var(&running_sum, &running_sum, &buckets[0], NULL); + secp256k1_gej_double_var(r, r, NULL); + secp256k1_gej_add_var(r, r, &running_sum, NULL); + } + return 1; +} + +/** + * Returns optimal bucket_window (number of bits of a scalar represented by a + * set of buckets) for a given number of points. + */ +static int secp256k1_pippenger_bucket_window(size_t n) { +#ifdef USE_ENDOMORPHISM + if (n <= 1) { + return 1; + } else if (n <= 4) { + return 2; + } else if (n <= 20) { + return 3; + } else if (n <= 57) { + return 4; + } else if (n <= 136) { + return 5; + } else if (n <= 235) { + return 6; + } else if (n <= 1260) { + return 7; + } else if (n <= 4420) { + return 9; + } else if (n <= 7880) { + return 10; + } else if (n <= 16050) { + return 11; + } else { + return PIPPENGER_MAX_BUCKET_WINDOW; + } +#else + if (n <= 1) { + return 1; + } else if (n <= 11) { + return 2; + } else if (n <= 45) { + return 3; + } else if (n <= 100) { + return 4; + } else if (n <= 275) { + return 5; + } else if (n <= 625) { + return 6; + } else if (n <= 1850) { + return 7; + } else if (n <= 3400) { + return 8; + } else if (n <= 9630) { + return 9; + } else if (n <= 17900) { + return 10; + } else if (n <= 32800) { + return 11; + } else { + return PIPPENGER_MAX_BUCKET_WINDOW; + } +#endif +} + +/** + * Returns the maximum optimal number of points for a bucket_window. + */ +static size_t secp256k1_pippenger_bucket_window_inv(int bucket_window) { + switch(bucket_window) { +#ifdef USE_ENDOMORPHISM + case 1: return 1; + case 2: return 4; + case 3: return 20; + case 4: return 57; + case 5: return 136; + case 6: return 235; + case 7: return 1260; + case 8: return 1260; + case 9: return 4420; + case 10: return 7880; + case 11: return 16050; + case PIPPENGER_MAX_BUCKET_WINDOW: return SIZE_MAX; +#else + case 1: return 1; + case 2: return 11; + case 3: return 45; + case 4: return 100; + case 5: return 275; + case 6: return 625; + case 7: return 1850; + case 8: return 3400; + case 9: return 9630; + case 10: return 17900; + case 11: return 32800; + case PIPPENGER_MAX_BUCKET_WINDOW: return SIZE_MAX; +#endif + } + return 0; +} + + +#ifdef USE_ENDOMORPHISM +SECP256K1_INLINE static void secp256k1_ecmult_endo_split(secp256k1_scalar *s1, secp256k1_scalar *s2, secp256k1_ge *p1, secp256k1_ge *p2) { + secp256k1_scalar tmp = *s1; + secp256k1_scalar_split_lambda(s1, s2, &tmp); + secp256k1_ge_mul_lambda(p2, p1); + + if (secp256k1_scalar_is_high(s1)) { + secp256k1_scalar_negate(s1, s1); + secp256k1_ge_neg(p1, p1); + } + if (secp256k1_scalar_is_high(s2)) { + secp256k1_scalar_negate(s2, s2); + secp256k1_ge_neg(p2, p2); + } +} +#endif + +/** + * Returns the scratch size required for a given number of points (excluding + * base point G) without considering alignment. + */ +static size_t secp256k1_pippenger_scratch_size(size_t n_points, int bucket_window) { +#ifdef USE_ENDOMORPHISM + size_t entries = 2*n_points + 2; +#else + size_t entries = n_points + 1; +#endif + size_t entry_size = sizeof(secp256k1_ge) + sizeof(secp256k1_scalar) + sizeof(struct secp256k1_pippenger_point_state) + (WNAF_SIZE(bucket_window+1)+1)*sizeof(int); + return (sizeof(secp256k1_gej) << bucket_window) + sizeof(struct secp256k1_pippenger_state) + entries * entry_size; +} + +static int secp256k1_ecmult_pippenger_batch(const secp256k1_callback* error_callback, const secp256k1_ecmult_context *ctx, secp256k1_scratch *scratch, secp256k1_gej *r, const secp256k1_scalar *inp_g_sc, secp256k1_ecmult_multi_callback cb, void *cbdata, size_t n_points, size_t cb_offset) { + const size_t scratch_checkpoint = secp256k1_scratch_checkpoint(error_callback, scratch); + /* Use 2(n+1) with the endomorphism, n+1 without, when calculating batch + * sizes. The reason for +1 is that we add the G scalar to the list of + * other scalars. */ +#ifdef USE_ENDOMORPHISM + size_t entries = 2*n_points + 2; +#else + size_t entries = n_points + 1; +#endif + secp256k1_ge *points; + secp256k1_scalar *scalars; + secp256k1_gej *buckets; + struct secp256k1_pippenger_state *state_space; + size_t idx = 0; + size_t point_idx = 0; + int i, j; + int bucket_window; + + (void)ctx; + secp256k1_gej_set_infinity(r); + if (inp_g_sc == NULL && n_points == 0) { + return 1; + } + + bucket_window = secp256k1_pippenger_bucket_window(n_points); + points = (secp256k1_ge *) secp256k1_scratch_alloc(error_callback, scratch, entries * sizeof(*points)); + scalars = (secp256k1_scalar *) secp256k1_scratch_alloc(error_callback, scratch, entries * sizeof(*scalars)); + state_space = (struct secp256k1_pippenger_state *) secp256k1_scratch_alloc(error_callback, scratch, sizeof(*state_space)); + if (points == NULL || scalars == NULL || state_space == NULL) { + secp256k1_scratch_apply_checkpoint(error_callback, scratch, scratch_checkpoint); + return 0; + } + + state_space->ps = (struct secp256k1_pippenger_point_state *) secp256k1_scratch_alloc(error_callback, scratch, entries * sizeof(*state_space->ps)); + state_space->wnaf_na = (int *) secp256k1_scratch_alloc(error_callback, scratch, entries*(WNAF_SIZE(bucket_window+1)) * sizeof(int)); + buckets = (secp256k1_gej *) secp256k1_scratch_alloc(error_callback, scratch, (1<ps == NULL || state_space->wnaf_na == NULL || buckets == NULL) { + secp256k1_scratch_apply_checkpoint(error_callback, scratch, scratch_checkpoint); + return 0; + } + + if (inp_g_sc != NULL) { + scalars[0] = *inp_g_sc; + points[0] = secp256k1_ge_const_g; + idx++; +#ifdef USE_ENDOMORPHISM + secp256k1_ecmult_endo_split(&scalars[0], &scalars[1], &points[0], &points[1]); + idx++; +#endif + } + + while (point_idx < n_points) { + if (!cb(&scalars[idx], &points[idx], point_idx + cb_offset, cbdata)) { + secp256k1_scratch_apply_checkpoint(error_callback, scratch, scratch_checkpoint); + return 0; + } + idx++; +#ifdef USE_ENDOMORPHISM + secp256k1_ecmult_endo_split(&scalars[idx - 1], &scalars[idx], &points[idx - 1], &points[idx]); + idx++; +#endif + point_idx++; + } + + secp256k1_ecmult_pippenger_wnaf(buckets, bucket_window, state_space, r, scalars, points, idx); + + /* Clear data */ + for(i = 0; (size_t)i < idx; i++) { + secp256k1_scalar_clear(&scalars[i]); + state_space->ps[i].skew_na = 0; + for(j = 0; j < WNAF_SIZE(bucket_window+1); j++) { + state_space->wnaf_na[i * WNAF_SIZE(bucket_window+1) + j] = 0; + } + } + for(i = 0; i < 1< max_alloc) { + break; + } + space_for_points = max_alloc - space_overhead; + + n_points = space_for_points/entry_size; + n_points = n_points > max_points ? max_points : n_points; + if (n_points > res) { + res = n_points; + } + if (n_points < max_points) { + /* A larger bucket_window may support even more points. But if we + * would choose that then the caller couldn't safely use any number + * smaller than what this function returns */ + break; + } + } + return res; +} + +/* Computes ecmult_multi by simply multiplying and adding each point. Does not + * require a scratch space */ +static int secp256k1_ecmult_multi_simple_var(const secp256k1_ecmult_context *ctx, secp256k1_gej *r, const secp256k1_scalar *inp_g_sc, secp256k1_ecmult_multi_callback cb, void *cbdata, size_t n_points) { + size_t point_idx; + secp256k1_scalar szero; + secp256k1_gej tmpj; + + secp256k1_scalar_set_int(&szero, 0); + secp256k1_gej_set_infinity(r); + secp256k1_gej_set_infinity(&tmpj); + /* r = inp_g_sc*G */ + secp256k1_ecmult(ctx, r, &tmpj, &szero, inp_g_sc); + for (point_idx = 0; point_idx < n_points; point_idx++) { + secp256k1_ge point; + secp256k1_gej pointj; + secp256k1_scalar scalar; + if (!cb(&scalar, &point, point_idx, cbdata)) { + return 0; + } + /* r += scalar*point */ + secp256k1_gej_set_ge(&pointj, &point); + secp256k1_ecmult(ctx, &tmpj, &pointj, &scalar, NULL); + secp256k1_gej_add_var(r, r, &tmpj, NULL); + } + return 1; +} + +/* Compute the number of batches and the batch size given the maximum batch size and the + * total number of points */ +static int secp256k1_ecmult_multi_batch_size_helper(size_t *n_batches, size_t *n_batch_points, size_t max_n_batch_points, size_t n) { + if (max_n_batch_points == 0) { + return 0; + } + if (max_n_batch_points > ECMULT_MAX_POINTS_PER_BATCH) { + max_n_batch_points = ECMULT_MAX_POINTS_PER_BATCH; + } + if (n == 0) { + *n_batches = 0; + *n_batch_points = 0; + return 1; + } + /* Compute ceil(n/max_n_batch_points) and ceil(n/n_batches) */ + *n_batches = 1 + (n - 1) / max_n_batch_points; + *n_batch_points = 1 + (n - 1) / *n_batches; + return 1; +} + +typedef int (*secp256k1_ecmult_multi_func)(const secp256k1_callback* error_callback, const secp256k1_ecmult_context*, secp256k1_scratch*, secp256k1_gej*, const secp256k1_scalar*, secp256k1_ecmult_multi_callback cb, void*, size_t); +static int secp256k1_ecmult_multi_var(const secp256k1_callback* error_callback, const secp256k1_ecmult_context *ctx, secp256k1_scratch *scratch, secp256k1_gej *r, const secp256k1_scalar *inp_g_sc, secp256k1_ecmult_multi_callback cb, void *cbdata, size_t n) { + size_t i; + + int (*f)(const secp256k1_callback* error_callback, const secp256k1_ecmult_context*, secp256k1_scratch*, secp256k1_gej*, const secp256k1_scalar*, secp256k1_ecmult_multi_callback cb, void*, size_t, size_t); + size_t n_batches; + size_t n_batch_points; + + secp256k1_gej_set_infinity(r); + if (inp_g_sc == NULL && n == 0) { + return 1; + } else if (n == 0) { + secp256k1_scalar szero; + secp256k1_scalar_set_int(&szero, 0); + secp256k1_ecmult(ctx, r, r, &szero, inp_g_sc); + return 1; + } + if (scratch == NULL) { + return secp256k1_ecmult_multi_simple_var(ctx, r, inp_g_sc, cb, cbdata, n); + } + + /* Compute the batch sizes for Pippenger's algorithm given a scratch space. If it's greater than + * a threshold use Pippenger's algorithm. Otherwise use Strauss' algorithm. + * As a first step check if there's enough space for Pippenger's algo (which requires less space + * than Strauss' algo) and if not, use the simple algorithm. */ + if (!secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, secp256k1_pippenger_max_points(error_callback, scratch), n)) { + return secp256k1_ecmult_multi_simple_var(ctx, r, inp_g_sc, cb, cbdata, n); + } + if (n_batch_points >= ECMULT_PIPPENGER_THRESHOLD) { + f = secp256k1_ecmult_pippenger_batch; + } else { + if (!secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, secp256k1_strauss_max_points(error_callback, scratch), n)) { + return secp256k1_ecmult_multi_simple_var(ctx, r, inp_g_sc, cb, cbdata, n); + } + f = secp256k1_ecmult_strauss_batch; + } + for(i = 0; i < n_batches; i++) { + size_t nbp = n < n_batch_points ? n : n_batch_points; + size_t offset = n_batch_points*i; + secp256k1_gej tmp; + if (!f(error_callback, ctx, scratch, &tmp, i == 0 ? inp_g_sc : NULL, cb, cbdata, nbp, offset)) { + return 0; + } + secp256k1_gej_add_var(r, r, &tmp, NULL); + n -= nbp; + } + return 1; +} + +#endif /* SECP256K1_ECMULT_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/field.h b/node_modules/secp256k1/src/secp256k1/src/field.h new file mode 100644 index 0000000..bb6692a --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/field.h @@ -0,0 +1,132 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_FIELD_H +#define SECP256K1_FIELD_H + +/** Field element module. + * + * Field elements can be represented in several ways, but code accessing + * it (and implementations) need to take certain properties into account: + * - Each field element can be normalized or not. + * - Each field element has a magnitude, which represents how far away + * its representation is away from normalization. Normalized elements + * always have a magnitude of 1, but a magnitude of 1 doesn't imply + * normality. + */ + +#if defined HAVE_CONFIG_H +#include "libsecp256k1-config.h" +#endif + +#if defined(USE_FIELD_10X26) +#include "field_10x26.h" +#elif defined(USE_FIELD_5X52) +#include "field_5x52.h" +#else +#error "Please select field implementation" +#endif + +#include "util.h" + +/** Normalize a field element. */ +static void secp256k1_fe_normalize(secp256k1_fe *r); + +/** Weakly normalize a field element: reduce it magnitude to 1, but don't fully normalize. */ +static void secp256k1_fe_normalize_weak(secp256k1_fe *r); + +/** Normalize a field element, without constant-time guarantee. */ +static void secp256k1_fe_normalize_var(secp256k1_fe *r); + +/** Verify whether a field element represents zero i.e. would normalize to a zero value. The field + * implementation may optionally normalize the input, but this should not be relied upon. */ +static int secp256k1_fe_normalizes_to_zero(secp256k1_fe *r); + +/** Verify whether a field element represents zero i.e. would normalize to a zero value. The field + * implementation may optionally normalize the input, but this should not be relied upon. */ +static int secp256k1_fe_normalizes_to_zero_var(secp256k1_fe *r); + +/** Set a field element equal to a small integer. Resulting field element is normalized. */ +static void secp256k1_fe_set_int(secp256k1_fe *r, int a); + +/** Sets a field element equal to zero, initializing all fields. */ +static void secp256k1_fe_clear(secp256k1_fe *a); + +/** Verify whether a field element is zero. Requires the input to be normalized. */ +static int secp256k1_fe_is_zero(const secp256k1_fe *a); + +/** Check the "oddness" of a field element. Requires the input to be normalized. */ +static int secp256k1_fe_is_odd(const secp256k1_fe *a); + +/** Compare two field elements. Requires magnitude-1 inputs. */ +static int secp256k1_fe_equal(const secp256k1_fe *a, const secp256k1_fe *b); + +/** Same as secp256k1_fe_equal, but may be variable time. */ +static int secp256k1_fe_equal_var(const secp256k1_fe *a, const secp256k1_fe *b); + +/** Compare two field elements. Requires both inputs to be normalized */ +static int secp256k1_fe_cmp_var(const secp256k1_fe *a, const secp256k1_fe *b); + +/** Set a field element equal to 32-byte big endian value. If successful, the resulting field element is normalized. */ +static int secp256k1_fe_set_b32(secp256k1_fe *r, const unsigned char *a); + +/** Convert a field element to a 32-byte big endian value. Requires the input to be normalized */ +static void secp256k1_fe_get_b32(unsigned char *r, const secp256k1_fe *a); + +/** Set a field element equal to the additive inverse of another. Takes a maximum magnitude of the input + * as an argument. The magnitude of the output is one higher. */ +static void secp256k1_fe_negate(secp256k1_fe *r, const secp256k1_fe *a, int m); + +/** Multiplies the passed field element with a small integer constant. Multiplies the magnitude by that + * small integer. */ +static void secp256k1_fe_mul_int(secp256k1_fe *r, int a); + +/** Adds a field element to another. The result has the sum of the inputs' magnitudes as magnitude. */ +static void secp256k1_fe_add(secp256k1_fe *r, const secp256k1_fe *a); + +/** Sets a field element to be the product of two others. Requires the inputs' magnitudes to be at most 8. + * The output magnitude is 1 (but not guaranteed to be normalized). */ +static void secp256k1_fe_mul(secp256k1_fe *r, const secp256k1_fe *a, const secp256k1_fe * SECP256K1_RESTRICT b); + +/** Sets a field element to be the square of another. Requires the input's magnitude to be at most 8. + * The output magnitude is 1 (but not guaranteed to be normalized). */ +static void secp256k1_fe_sqr(secp256k1_fe *r, const secp256k1_fe *a); + +/** If a has a square root, it is computed in r and 1 is returned. If a does not + * have a square root, the root of its negation is computed and 0 is returned. + * The input's magnitude can be at most 8. The output magnitude is 1 (but not + * guaranteed to be normalized). The result in r will always be a square + * itself. */ +static int secp256k1_fe_sqrt(secp256k1_fe *r, const secp256k1_fe *a); + +/** Checks whether a field element is a quadratic residue. */ +static int secp256k1_fe_is_quad_var(const secp256k1_fe *a); + +/** Sets a field element to be the (modular) inverse of another. Requires the input's magnitude to be + * at most 8. The output magnitude is 1 (but not guaranteed to be normalized). */ +static void secp256k1_fe_inv(secp256k1_fe *r, const secp256k1_fe *a); + +/** Potentially faster version of secp256k1_fe_inv, without constant-time guarantee. */ +static void secp256k1_fe_inv_var(secp256k1_fe *r, const secp256k1_fe *a); + +/** Calculate the (modular) inverses of a batch of field elements. Requires the inputs' magnitudes to be + * at most 8. The output magnitudes are 1 (but not guaranteed to be normalized). The inputs and + * outputs must not overlap in memory. */ +static void secp256k1_fe_inv_all_var(secp256k1_fe *r, const secp256k1_fe *a, size_t len); + +/** Convert a field element to the storage type. */ +static void secp256k1_fe_to_storage(secp256k1_fe_storage *r, const secp256k1_fe *a); + +/** Convert a field element back from the storage type. */ +static void secp256k1_fe_from_storage(secp256k1_fe *r, const secp256k1_fe_storage *a); + +/** If flag is true, set *r equal to *a; otherwise leave it. Constant-time. */ +static void secp256k1_fe_storage_cmov(secp256k1_fe_storage *r, const secp256k1_fe_storage *a, int flag); + +/** If flag is true, set *r equal to *a; otherwise leave it. Constant-time. */ +static void secp256k1_fe_cmov(secp256k1_fe *r, const secp256k1_fe *a, int flag); + +#endif /* SECP256K1_FIELD_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/field_10x26.h b/node_modules/secp256k1/src/secp256k1/src/field_10x26.h new file mode 100644 index 0000000..5ff03c8 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/field_10x26.h @@ -0,0 +1,50 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_FIELD_REPR_H +#define SECP256K1_FIELD_REPR_H + +#include + +typedef struct { + /* X = sum(i=0..9, n[i]*2^(i*26)) mod p + * where p = 2^256 - 0x1000003D1 + */ + uint32_t n[10]; +#ifdef VERIFY + int magnitude; + int normalized; +#endif +} secp256k1_fe; + +/* Unpacks a constant into a overlapping multi-limbed FE element. */ +#define SECP256K1_FE_CONST_INNER(d7, d6, d5, d4, d3, d2, d1, d0) { \ + (d0) & 0x3FFFFFFUL, \ + (((uint32_t)d0) >> 26) | (((uint32_t)(d1) & 0xFFFFFUL) << 6), \ + (((uint32_t)d1) >> 20) | (((uint32_t)(d2) & 0x3FFFUL) << 12), \ + (((uint32_t)d2) >> 14) | (((uint32_t)(d3) & 0xFFUL) << 18), \ + (((uint32_t)d3) >> 8) | (((uint32_t)(d4) & 0x3UL) << 24), \ + (((uint32_t)d4) >> 2) & 0x3FFFFFFUL, \ + (((uint32_t)d4) >> 28) | (((uint32_t)(d5) & 0x3FFFFFUL) << 4), \ + (((uint32_t)d5) >> 22) | (((uint32_t)(d6) & 0xFFFFUL) << 10), \ + (((uint32_t)d6) >> 16) | (((uint32_t)(d7) & 0x3FFUL) << 16), \ + (((uint32_t)d7) >> 10) \ +} + +#ifdef VERIFY +#define SECP256K1_FE_CONST(d7, d6, d5, d4, d3, d2, d1, d0) {SECP256K1_FE_CONST_INNER((d7), (d6), (d5), (d4), (d3), (d2), (d1), (d0)), 1, 1} +#else +#define SECP256K1_FE_CONST(d7, d6, d5, d4, d3, d2, d1, d0) {SECP256K1_FE_CONST_INNER((d7), (d6), (d5), (d4), (d3), (d2), (d1), (d0))} +#endif + +typedef struct { + uint32_t n[8]; +} secp256k1_fe_storage; + +#define SECP256K1_FE_STORAGE_CONST(d7, d6, d5, d4, d3, d2, d1, d0) {{ (d0), (d1), (d2), (d3), (d4), (d5), (d6), (d7) }} +#define SECP256K1_FE_STORAGE_CONST_GET(d) d.n[7], d.n[6], d.n[5], d.n[4],d.n[3], d.n[2], d.n[1], d.n[0] + +#endif /* SECP256K1_FIELD_REPR_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/field_10x26_impl.h b/node_modules/secp256k1/src/secp256k1/src/field_10x26_impl.h new file mode 100644 index 0000000..4ae4fdc --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/field_10x26_impl.h @@ -0,0 +1,1162 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_FIELD_REPR_IMPL_H +#define SECP256K1_FIELD_REPR_IMPL_H + +#include "util.h" +#include "field.h" + +#ifdef VERIFY +static void secp256k1_fe_verify(const secp256k1_fe *a) { + const uint32_t *d = a->n; + int m = a->normalized ? 1 : 2 * a->magnitude, r = 1; + r &= (d[0] <= 0x3FFFFFFUL * m); + r &= (d[1] <= 0x3FFFFFFUL * m); + r &= (d[2] <= 0x3FFFFFFUL * m); + r &= (d[3] <= 0x3FFFFFFUL * m); + r &= (d[4] <= 0x3FFFFFFUL * m); + r &= (d[5] <= 0x3FFFFFFUL * m); + r &= (d[6] <= 0x3FFFFFFUL * m); + r &= (d[7] <= 0x3FFFFFFUL * m); + r &= (d[8] <= 0x3FFFFFFUL * m); + r &= (d[9] <= 0x03FFFFFUL * m); + r &= (a->magnitude >= 0); + r &= (a->magnitude <= 32); + if (a->normalized) { + r &= (a->magnitude <= 1); + if (r && (d[9] == 0x03FFFFFUL)) { + uint32_t mid = d[8] & d[7] & d[6] & d[5] & d[4] & d[3] & d[2]; + if (mid == 0x3FFFFFFUL) { + r &= ((d[1] + 0x40UL + ((d[0] + 0x3D1UL) >> 26)) <= 0x3FFFFFFUL); + } + } + } + VERIFY_CHECK(r == 1); +} +#endif + +static void secp256k1_fe_normalize(secp256k1_fe *r) { + uint32_t t0 = r->n[0], t1 = r->n[1], t2 = r->n[2], t3 = r->n[3], t4 = r->n[4], + t5 = r->n[5], t6 = r->n[6], t7 = r->n[7], t8 = r->n[8], t9 = r->n[9]; + + /* Reduce t9 at the start so there will be at most a single carry from the first pass */ + uint32_t m; + uint32_t x = t9 >> 22; t9 &= 0x03FFFFFUL; + + /* The first pass ensures the magnitude is 1, ... */ + t0 += x * 0x3D1UL; t1 += (x << 6); + t1 += (t0 >> 26); t0 &= 0x3FFFFFFUL; + t2 += (t1 >> 26); t1 &= 0x3FFFFFFUL; + t3 += (t2 >> 26); t2 &= 0x3FFFFFFUL; m = t2; + t4 += (t3 >> 26); t3 &= 0x3FFFFFFUL; m &= t3; + t5 += (t4 >> 26); t4 &= 0x3FFFFFFUL; m &= t4; + t6 += (t5 >> 26); t5 &= 0x3FFFFFFUL; m &= t5; + t7 += (t6 >> 26); t6 &= 0x3FFFFFFUL; m &= t6; + t8 += (t7 >> 26); t7 &= 0x3FFFFFFUL; m &= t7; + t9 += (t8 >> 26); t8 &= 0x3FFFFFFUL; m &= t8; + + /* ... except for a possible carry at bit 22 of t9 (i.e. bit 256 of the field element) */ + VERIFY_CHECK(t9 >> 23 == 0); + + /* At most a single final reduction is needed; check if the value is >= the field characteristic */ + x = (t9 >> 22) | ((t9 == 0x03FFFFFUL) & (m == 0x3FFFFFFUL) + & ((t1 + 0x40UL + ((t0 + 0x3D1UL) >> 26)) > 0x3FFFFFFUL)); + + /* Apply the final reduction (for constant-time behaviour, we do it always) */ + t0 += x * 0x3D1UL; t1 += (x << 6); + t1 += (t0 >> 26); t0 &= 0x3FFFFFFUL; + t2 += (t1 >> 26); t1 &= 0x3FFFFFFUL; + t3 += (t2 >> 26); t2 &= 0x3FFFFFFUL; + t4 += (t3 >> 26); t3 &= 0x3FFFFFFUL; + t5 += (t4 >> 26); t4 &= 0x3FFFFFFUL; + t6 += (t5 >> 26); t5 &= 0x3FFFFFFUL; + t7 += (t6 >> 26); t6 &= 0x3FFFFFFUL; + t8 += (t7 >> 26); t7 &= 0x3FFFFFFUL; + t9 += (t8 >> 26); t8 &= 0x3FFFFFFUL; + + /* If t9 didn't carry to bit 22 already, then it should have after any final reduction */ + VERIFY_CHECK(t9 >> 22 == x); + + /* Mask off the possible multiple of 2^256 from the final reduction */ + t9 &= 0x03FFFFFUL; + + r->n[0] = t0; r->n[1] = t1; r->n[2] = t2; r->n[3] = t3; r->n[4] = t4; + r->n[5] = t5; r->n[6] = t6; r->n[7] = t7; r->n[8] = t8; r->n[9] = t9; + +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 1; + secp256k1_fe_verify(r); +#endif +} + +static void secp256k1_fe_normalize_weak(secp256k1_fe *r) { + uint32_t t0 = r->n[0], t1 = r->n[1], t2 = r->n[2], t3 = r->n[3], t4 = r->n[4], + t5 = r->n[5], t6 = r->n[6], t7 = r->n[7], t8 = r->n[8], t9 = r->n[9]; + + /* Reduce t9 at the start so there will be at most a single carry from the first pass */ + uint32_t x = t9 >> 22; t9 &= 0x03FFFFFUL; + + /* The first pass ensures the magnitude is 1, ... */ + t0 += x * 0x3D1UL; t1 += (x << 6); + t1 += (t0 >> 26); t0 &= 0x3FFFFFFUL; + t2 += (t1 >> 26); t1 &= 0x3FFFFFFUL; + t3 += (t2 >> 26); t2 &= 0x3FFFFFFUL; + t4 += (t3 >> 26); t3 &= 0x3FFFFFFUL; + t5 += (t4 >> 26); t4 &= 0x3FFFFFFUL; + t6 += (t5 >> 26); t5 &= 0x3FFFFFFUL; + t7 += (t6 >> 26); t6 &= 0x3FFFFFFUL; + t8 += (t7 >> 26); t7 &= 0x3FFFFFFUL; + t9 += (t8 >> 26); t8 &= 0x3FFFFFFUL; + + /* ... except for a possible carry at bit 22 of t9 (i.e. bit 256 of the field element) */ + VERIFY_CHECK(t9 >> 23 == 0); + + r->n[0] = t0; r->n[1] = t1; r->n[2] = t2; r->n[3] = t3; r->n[4] = t4; + r->n[5] = t5; r->n[6] = t6; r->n[7] = t7; r->n[8] = t8; r->n[9] = t9; + +#ifdef VERIFY + r->magnitude = 1; + secp256k1_fe_verify(r); +#endif +} + +static void secp256k1_fe_normalize_var(secp256k1_fe *r) { + uint32_t t0 = r->n[0], t1 = r->n[1], t2 = r->n[2], t3 = r->n[3], t4 = r->n[4], + t5 = r->n[5], t6 = r->n[6], t7 = r->n[7], t8 = r->n[8], t9 = r->n[9]; + + /* Reduce t9 at the start so there will be at most a single carry from the first pass */ + uint32_t m; + uint32_t x = t9 >> 22; t9 &= 0x03FFFFFUL; + + /* The first pass ensures the magnitude is 1, ... */ + t0 += x * 0x3D1UL; t1 += (x << 6); + t1 += (t0 >> 26); t0 &= 0x3FFFFFFUL; + t2 += (t1 >> 26); t1 &= 0x3FFFFFFUL; + t3 += (t2 >> 26); t2 &= 0x3FFFFFFUL; m = t2; + t4 += (t3 >> 26); t3 &= 0x3FFFFFFUL; m &= t3; + t5 += (t4 >> 26); t4 &= 0x3FFFFFFUL; m &= t4; + t6 += (t5 >> 26); t5 &= 0x3FFFFFFUL; m &= t5; + t7 += (t6 >> 26); t6 &= 0x3FFFFFFUL; m &= t6; + t8 += (t7 >> 26); t7 &= 0x3FFFFFFUL; m &= t7; + t9 += (t8 >> 26); t8 &= 0x3FFFFFFUL; m &= t8; + + /* ... except for a possible carry at bit 22 of t9 (i.e. bit 256 of the field element) */ + VERIFY_CHECK(t9 >> 23 == 0); + + /* At most a single final reduction is needed; check if the value is >= the field characteristic */ + x = (t9 >> 22) | ((t9 == 0x03FFFFFUL) & (m == 0x3FFFFFFUL) + & ((t1 + 0x40UL + ((t0 + 0x3D1UL) >> 26)) > 0x3FFFFFFUL)); + + if (x) { + t0 += 0x3D1UL; t1 += (x << 6); + t1 += (t0 >> 26); t0 &= 0x3FFFFFFUL; + t2 += (t1 >> 26); t1 &= 0x3FFFFFFUL; + t3 += (t2 >> 26); t2 &= 0x3FFFFFFUL; + t4 += (t3 >> 26); t3 &= 0x3FFFFFFUL; + t5 += (t4 >> 26); t4 &= 0x3FFFFFFUL; + t6 += (t5 >> 26); t5 &= 0x3FFFFFFUL; + t7 += (t6 >> 26); t6 &= 0x3FFFFFFUL; + t8 += (t7 >> 26); t7 &= 0x3FFFFFFUL; + t9 += (t8 >> 26); t8 &= 0x3FFFFFFUL; + + /* If t9 didn't carry to bit 22 already, then it should have after any final reduction */ + VERIFY_CHECK(t9 >> 22 == x); + + /* Mask off the possible multiple of 2^256 from the final reduction */ + t9 &= 0x03FFFFFUL; + } + + r->n[0] = t0; r->n[1] = t1; r->n[2] = t2; r->n[3] = t3; r->n[4] = t4; + r->n[5] = t5; r->n[6] = t6; r->n[7] = t7; r->n[8] = t8; r->n[9] = t9; + +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 1; + secp256k1_fe_verify(r); +#endif +} + +static int secp256k1_fe_normalizes_to_zero(secp256k1_fe *r) { + uint32_t t0 = r->n[0], t1 = r->n[1], t2 = r->n[2], t3 = r->n[3], t4 = r->n[4], + t5 = r->n[5], t6 = r->n[6], t7 = r->n[7], t8 = r->n[8], t9 = r->n[9]; + + /* z0 tracks a possible raw value of 0, z1 tracks a possible raw value of P */ + uint32_t z0, z1; + + /* Reduce t9 at the start so there will be at most a single carry from the first pass */ + uint32_t x = t9 >> 22; t9 &= 0x03FFFFFUL; + + /* The first pass ensures the magnitude is 1, ... */ + t0 += x * 0x3D1UL; t1 += (x << 6); + t1 += (t0 >> 26); t0 &= 0x3FFFFFFUL; z0 = t0; z1 = t0 ^ 0x3D0UL; + t2 += (t1 >> 26); t1 &= 0x3FFFFFFUL; z0 |= t1; z1 &= t1 ^ 0x40UL; + t3 += (t2 >> 26); t2 &= 0x3FFFFFFUL; z0 |= t2; z1 &= t2; + t4 += (t3 >> 26); t3 &= 0x3FFFFFFUL; z0 |= t3; z1 &= t3; + t5 += (t4 >> 26); t4 &= 0x3FFFFFFUL; z0 |= t4; z1 &= t4; + t6 += (t5 >> 26); t5 &= 0x3FFFFFFUL; z0 |= t5; z1 &= t5; + t7 += (t6 >> 26); t6 &= 0x3FFFFFFUL; z0 |= t6; z1 &= t6; + t8 += (t7 >> 26); t7 &= 0x3FFFFFFUL; z0 |= t7; z1 &= t7; + t9 += (t8 >> 26); t8 &= 0x3FFFFFFUL; z0 |= t8; z1 &= t8; + z0 |= t9; z1 &= t9 ^ 0x3C00000UL; + + /* ... except for a possible carry at bit 22 of t9 (i.e. bit 256 of the field element) */ + VERIFY_CHECK(t9 >> 23 == 0); + + return (z0 == 0) | (z1 == 0x3FFFFFFUL); +} + +static int secp256k1_fe_normalizes_to_zero_var(secp256k1_fe *r) { + uint32_t t0, t1, t2, t3, t4, t5, t6, t7, t8, t9; + uint32_t z0, z1; + uint32_t x; + + t0 = r->n[0]; + t9 = r->n[9]; + + /* Reduce t9 at the start so there will be at most a single carry from the first pass */ + x = t9 >> 22; + + /* The first pass ensures the magnitude is 1, ... */ + t0 += x * 0x3D1UL; + + /* z0 tracks a possible raw value of 0, z1 tracks a possible raw value of P */ + z0 = t0 & 0x3FFFFFFUL; + z1 = z0 ^ 0x3D0UL; + + /* Fast return path should catch the majority of cases */ + if ((z0 != 0UL) & (z1 != 0x3FFFFFFUL)) { + return 0; + } + + t1 = r->n[1]; + t2 = r->n[2]; + t3 = r->n[3]; + t4 = r->n[4]; + t5 = r->n[5]; + t6 = r->n[6]; + t7 = r->n[7]; + t8 = r->n[8]; + + t9 &= 0x03FFFFFUL; + t1 += (x << 6); + + t1 += (t0 >> 26); + t2 += (t1 >> 26); t1 &= 0x3FFFFFFUL; z0 |= t1; z1 &= t1 ^ 0x40UL; + t3 += (t2 >> 26); t2 &= 0x3FFFFFFUL; z0 |= t2; z1 &= t2; + t4 += (t3 >> 26); t3 &= 0x3FFFFFFUL; z0 |= t3; z1 &= t3; + t5 += (t4 >> 26); t4 &= 0x3FFFFFFUL; z0 |= t4; z1 &= t4; + t6 += (t5 >> 26); t5 &= 0x3FFFFFFUL; z0 |= t5; z1 &= t5; + t7 += (t6 >> 26); t6 &= 0x3FFFFFFUL; z0 |= t6; z1 &= t6; + t8 += (t7 >> 26); t7 &= 0x3FFFFFFUL; z0 |= t7; z1 &= t7; + t9 += (t8 >> 26); t8 &= 0x3FFFFFFUL; z0 |= t8; z1 &= t8; + z0 |= t9; z1 &= t9 ^ 0x3C00000UL; + + /* ... except for a possible carry at bit 22 of t9 (i.e. bit 256 of the field element) */ + VERIFY_CHECK(t9 >> 23 == 0); + + return (z0 == 0) | (z1 == 0x3FFFFFFUL); +} + +SECP256K1_INLINE static void secp256k1_fe_set_int(secp256k1_fe *r, int a) { + r->n[0] = a; + r->n[1] = r->n[2] = r->n[3] = r->n[4] = r->n[5] = r->n[6] = r->n[7] = r->n[8] = r->n[9] = 0; +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 1; + secp256k1_fe_verify(r); +#endif +} + +SECP256K1_INLINE static int secp256k1_fe_is_zero(const secp256k1_fe *a) { + const uint32_t *t = a->n; +#ifdef VERIFY + VERIFY_CHECK(a->normalized); + secp256k1_fe_verify(a); +#endif + return (t[0] | t[1] | t[2] | t[3] | t[4] | t[5] | t[6] | t[7] | t[8] | t[9]) == 0; +} + +SECP256K1_INLINE static int secp256k1_fe_is_odd(const secp256k1_fe *a) { +#ifdef VERIFY + VERIFY_CHECK(a->normalized); + secp256k1_fe_verify(a); +#endif + return a->n[0] & 1; +} + +SECP256K1_INLINE static void secp256k1_fe_clear(secp256k1_fe *a) { + int i; +#ifdef VERIFY + a->magnitude = 0; + a->normalized = 1; +#endif + for (i=0; i<10; i++) { + a->n[i] = 0; + } +} + +static int secp256k1_fe_cmp_var(const secp256k1_fe *a, const secp256k1_fe *b) { + int i; +#ifdef VERIFY + VERIFY_CHECK(a->normalized); + VERIFY_CHECK(b->normalized); + secp256k1_fe_verify(a); + secp256k1_fe_verify(b); +#endif + for (i = 9; i >= 0; i--) { + if (a->n[i] > b->n[i]) { + return 1; + } + if (a->n[i] < b->n[i]) { + return -1; + } + } + return 0; +} + +static int secp256k1_fe_set_b32(secp256k1_fe *r, const unsigned char *a) { + r->n[0] = (uint32_t)a[31] | ((uint32_t)a[30] << 8) | ((uint32_t)a[29] << 16) | ((uint32_t)(a[28] & 0x3) << 24); + r->n[1] = (uint32_t)((a[28] >> 2) & 0x3f) | ((uint32_t)a[27] << 6) | ((uint32_t)a[26] << 14) | ((uint32_t)(a[25] & 0xf) << 22); + r->n[2] = (uint32_t)((a[25] >> 4) & 0xf) | ((uint32_t)a[24] << 4) | ((uint32_t)a[23] << 12) | ((uint32_t)(a[22] & 0x3f) << 20); + r->n[3] = (uint32_t)((a[22] >> 6) & 0x3) | ((uint32_t)a[21] << 2) | ((uint32_t)a[20] << 10) | ((uint32_t)a[19] << 18); + r->n[4] = (uint32_t)a[18] | ((uint32_t)a[17] << 8) | ((uint32_t)a[16] << 16) | ((uint32_t)(a[15] & 0x3) << 24); + r->n[5] = (uint32_t)((a[15] >> 2) & 0x3f) | ((uint32_t)a[14] << 6) | ((uint32_t)a[13] << 14) | ((uint32_t)(a[12] & 0xf) << 22); + r->n[6] = (uint32_t)((a[12] >> 4) & 0xf) | ((uint32_t)a[11] << 4) | ((uint32_t)a[10] << 12) | ((uint32_t)(a[9] & 0x3f) << 20); + r->n[7] = (uint32_t)((a[9] >> 6) & 0x3) | ((uint32_t)a[8] << 2) | ((uint32_t)a[7] << 10) | ((uint32_t)a[6] << 18); + r->n[8] = (uint32_t)a[5] | ((uint32_t)a[4] << 8) | ((uint32_t)a[3] << 16) | ((uint32_t)(a[2] & 0x3) << 24); + r->n[9] = (uint32_t)((a[2] >> 2) & 0x3f) | ((uint32_t)a[1] << 6) | ((uint32_t)a[0] << 14); + + if (r->n[9] == 0x3FFFFFUL && (r->n[8] & r->n[7] & r->n[6] & r->n[5] & r->n[4] & r->n[3] & r->n[2]) == 0x3FFFFFFUL && (r->n[1] + 0x40UL + ((r->n[0] + 0x3D1UL) >> 26)) > 0x3FFFFFFUL) { + return 0; + } +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 1; + secp256k1_fe_verify(r); +#endif + return 1; +} + +/** Convert a field element to a 32-byte big endian value. Requires the input to be normalized */ +static void secp256k1_fe_get_b32(unsigned char *r, const secp256k1_fe *a) { +#ifdef VERIFY + VERIFY_CHECK(a->normalized); + secp256k1_fe_verify(a); +#endif + r[0] = (a->n[9] >> 14) & 0xff; + r[1] = (a->n[9] >> 6) & 0xff; + r[2] = ((a->n[9] & 0x3F) << 2) | ((a->n[8] >> 24) & 0x3); + r[3] = (a->n[8] >> 16) & 0xff; + r[4] = (a->n[8] >> 8) & 0xff; + r[5] = a->n[8] & 0xff; + r[6] = (a->n[7] >> 18) & 0xff; + r[7] = (a->n[7] >> 10) & 0xff; + r[8] = (a->n[7] >> 2) & 0xff; + r[9] = ((a->n[7] & 0x3) << 6) | ((a->n[6] >> 20) & 0x3f); + r[10] = (a->n[6] >> 12) & 0xff; + r[11] = (a->n[6] >> 4) & 0xff; + r[12] = ((a->n[6] & 0xf) << 4) | ((a->n[5] >> 22) & 0xf); + r[13] = (a->n[5] >> 14) & 0xff; + r[14] = (a->n[5] >> 6) & 0xff; + r[15] = ((a->n[5] & 0x3f) << 2) | ((a->n[4] >> 24) & 0x3); + r[16] = (a->n[4] >> 16) & 0xff; + r[17] = (a->n[4] >> 8) & 0xff; + r[18] = a->n[4] & 0xff; + r[19] = (a->n[3] >> 18) & 0xff; + r[20] = (a->n[3] >> 10) & 0xff; + r[21] = (a->n[3] >> 2) & 0xff; + r[22] = ((a->n[3] & 0x3) << 6) | ((a->n[2] >> 20) & 0x3f); + r[23] = (a->n[2] >> 12) & 0xff; + r[24] = (a->n[2] >> 4) & 0xff; + r[25] = ((a->n[2] & 0xf) << 4) | ((a->n[1] >> 22) & 0xf); + r[26] = (a->n[1] >> 14) & 0xff; + r[27] = (a->n[1] >> 6) & 0xff; + r[28] = ((a->n[1] & 0x3f) << 2) | ((a->n[0] >> 24) & 0x3); + r[29] = (a->n[0] >> 16) & 0xff; + r[30] = (a->n[0] >> 8) & 0xff; + r[31] = a->n[0] & 0xff; +} + +SECP256K1_INLINE static void secp256k1_fe_negate(secp256k1_fe *r, const secp256k1_fe *a, int m) { +#ifdef VERIFY + VERIFY_CHECK(a->magnitude <= m); + secp256k1_fe_verify(a); +#endif + r->n[0] = 0x3FFFC2FUL * 2 * (m + 1) - a->n[0]; + r->n[1] = 0x3FFFFBFUL * 2 * (m + 1) - a->n[1]; + r->n[2] = 0x3FFFFFFUL * 2 * (m + 1) - a->n[2]; + r->n[3] = 0x3FFFFFFUL * 2 * (m + 1) - a->n[3]; + r->n[4] = 0x3FFFFFFUL * 2 * (m + 1) - a->n[4]; + r->n[5] = 0x3FFFFFFUL * 2 * (m + 1) - a->n[5]; + r->n[6] = 0x3FFFFFFUL * 2 * (m + 1) - a->n[6]; + r->n[7] = 0x3FFFFFFUL * 2 * (m + 1) - a->n[7]; + r->n[8] = 0x3FFFFFFUL * 2 * (m + 1) - a->n[8]; + r->n[9] = 0x03FFFFFUL * 2 * (m + 1) - a->n[9]; +#ifdef VERIFY + r->magnitude = m + 1; + r->normalized = 0; + secp256k1_fe_verify(r); +#endif +} + +SECP256K1_INLINE static void secp256k1_fe_mul_int(secp256k1_fe *r, int a) { + r->n[0] *= a; + r->n[1] *= a; + r->n[2] *= a; + r->n[3] *= a; + r->n[4] *= a; + r->n[5] *= a; + r->n[6] *= a; + r->n[7] *= a; + r->n[8] *= a; + r->n[9] *= a; +#ifdef VERIFY + r->magnitude *= a; + r->normalized = 0; + secp256k1_fe_verify(r); +#endif +} + +SECP256K1_INLINE static void secp256k1_fe_add(secp256k1_fe *r, const secp256k1_fe *a) { +#ifdef VERIFY + secp256k1_fe_verify(a); +#endif + r->n[0] += a->n[0]; + r->n[1] += a->n[1]; + r->n[2] += a->n[2]; + r->n[3] += a->n[3]; + r->n[4] += a->n[4]; + r->n[5] += a->n[5]; + r->n[6] += a->n[6]; + r->n[7] += a->n[7]; + r->n[8] += a->n[8]; + r->n[9] += a->n[9]; +#ifdef VERIFY + r->magnitude += a->magnitude; + r->normalized = 0; + secp256k1_fe_verify(r); +#endif +} + +#if defined(USE_EXTERNAL_ASM) + +/* External assembler implementation */ +void secp256k1_fe_mul_inner(uint32_t *r, const uint32_t *a, const uint32_t * SECP256K1_RESTRICT b); +void secp256k1_fe_sqr_inner(uint32_t *r, const uint32_t *a); + +#else + +#ifdef VERIFY +#define VERIFY_BITS(x, n) VERIFY_CHECK(((x) >> (n)) == 0) +#else +#define VERIFY_BITS(x, n) do { } while(0) +#endif + +SECP256K1_INLINE static void secp256k1_fe_mul_inner(uint32_t *r, const uint32_t *a, const uint32_t * SECP256K1_RESTRICT b) { + uint64_t c, d; + uint64_t u0, u1, u2, u3, u4, u5, u6, u7, u8; + uint32_t t9, t1, t0, t2, t3, t4, t5, t6, t7; + const uint32_t M = 0x3FFFFFFUL, R0 = 0x3D10UL, R1 = 0x400UL; + + VERIFY_BITS(a[0], 30); + VERIFY_BITS(a[1], 30); + VERIFY_BITS(a[2], 30); + VERIFY_BITS(a[3], 30); + VERIFY_BITS(a[4], 30); + VERIFY_BITS(a[5], 30); + VERIFY_BITS(a[6], 30); + VERIFY_BITS(a[7], 30); + VERIFY_BITS(a[8], 30); + VERIFY_BITS(a[9], 26); + VERIFY_BITS(b[0], 30); + VERIFY_BITS(b[1], 30); + VERIFY_BITS(b[2], 30); + VERIFY_BITS(b[3], 30); + VERIFY_BITS(b[4], 30); + VERIFY_BITS(b[5], 30); + VERIFY_BITS(b[6], 30); + VERIFY_BITS(b[7], 30); + VERIFY_BITS(b[8], 30); + VERIFY_BITS(b[9], 26); + + /** [... a b c] is a shorthand for ... + a<<52 + b<<26 + c<<0 mod n. + * for 0 <= x <= 9, px is a shorthand for sum(a[i]*b[x-i], i=0..x). + * for 9 <= x <= 18, px is a shorthand for sum(a[i]*b[x-i], i=(x-9)..9) + * Note that [x 0 0 0 0 0 0 0 0 0 0] = [x*R1 x*R0]. + */ + + d = (uint64_t)a[0] * b[9] + + (uint64_t)a[1] * b[8] + + (uint64_t)a[2] * b[7] + + (uint64_t)a[3] * b[6] + + (uint64_t)a[4] * b[5] + + (uint64_t)a[5] * b[4] + + (uint64_t)a[6] * b[3] + + (uint64_t)a[7] * b[2] + + (uint64_t)a[8] * b[1] + + (uint64_t)a[9] * b[0]; + /* VERIFY_BITS(d, 64); */ + /* [d 0 0 0 0 0 0 0 0 0] = [p9 0 0 0 0 0 0 0 0 0] */ + t9 = d & M; d >>= 26; + VERIFY_BITS(t9, 26); + VERIFY_BITS(d, 38); + /* [d t9 0 0 0 0 0 0 0 0 0] = [p9 0 0 0 0 0 0 0 0 0] */ + + c = (uint64_t)a[0] * b[0]; + VERIFY_BITS(c, 60); + /* [d t9 0 0 0 0 0 0 0 0 c] = [p9 0 0 0 0 0 0 0 0 p0] */ + d += (uint64_t)a[1] * b[9] + + (uint64_t)a[2] * b[8] + + (uint64_t)a[3] * b[7] + + (uint64_t)a[4] * b[6] + + (uint64_t)a[5] * b[5] + + (uint64_t)a[6] * b[4] + + (uint64_t)a[7] * b[3] + + (uint64_t)a[8] * b[2] + + (uint64_t)a[9] * b[1]; + VERIFY_BITS(d, 63); + /* [d t9 0 0 0 0 0 0 0 0 c] = [p10 p9 0 0 0 0 0 0 0 0 p0] */ + u0 = d & M; d >>= 26; c += u0 * R0; + VERIFY_BITS(u0, 26); + VERIFY_BITS(d, 37); + VERIFY_BITS(c, 61); + /* [d u0 t9 0 0 0 0 0 0 0 0 c-u0*R0] = [p10 p9 0 0 0 0 0 0 0 0 p0] */ + t0 = c & M; c >>= 26; c += u0 * R1; + VERIFY_BITS(t0, 26); + VERIFY_BITS(c, 37); + /* [d u0 t9 0 0 0 0 0 0 0 c-u0*R1 t0-u0*R0] = [p10 p9 0 0 0 0 0 0 0 0 p0] */ + /* [d 0 t9 0 0 0 0 0 0 0 c t0] = [p10 p9 0 0 0 0 0 0 0 0 p0] */ + + c += (uint64_t)a[0] * b[1] + + (uint64_t)a[1] * b[0]; + VERIFY_BITS(c, 62); + /* [d 0 t9 0 0 0 0 0 0 0 c t0] = [p10 p9 0 0 0 0 0 0 0 p1 p0] */ + d += (uint64_t)a[2] * b[9] + + (uint64_t)a[3] * b[8] + + (uint64_t)a[4] * b[7] + + (uint64_t)a[5] * b[6] + + (uint64_t)a[6] * b[5] + + (uint64_t)a[7] * b[4] + + (uint64_t)a[8] * b[3] + + (uint64_t)a[9] * b[2]; + VERIFY_BITS(d, 63); + /* [d 0 t9 0 0 0 0 0 0 0 c t0] = [p11 p10 p9 0 0 0 0 0 0 0 p1 p0] */ + u1 = d & M; d >>= 26; c += u1 * R0; + VERIFY_BITS(u1, 26); + VERIFY_BITS(d, 37); + VERIFY_BITS(c, 63); + /* [d u1 0 t9 0 0 0 0 0 0 0 c-u1*R0 t0] = [p11 p10 p9 0 0 0 0 0 0 0 p1 p0] */ + t1 = c & M; c >>= 26; c += u1 * R1; + VERIFY_BITS(t1, 26); + VERIFY_BITS(c, 38); + /* [d u1 0 t9 0 0 0 0 0 0 c-u1*R1 t1-u1*R0 t0] = [p11 p10 p9 0 0 0 0 0 0 0 p1 p0] */ + /* [d 0 0 t9 0 0 0 0 0 0 c t1 t0] = [p11 p10 p9 0 0 0 0 0 0 0 p1 p0] */ + + c += (uint64_t)a[0] * b[2] + + (uint64_t)a[1] * b[1] + + (uint64_t)a[2] * b[0]; + VERIFY_BITS(c, 62); + /* [d 0 0 t9 0 0 0 0 0 0 c t1 t0] = [p11 p10 p9 0 0 0 0 0 0 p2 p1 p0] */ + d += (uint64_t)a[3] * b[9] + + (uint64_t)a[4] * b[8] + + (uint64_t)a[5] * b[7] + + (uint64_t)a[6] * b[6] + + (uint64_t)a[7] * b[5] + + (uint64_t)a[8] * b[4] + + (uint64_t)a[9] * b[3]; + VERIFY_BITS(d, 63); + /* [d 0 0 t9 0 0 0 0 0 0 c t1 t0] = [p12 p11 p10 p9 0 0 0 0 0 0 p2 p1 p0] */ + u2 = d & M; d >>= 26; c += u2 * R0; + VERIFY_BITS(u2, 26); + VERIFY_BITS(d, 37); + VERIFY_BITS(c, 63); + /* [d u2 0 0 t9 0 0 0 0 0 0 c-u2*R0 t1 t0] = [p12 p11 p10 p9 0 0 0 0 0 0 p2 p1 p0] */ + t2 = c & M; c >>= 26; c += u2 * R1; + VERIFY_BITS(t2, 26); + VERIFY_BITS(c, 38); + /* [d u2 0 0 t9 0 0 0 0 0 c-u2*R1 t2-u2*R0 t1 t0] = [p12 p11 p10 p9 0 0 0 0 0 0 p2 p1 p0] */ + /* [d 0 0 0 t9 0 0 0 0 0 c t2 t1 t0] = [p12 p11 p10 p9 0 0 0 0 0 0 p2 p1 p0] */ + + c += (uint64_t)a[0] * b[3] + + (uint64_t)a[1] * b[2] + + (uint64_t)a[2] * b[1] + + (uint64_t)a[3] * b[0]; + VERIFY_BITS(c, 63); + /* [d 0 0 0 t9 0 0 0 0 0 c t2 t1 t0] = [p12 p11 p10 p9 0 0 0 0 0 p3 p2 p1 p0] */ + d += (uint64_t)a[4] * b[9] + + (uint64_t)a[5] * b[8] + + (uint64_t)a[6] * b[7] + + (uint64_t)a[7] * b[6] + + (uint64_t)a[8] * b[5] + + (uint64_t)a[9] * b[4]; + VERIFY_BITS(d, 63); + /* [d 0 0 0 t9 0 0 0 0 0 c t2 t1 t0] = [p13 p12 p11 p10 p9 0 0 0 0 0 p3 p2 p1 p0] */ + u3 = d & M; d >>= 26; c += u3 * R0; + VERIFY_BITS(u3, 26); + VERIFY_BITS(d, 37); + /* VERIFY_BITS(c, 64); */ + /* [d u3 0 0 0 t9 0 0 0 0 0 c-u3*R0 t2 t1 t0] = [p13 p12 p11 p10 p9 0 0 0 0 0 p3 p2 p1 p0] */ + t3 = c & M; c >>= 26; c += u3 * R1; + VERIFY_BITS(t3, 26); + VERIFY_BITS(c, 39); + /* [d u3 0 0 0 t9 0 0 0 0 c-u3*R1 t3-u3*R0 t2 t1 t0] = [p13 p12 p11 p10 p9 0 0 0 0 0 p3 p2 p1 p0] */ + /* [d 0 0 0 0 t9 0 0 0 0 c t3 t2 t1 t0] = [p13 p12 p11 p10 p9 0 0 0 0 0 p3 p2 p1 p0] */ + + c += (uint64_t)a[0] * b[4] + + (uint64_t)a[1] * b[3] + + (uint64_t)a[2] * b[2] + + (uint64_t)a[3] * b[1] + + (uint64_t)a[4] * b[0]; + VERIFY_BITS(c, 63); + /* [d 0 0 0 0 t9 0 0 0 0 c t3 t2 t1 t0] = [p13 p12 p11 p10 p9 0 0 0 0 p4 p3 p2 p1 p0] */ + d += (uint64_t)a[5] * b[9] + + (uint64_t)a[6] * b[8] + + (uint64_t)a[7] * b[7] + + (uint64_t)a[8] * b[6] + + (uint64_t)a[9] * b[5]; + VERIFY_BITS(d, 62); + /* [d 0 0 0 0 t9 0 0 0 0 c t3 t2 t1 t0] = [p14 p13 p12 p11 p10 p9 0 0 0 0 p4 p3 p2 p1 p0] */ + u4 = d & M; d >>= 26; c += u4 * R0; + VERIFY_BITS(u4, 26); + VERIFY_BITS(d, 36); + /* VERIFY_BITS(c, 64); */ + /* [d u4 0 0 0 0 t9 0 0 0 0 c-u4*R0 t3 t2 t1 t0] = [p14 p13 p12 p11 p10 p9 0 0 0 0 p4 p3 p2 p1 p0] */ + t4 = c & M; c >>= 26; c += u4 * R1; + VERIFY_BITS(t4, 26); + VERIFY_BITS(c, 39); + /* [d u4 0 0 0 0 t9 0 0 0 c-u4*R1 t4-u4*R0 t3 t2 t1 t0] = [p14 p13 p12 p11 p10 p9 0 0 0 0 p4 p3 p2 p1 p0] */ + /* [d 0 0 0 0 0 t9 0 0 0 c t4 t3 t2 t1 t0] = [p14 p13 p12 p11 p10 p9 0 0 0 0 p4 p3 p2 p1 p0] */ + + c += (uint64_t)a[0] * b[5] + + (uint64_t)a[1] * b[4] + + (uint64_t)a[2] * b[3] + + (uint64_t)a[3] * b[2] + + (uint64_t)a[4] * b[1] + + (uint64_t)a[5] * b[0]; + VERIFY_BITS(c, 63); + /* [d 0 0 0 0 0 t9 0 0 0 c t4 t3 t2 t1 t0] = [p14 p13 p12 p11 p10 p9 0 0 0 p5 p4 p3 p2 p1 p0] */ + d += (uint64_t)a[6] * b[9] + + (uint64_t)a[7] * b[8] + + (uint64_t)a[8] * b[7] + + (uint64_t)a[9] * b[6]; + VERIFY_BITS(d, 62); + /* [d 0 0 0 0 0 t9 0 0 0 c t4 t3 t2 t1 t0] = [p15 p14 p13 p12 p11 p10 p9 0 0 0 p5 p4 p3 p2 p1 p0] */ + u5 = d & M; d >>= 26; c += u5 * R0; + VERIFY_BITS(u5, 26); + VERIFY_BITS(d, 36); + /* VERIFY_BITS(c, 64); */ + /* [d u5 0 0 0 0 0 t9 0 0 0 c-u5*R0 t4 t3 t2 t1 t0] = [p15 p14 p13 p12 p11 p10 p9 0 0 0 p5 p4 p3 p2 p1 p0] */ + t5 = c & M; c >>= 26; c += u5 * R1; + VERIFY_BITS(t5, 26); + VERIFY_BITS(c, 39); + /* [d u5 0 0 0 0 0 t9 0 0 c-u5*R1 t5-u5*R0 t4 t3 t2 t1 t0] = [p15 p14 p13 p12 p11 p10 p9 0 0 0 p5 p4 p3 p2 p1 p0] */ + /* [d 0 0 0 0 0 0 t9 0 0 c t5 t4 t3 t2 t1 t0] = [p15 p14 p13 p12 p11 p10 p9 0 0 0 p5 p4 p3 p2 p1 p0] */ + + c += (uint64_t)a[0] * b[6] + + (uint64_t)a[1] * b[5] + + (uint64_t)a[2] * b[4] + + (uint64_t)a[3] * b[3] + + (uint64_t)a[4] * b[2] + + (uint64_t)a[5] * b[1] + + (uint64_t)a[6] * b[0]; + VERIFY_BITS(c, 63); + /* [d 0 0 0 0 0 0 t9 0 0 c t5 t4 t3 t2 t1 t0] = [p15 p14 p13 p12 p11 p10 p9 0 0 p6 p5 p4 p3 p2 p1 p0] */ + d += (uint64_t)a[7] * b[9] + + (uint64_t)a[8] * b[8] + + (uint64_t)a[9] * b[7]; + VERIFY_BITS(d, 61); + /* [d 0 0 0 0 0 0 t9 0 0 c t5 t4 t3 t2 t1 t0] = [p16 p15 p14 p13 p12 p11 p10 p9 0 0 p6 p5 p4 p3 p2 p1 p0] */ + u6 = d & M; d >>= 26; c += u6 * R0; + VERIFY_BITS(u6, 26); + VERIFY_BITS(d, 35); + /* VERIFY_BITS(c, 64); */ + /* [d u6 0 0 0 0 0 0 t9 0 0 c-u6*R0 t5 t4 t3 t2 t1 t0] = [p16 p15 p14 p13 p12 p11 p10 p9 0 0 p6 p5 p4 p3 p2 p1 p0] */ + t6 = c & M; c >>= 26; c += u6 * R1; + VERIFY_BITS(t6, 26); + VERIFY_BITS(c, 39); + /* [d u6 0 0 0 0 0 0 t9 0 c-u6*R1 t6-u6*R0 t5 t4 t3 t2 t1 t0] = [p16 p15 p14 p13 p12 p11 p10 p9 0 0 p6 p5 p4 p3 p2 p1 p0] */ + /* [d 0 0 0 0 0 0 0 t9 0 c t6 t5 t4 t3 t2 t1 t0] = [p16 p15 p14 p13 p12 p11 p10 p9 0 0 p6 p5 p4 p3 p2 p1 p0] */ + + c += (uint64_t)a[0] * b[7] + + (uint64_t)a[1] * b[6] + + (uint64_t)a[2] * b[5] + + (uint64_t)a[3] * b[4] + + (uint64_t)a[4] * b[3] + + (uint64_t)a[5] * b[2] + + (uint64_t)a[6] * b[1] + + (uint64_t)a[7] * b[0]; + /* VERIFY_BITS(c, 64); */ + VERIFY_CHECK(c <= 0x8000007C00000007ULL); + /* [d 0 0 0 0 0 0 0 t9 0 c t6 t5 t4 t3 t2 t1 t0] = [p16 p15 p14 p13 p12 p11 p10 p9 0 p7 p6 p5 p4 p3 p2 p1 p0] */ + d += (uint64_t)a[8] * b[9] + + (uint64_t)a[9] * b[8]; + VERIFY_BITS(d, 58); + /* [d 0 0 0 0 0 0 0 t9 0 c t6 t5 t4 t3 t2 t1 t0] = [p17 p16 p15 p14 p13 p12 p11 p10 p9 0 p7 p6 p5 p4 p3 p2 p1 p0] */ + u7 = d & M; d >>= 26; c += u7 * R0; + VERIFY_BITS(u7, 26); + VERIFY_BITS(d, 32); + /* VERIFY_BITS(c, 64); */ + VERIFY_CHECK(c <= 0x800001703FFFC2F7ULL); + /* [d u7 0 0 0 0 0 0 0 t9 0 c-u7*R0 t6 t5 t4 t3 t2 t1 t0] = [p17 p16 p15 p14 p13 p12 p11 p10 p9 0 p7 p6 p5 p4 p3 p2 p1 p0] */ + t7 = c & M; c >>= 26; c += u7 * R1; + VERIFY_BITS(t7, 26); + VERIFY_BITS(c, 38); + /* [d u7 0 0 0 0 0 0 0 t9 c-u7*R1 t7-u7*R0 t6 t5 t4 t3 t2 t1 t0] = [p17 p16 p15 p14 p13 p12 p11 p10 p9 0 p7 p6 p5 p4 p3 p2 p1 p0] */ + /* [d 0 0 0 0 0 0 0 0 t9 c t7 t6 t5 t4 t3 t2 t1 t0] = [p17 p16 p15 p14 p13 p12 p11 p10 p9 0 p7 p6 p5 p4 p3 p2 p1 p0] */ + + c += (uint64_t)a[0] * b[8] + + (uint64_t)a[1] * b[7] + + (uint64_t)a[2] * b[6] + + (uint64_t)a[3] * b[5] + + (uint64_t)a[4] * b[4] + + (uint64_t)a[5] * b[3] + + (uint64_t)a[6] * b[2] + + (uint64_t)a[7] * b[1] + + (uint64_t)a[8] * b[0]; + /* VERIFY_BITS(c, 64); */ + VERIFY_CHECK(c <= 0x9000007B80000008ULL); + /* [d 0 0 0 0 0 0 0 0 t9 c t7 t6 t5 t4 t3 t2 t1 t0] = [p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + d += (uint64_t)a[9] * b[9]; + VERIFY_BITS(d, 57); + /* [d 0 0 0 0 0 0 0 0 t9 c t7 t6 t5 t4 t3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + u8 = d & M; d >>= 26; c += u8 * R0; + VERIFY_BITS(u8, 26); + VERIFY_BITS(d, 31); + /* VERIFY_BITS(c, 64); */ + VERIFY_CHECK(c <= 0x9000016FBFFFC2F8ULL); + /* [d u8 0 0 0 0 0 0 0 0 t9 c-u8*R0 t7 t6 t5 t4 t3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + + r[3] = t3; + VERIFY_BITS(r[3], 26); + /* [d u8 0 0 0 0 0 0 0 0 t9 c-u8*R0 t7 t6 t5 t4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[4] = t4; + VERIFY_BITS(r[4], 26); + /* [d u8 0 0 0 0 0 0 0 0 t9 c-u8*R0 t7 t6 t5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[5] = t5; + VERIFY_BITS(r[5], 26); + /* [d u8 0 0 0 0 0 0 0 0 t9 c-u8*R0 t7 t6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[6] = t6; + VERIFY_BITS(r[6], 26); + /* [d u8 0 0 0 0 0 0 0 0 t9 c-u8*R0 t7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[7] = t7; + VERIFY_BITS(r[7], 26); + /* [d u8 0 0 0 0 0 0 0 0 t9 c-u8*R0 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + + r[8] = c & M; c >>= 26; c += u8 * R1; + VERIFY_BITS(r[8], 26); + VERIFY_BITS(c, 39); + /* [d u8 0 0 0 0 0 0 0 0 t9+c-u8*R1 r8-u8*R0 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + /* [d 0 0 0 0 0 0 0 0 0 t9+c r8 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + c += d * R0 + t9; + VERIFY_BITS(c, 45); + /* [d 0 0 0 0 0 0 0 0 0 c-d*R0 r8 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[9] = c & (M >> 4); c >>= 22; c += d * (R1 << 4); + VERIFY_BITS(r[9], 22); + VERIFY_BITS(c, 46); + /* [d 0 0 0 0 0 0 0 0 r9+((c-d*R1<<4)<<22)-d*R0 r8 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + /* [d 0 0 0 0 0 0 0 -d*R1 r9+(c<<22)-d*R0 r8 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + /* [r9+(c<<22) r8 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + + d = c * (R0 >> 4) + t0; + VERIFY_BITS(d, 56); + /* [r9+(c<<22) r8 r7 r6 r5 r4 r3 t2 t1 d-c*R0>>4] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[0] = d & M; d >>= 26; + VERIFY_BITS(r[0], 26); + VERIFY_BITS(d, 30); + /* [r9+(c<<22) r8 r7 r6 r5 r4 r3 t2 t1+d r0-c*R0>>4] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + d += c * (R1 >> 4) + t1; + VERIFY_BITS(d, 53); + VERIFY_CHECK(d <= 0x10000003FFFFBFULL); + /* [r9+(c<<22) r8 r7 r6 r5 r4 r3 t2 d-c*R1>>4 r0-c*R0>>4] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + /* [r9 r8 r7 r6 r5 r4 r3 t2 d r0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[1] = d & M; d >>= 26; + VERIFY_BITS(r[1], 26); + VERIFY_BITS(d, 27); + VERIFY_CHECK(d <= 0x4000000ULL); + /* [r9 r8 r7 r6 r5 r4 r3 t2+d r1 r0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + d += t2; + VERIFY_BITS(d, 27); + /* [r9 r8 r7 r6 r5 r4 r3 d r1 r0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[2] = d; + VERIFY_BITS(r[2], 27); + /* [r9 r8 r7 r6 r5 r4 r3 r2 r1 r0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ +} + +SECP256K1_INLINE static void secp256k1_fe_sqr_inner(uint32_t *r, const uint32_t *a) { + uint64_t c, d; + uint64_t u0, u1, u2, u3, u4, u5, u6, u7, u8; + uint32_t t9, t0, t1, t2, t3, t4, t5, t6, t7; + const uint32_t M = 0x3FFFFFFUL, R0 = 0x3D10UL, R1 = 0x400UL; + + VERIFY_BITS(a[0], 30); + VERIFY_BITS(a[1], 30); + VERIFY_BITS(a[2], 30); + VERIFY_BITS(a[3], 30); + VERIFY_BITS(a[4], 30); + VERIFY_BITS(a[5], 30); + VERIFY_BITS(a[6], 30); + VERIFY_BITS(a[7], 30); + VERIFY_BITS(a[8], 30); + VERIFY_BITS(a[9], 26); + + /** [... a b c] is a shorthand for ... + a<<52 + b<<26 + c<<0 mod n. + * px is a shorthand for sum(a[i]*a[x-i], i=0..x). + * Note that [x 0 0 0 0 0 0 0 0 0 0] = [x*R1 x*R0]. + */ + + d = (uint64_t)(a[0]*2) * a[9] + + (uint64_t)(a[1]*2) * a[8] + + (uint64_t)(a[2]*2) * a[7] + + (uint64_t)(a[3]*2) * a[6] + + (uint64_t)(a[4]*2) * a[5]; + /* VERIFY_BITS(d, 64); */ + /* [d 0 0 0 0 0 0 0 0 0] = [p9 0 0 0 0 0 0 0 0 0] */ + t9 = d & M; d >>= 26; + VERIFY_BITS(t9, 26); + VERIFY_BITS(d, 38); + /* [d t9 0 0 0 0 0 0 0 0 0] = [p9 0 0 0 0 0 0 0 0 0] */ + + c = (uint64_t)a[0] * a[0]; + VERIFY_BITS(c, 60); + /* [d t9 0 0 0 0 0 0 0 0 c] = [p9 0 0 0 0 0 0 0 0 p0] */ + d += (uint64_t)(a[1]*2) * a[9] + + (uint64_t)(a[2]*2) * a[8] + + (uint64_t)(a[3]*2) * a[7] + + (uint64_t)(a[4]*2) * a[6] + + (uint64_t)a[5] * a[5]; + VERIFY_BITS(d, 63); + /* [d t9 0 0 0 0 0 0 0 0 c] = [p10 p9 0 0 0 0 0 0 0 0 p0] */ + u0 = d & M; d >>= 26; c += u0 * R0; + VERIFY_BITS(u0, 26); + VERIFY_BITS(d, 37); + VERIFY_BITS(c, 61); + /* [d u0 t9 0 0 0 0 0 0 0 0 c-u0*R0] = [p10 p9 0 0 0 0 0 0 0 0 p0] */ + t0 = c & M; c >>= 26; c += u0 * R1; + VERIFY_BITS(t0, 26); + VERIFY_BITS(c, 37); + /* [d u0 t9 0 0 0 0 0 0 0 c-u0*R1 t0-u0*R0] = [p10 p9 0 0 0 0 0 0 0 0 p0] */ + /* [d 0 t9 0 0 0 0 0 0 0 c t0] = [p10 p9 0 0 0 0 0 0 0 0 p0] */ + + c += (uint64_t)(a[0]*2) * a[1]; + VERIFY_BITS(c, 62); + /* [d 0 t9 0 0 0 0 0 0 0 c t0] = [p10 p9 0 0 0 0 0 0 0 p1 p0] */ + d += (uint64_t)(a[2]*2) * a[9] + + (uint64_t)(a[3]*2) * a[8] + + (uint64_t)(a[4]*2) * a[7] + + (uint64_t)(a[5]*2) * a[6]; + VERIFY_BITS(d, 63); + /* [d 0 t9 0 0 0 0 0 0 0 c t0] = [p11 p10 p9 0 0 0 0 0 0 0 p1 p0] */ + u1 = d & M; d >>= 26; c += u1 * R0; + VERIFY_BITS(u1, 26); + VERIFY_BITS(d, 37); + VERIFY_BITS(c, 63); + /* [d u1 0 t9 0 0 0 0 0 0 0 c-u1*R0 t0] = [p11 p10 p9 0 0 0 0 0 0 0 p1 p0] */ + t1 = c & M; c >>= 26; c += u1 * R1; + VERIFY_BITS(t1, 26); + VERIFY_BITS(c, 38); + /* [d u1 0 t9 0 0 0 0 0 0 c-u1*R1 t1-u1*R0 t0] = [p11 p10 p9 0 0 0 0 0 0 0 p1 p0] */ + /* [d 0 0 t9 0 0 0 0 0 0 c t1 t0] = [p11 p10 p9 0 0 0 0 0 0 0 p1 p0] */ + + c += (uint64_t)(a[0]*2) * a[2] + + (uint64_t)a[1] * a[1]; + VERIFY_BITS(c, 62); + /* [d 0 0 t9 0 0 0 0 0 0 c t1 t0] = [p11 p10 p9 0 0 0 0 0 0 p2 p1 p0] */ + d += (uint64_t)(a[3]*2) * a[9] + + (uint64_t)(a[4]*2) * a[8] + + (uint64_t)(a[5]*2) * a[7] + + (uint64_t)a[6] * a[6]; + VERIFY_BITS(d, 63); + /* [d 0 0 t9 0 0 0 0 0 0 c t1 t0] = [p12 p11 p10 p9 0 0 0 0 0 0 p2 p1 p0] */ + u2 = d & M; d >>= 26; c += u2 * R0; + VERIFY_BITS(u2, 26); + VERIFY_BITS(d, 37); + VERIFY_BITS(c, 63); + /* [d u2 0 0 t9 0 0 0 0 0 0 c-u2*R0 t1 t0] = [p12 p11 p10 p9 0 0 0 0 0 0 p2 p1 p0] */ + t2 = c & M; c >>= 26; c += u2 * R1; + VERIFY_BITS(t2, 26); + VERIFY_BITS(c, 38); + /* [d u2 0 0 t9 0 0 0 0 0 c-u2*R1 t2-u2*R0 t1 t0] = [p12 p11 p10 p9 0 0 0 0 0 0 p2 p1 p0] */ + /* [d 0 0 0 t9 0 0 0 0 0 c t2 t1 t0] = [p12 p11 p10 p9 0 0 0 0 0 0 p2 p1 p0] */ + + c += (uint64_t)(a[0]*2) * a[3] + + (uint64_t)(a[1]*2) * a[2]; + VERIFY_BITS(c, 63); + /* [d 0 0 0 t9 0 0 0 0 0 c t2 t1 t0] = [p12 p11 p10 p9 0 0 0 0 0 p3 p2 p1 p0] */ + d += (uint64_t)(a[4]*2) * a[9] + + (uint64_t)(a[5]*2) * a[8] + + (uint64_t)(a[6]*2) * a[7]; + VERIFY_BITS(d, 63); + /* [d 0 0 0 t9 0 0 0 0 0 c t2 t1 t0] = [p13 p12 p11 p10 p9 0 0 0 0 0 p3 p2 p1 p0] */ + u3 = d & M; d >>= 26; c += u3 * R0; + VERIFY_BITS(u3, 26); + VERIFY_BITS(d, 37); + /* VERIFY_BITS(c, 64); */ + /* [d u3 0 0 0 t9 0 0 0 0 0 c-u3*R0 t2 t1 t0] = [p13 p12 p11 p10 p9 0 0 0 0 0 p3 p2 p1 p0] */ + t3 = c & M; c >>= 26; c += u3 * R1; + VERIFY_BITS(t3, 26); + VERIFY_BITS(c, 39); + /* [d u3 0 0 0 t9 0 0 0 0 c-u3*R1 t3-u3*R0 t2 t1 t0] = [p13 p12 p11 p10 p9 0 0 0 0 0 p3 p2 p1 p0] */ + /* [d 0 0 0 0 t9 0 0 0 0 c t3 t2 t1 t0] = [p13 p12 p11 p10 p9 0 0 0 0 0 p3 p2 p1 p0] */ + + c += (uint64_t)(a[0]*2) * a[4] + + (uint64_t)(a[1]*2) * a[3] + + (uint64_t)a[2] * a[2]; + VERIFY_BITS(c, 63); + /* [d 0 0 0 0 t9 0 0 0 0 c t3 t2 t1 t0] = [p13 p12 p11 p10 p9 0 0 0 0 p4 p3 p2 p1 p0] */ + d += (uint64_t)(a[5]*2) * a[9] + + (uint64_t)(a[6]*2) * a[8] + + (uint64_t)a[7] * a[7]; + VERIFY_BITS(d, 62); + /* [d 0 0 0 0 t9 0 0 0 0 c t3 t2 t1 t0] = [p14 p13 p12 p11 p10 p9 0 0 0 0 p4 p3 p2 p1 p0] */ + u4 = d & M; d >>= 26; c += u4 * R0; + VERIFY_BITS(u4, 26); + VERIFY_BITS(d, 36); + /* VERIFY_BITS(c, 64); */ + /* [d u4 0 0 0 0 t9 0 0 0 0 c-u4*R0 t3 t2 t1 t0] = [p14 p13 p12 p11 p10 p9 0 0 0 0 p4 p3 p2 p1 p0] */ + t4 = c & M; c >>= 26; c += u4 * R1; + VERIFY_BITS(t4, 26); + VERIFY_BITS(c, 39); + /* [d u4 0 0 0 0 t9 0 0 0 c-u4*R1 t4-u4*R0 t3 t2 t1 t0] = [p14 p13 p12 p11 p10 p9 0 0 0 0 p4 p3 p2 p1 p0] */ + /* [d 0 0 0 0 0 t9 0 0 0 c t4 t3 t2 t1 t0] = [p14 p13 p12 p11 p10 p9 0 0 0 0 p4 p3 p2 p1 p0] */ + + c += (uint64_t)(a[0]*2) * a[5] + + (uint64_t)(a[1]*2) * a[4] + + (uint64_t)(a[2]*2) * a[3]; + VERIFY_BITS(c, 63); + /* [d 0 0 0 0 0 t9 0 0 0 c t4 t3 t2 t1 t0] = [p14 p13 p12 p11 p10 p9 0 0 0 p5 p4 p3 p2 p1 p0] */ + d += (uint64_t)(a[6]*2) * a[9] + + (uint64_t)(a[7]*2) * a[8]; + VERIFY_BITS(d, 62); + /* [d 0 0 0 0 0 t9 0 0 0 c t4 t3 t2 t1 t0] = [p15 p14 p13 p12 p11 p10 p9 0 0 0 p5 p4 p3 p2 p1 p0] */ + u5 = d & M; d >>= 26; c += u5 * R0; + VERIFY_BITS(u5, 26); + VERIFY_BITS(d, 36); + /* VERIFY_BITS(c, 64); */ + /* [d u5 0 0 0 0 0 t9 0 0 0 c-u5*R0 t4 t3 t2 t1 t0] = [p15 p14 p13 p12 p11 p10 p9 0 0 0 p5 p4 p3 p2 p1 p0] */ + t5 = c & M; c >>= 26; c += u5 * R1; + VERIFY_BITS(t5, 26); + VERIFY_BITS(c, 39); + /* [d u5 0 0 0 0 0 t9 0 0 c-u5*R1 t5-u5*R0 t4 t3 t2 t1 t0] = [p15 p14 p13 p12 p11 p10 p9 0 0 0 p5 p4 p3 p2 p1 p0] */ + /* [d 0 0 0 0 0 0 t9 0 0 c t5 t4 t3 t2 t1 t0] = [p15 p14 p13 p12 p11 p10 p9 0 0 0 p5 p4 p3 p2 p1 p0] */ + + c += (uint64_t)(a[0]*2) * a[6] + + (uint64_t)(a[1]*2) * a[5] + + (uint64_t)(a[2]*2) * a[4] + + (uint64_t)a[3] * a[3]; + VERIFY_BITS(c, 63); + /* [d 0 0 0 0 0 0 t9 0 0 c t5 t4 t3 t2 t1 t0] = [p15 p14 p13 p12 p11 p10 p9 0 0 p6 p5 p4 p3 p2 p1 p0] */ + d += (uint64_t)(a[7]*2) * a[9] + + (uint64_t)a[8] * a[8]; + VERIFY_BITS(d, 61); + /* [d 0 0 0 0 0 0 t9 0 0 c t5 t4 t3 t2 t1 t0] = [p16 p15 p14 p13 p12 p11 p10 p9 0 0 p6 p5 p4 p3 p2 p1 p0] */ + u6 = d & M; d >>= 26; c += u6 * R0; + VERIFY_BITS(u6, 26); + VERIFY_BITS(d, 35); + /* VERIFY_BITS(c, 64); */ + /* [d u6 0 0 0 0 0 0 t9 0 0 c-u6*R0 t5 t4 t3 t2 t1 t0] = [p16 p15 p14 p13 p12 p11 p10 p9 0 0 p6 p5 p4 p3 p2 p1 p0] */ + t6 = c & M; c >>= 26; c += u6 * R1; + VERIFY_BITS(t6, 26); + VERIFY_BITS(c, 39); + /* [d u6 0 0 0 0 0 0 t9 0 c-u6*R1 t6-u6*R0 t5 t4 t3 t2 t1 t0] = [p16 p15 p14 p13 p12 p11 p10 p9 0 0 p6 p5 p4 p3 p2 p1 p0] */ + /* [d 0 0 0 0 0 0 0 t9 0 c t6 t5 t4 t3 t2 t1 t0] = [p16 p15 p14 p13 p12 p11 p10 p9 0 0 p6 p5 p4 p3 p2 p1 p0] */ + + c += (uint64_t)(a[0]*2) * a[7] + + (uint64_t)(a[1]*2) * a[6] + + (uint64_t)(a[2]*2) * a[5] + + (uint64_t)(a[3]*2) * a[4]; + /* VERIFY_BITS(c, 64); */ + VERIFY_CHECK(c <= 0x8000007C00000007ULL); + /* [d 0 0 0 0 0 0 0 t9 0 c t6 t5 t4 t3 t2 t1 t0] = [p16 p15 p14 p13 p12 p11 p10 p9 0 p7 p6 p5 p4 p3 p2 p1 p0] */ + d += (uint64_t)(a[8]*2) * a[9]; + VERIFY_BITS(d, 58); + /* [d 0 0 0 0 0 0 0 t9 0 c t6 t5 t4 t3 t2 t1 t0] = [p17 p16 p15 p14 p13 p12 p11 p10 p9 0 p7 p6 p5 p4 p3 p2 p1 p0] */ + u7 = d & M; d >>= 26; c += u7 * R0; + VERIFY_BITS(u7, 26); + VERIFY_BITS(d, 32); + /* VERIFY_BITS(c, 64); */ + VERIFY_CHECK(c <= 0x800001703FFFC2F7ULL); + /* [d u7 0 0 0 0 0 0 0 t9 0 c-u7*R0 t6 t5 t4 t3 t2 t1 t0] = [p17 p16 p15 p14 p13 p12 p11 p10 p9 0 p7 p6 p5 p4 p3 p2 p1 p0] */ + t7 = c & M; c >>= 26; c += u7 * R1; + VERIFY_BITS(t7, 26); + VERIFY_BITS(c, 38); + /* [d u7 0 0 0 0 0 0 0 t9 c-u7*R1 t7-u7*R0 t6 t5 t4 t3 t2 t1 t0] = [p17 p16 p15 p14 p13 p12 p11 p10 p9 0 p7 p6 p5 p4 p3 p2 p1 p0] */ + /* [d 0 0 0 0 0 0 0 0 t9 c t7 t6 t5 t4 t3 t2 t1 t0] = [p17 p16 p15 p14 p13 p12 p11 p10 p9 0 p7 p6 p5 p4 p3 p2 p1 p0] */ + + c += (uint64_t)(a[0]*2) * a[8] + + (uint64_t)(a[1]*2) * a[7] + + (uint64_t)(a[2]*2) * a[6] + + (uint64_t)(a[3]*2) * a[5] + + (uint64_t)a[4] * a[4]; + /* VERIFY_BITS(c, 64); */ + VERIFY_CHECK(c <= 0x9000007B80000008ULL); + /* [d 0 0 0 0 0 0 0 0 t9 c t7 t6 t5 t4 t3 t2 t1 t0] = [p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + d += (uint64_t)a[9] * a[9]; + VERIFY_BITS(d, 57); + /* [d 0 0 0 0 0 0 0 0 t9 c t7 t6 t5 t4 t3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + u8 = d & M; d >>= 26; c += u8 * R0; + VERIFY_BITS(u8, 26); + VERIFY_BITS(d, 31); + /* VERIFY_BITS(c, 64); */ + VERIFY_CHECK(c <= 0x9000016FBFFFC2F8ULL); + /* [d u8 0 0 0 0 0 0 0 0 t9 c-u8*R0 t7 t6 t5 t4 t3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + + r[3] = t3; + VERIFY_BITS(r[3], 26); + /* [d u8 0 0 0 0 0 0 0 0 t9 c-u8*R0 t7 t6 t5 t4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[4] = t4; + VERIFY_BITS(r[4], 26); + /* [d u8 0 0 0 0 0 0 0 0 t9 c-u8*R0 t7 t6 t5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[5] = t5; + VERIFY_BITS(r[5], 26); + /* [d u8 0 0 0 0 0 0 0 0 t9 c-u8*R0 t7 t6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[6] = t6; + VERIFY_BITS(r[6], 26); + /* [d u8 0 0 0 0 0 0 0 0 t9 c-u8*R0 t7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[7] = t7; + VERIFY_BITS(r[7], 26); + /* [d u8 0 0 0 0 0 0 0 0 t9 c-u8*R0 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + + r[8] = c & M; c >>= 26; c += u8 * R1; + VERIFY_BITS(r[8], 26); + VERIFY_BITS(c, 39); + /* [d u8 0 0 0 0 0 0 0 0 t9+c-u8*R1 r8-u8*R0 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + /* [d 0 0 0 0 0 0 0 0 0 t9+c r8 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + c += d * R0 + t9; + VERIFY_BITS(c, 45); + /* [d 0 0 0 0 0 0 0 0 0 c-d*R0 r8 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[9] = c & (M >> 4); c >>= 22; c += d * (R1 << 4); + VERIFY_BITS(r[9], 22); + VERIFY_BITS(c, 46); + /* [d 0 0 0 0 0 0 0 0 r9+((c-d*R1<<4)<<22)-d*R0 r8 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + /* [d 0 0 0 0 0 0 0 -d*R1 r9+(c<<22)-d*R0 r8 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + /* [r9+(c<<22) r8 r7 r6 r5 r4 r3 t2 t1 t0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + + d = c * (R0 >> 4) + t0; + VERIFY_BITS(d, 56); + /* [r9+(c<<22) r8 r7 r6 r5 r4 r3 t2 t1 d-c*R0>>4] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[0] = d & M; d >>= 26; + VERIFY_BITS(r[0], 26); + VERIFY_BITS(d, 30); + /* [r9+(c<<22) r8 r7 r6 r5 r4 r3 t2 t1+d r0-c*R0>>4] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + d += c * (R1 >> 4) + t1; + VERIFY_BITS(d, 53); + VERIFY_CHECK(d <= 0x10000003FFFFBFULL); + /* [r9+(c<<22) r8 r7 r6 r5 r4 r3 t2 d-c*R1>>4 r0-c*R0>>4] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + /* [r9 r8 r7 r6 r5 r4 r3 t2 d r0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[1] = d & M; d >>= 26; + VERIFY_BITS(r[1], 26); + VERIFY_BITS(d, 27); + VERIFY_CHECK(d <= 0x4000000ULL); + /* [r9 r8 r7 r6 r5 r4 r3 t2+d r1 r0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + d += t2; + VERIFY_BITS(d, 27); + /* [r9 r8 r7 r6 r5 r4 r3 d r1 r0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[2] = d; + VERIFY_BITS(r[2], 27); + /* [r9 r8 r7 r6 r5 r4 r3 r2 r1 r0] = [p18 p17 p16 p15 p14 p13 p12 p11 p10 p9 p8 p7 p6 p5 p4 p3 p2 p1 p0] */ +} +#endif + +static void secp256k1_fe_mul(secp256k1_fe *r, const secp256k1_fe *a, const secp256k1_fe * SECP256K1_RESTRICT b) { +#ifdef VERIFY + VERIFY_CHECK(a->magnitude <= 8); + VERIFY_CHECK(b->magnitude <= 8); + secp256k1_fe_verify(a); + secp256k1_fe_verify(b); + VERIFY_CHECK(r != b); + VERIFY_CHECK(a != b); +#endif + secp256k1_fe_mul_inner(r->n, a->n, b->n); +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 0; + secp256k1_fe_verify(r); +#endif +} + +static void secp256k1_fe_sqr(secp256k1_fe *r, const secp256k1_fe *a) { +#ifdef VERIFY + VERIFY_CHECK(a->magnitude <= 8); + secp256k1_fe_verify(a); +#endif + secp256k1_fe_sqr_inner(r->n, a->n); +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 0; + secp256k1_fe_verify(r); +#endif +} + +static SECP256K1_INLINE void secp256k1_fe_cmov(secp256k1_fe *r, const secp256k1_fe *a, int flag) { + uint32_t mask0, mask1; + mask0 = flag + ~((uint32_t)0); + mask1 = ~mask0; + r->n[0] = (r->n[0] & mask0) | (a->n[0] & mask1); + r->n[1] = (r->n[1] & mask0) | (a->n[1] & mask1); + r->n[2] = (r->n[2] & mask0) | (a->n[2] & mask1); + r->n[3] = (r->n[3] & mask0) | (a->n[3] & mask1); + r->n[4] = (r->n[4] & mask0) | (a->n[4] & mask1); + r->n[5] = (r->n[5] & mask0) | (a->n[5] & mask1); + r->n[6] = (r->n[6] & mask0) | (a->n[6] & mask1); + r->n[7] = (r->n[7] & mask0) | (a->n[7] & mask1); + r->n[8] = (r->n[8] & mask0) | (a->n[8] & mask1); + r->n[9] = (r->n[9] & mask0) | (a->n[9] & mask1); +#ifdef VERIFY + if (a->magnitude > r->magnitude) { + r->magnitude = a->magnitude; + } + r->normalized &= a->normalized; +#endif +} + +static SECP256K1_INLINE void secp256k1_fe_storage_cmov(secp256k1_fe_storage *r, const secp256k1_fe_storage *a, int flag) { + uint32_t mask0, mask1; + mask0 = flag + ~((uint32_t)0); + mask1 = ~mask0; + r->n[0] = (r->n[0] & mask0) | (a->n[0] & mask1); + r->n[1] = (r->n[1] & mask0) | (a->n[1] & mask1); + r->n[2] = (r->n[2] & mask0) | (a->n[2] & mask1); + r->n[3] = (r->n[3] & mask0) | (a->n[3] & mask1); + r->n[4] = (r->n[4] & mask0) | (a->n[4] & mask1); + r->n[5] = (r->n[5] & mask0) | (a->n[5] & mask1); + r->n[6] = (r->n[6] & mask0) | (a->n[6] & mask1); + r->n[7] = (r->n[7] & mask0) | (a->n[7] & mask1); +} + +static void secp256k1_fe_to_storage(secp256k1_fe_storage *r, const secp256k1_fe *a) { +#ifdef VERIFY + VERIFY_CHECK(a->normalized); +#endif + r->n[0] = a->n[0] | a->n[1] << 26; + r->n[1] = a->n[1] >> 6 | a->n[2] << 20; + r->n[2] = a->n[2] >> 12 | a->n[3] << 14; + r->n[3] = a->n[3] >> 18 | a->n[4] << 8; + r->n[4] = a->n[4] >> 24 | a->n[5] << 2 | a->n[6] << 28; + r->n[5] = a->n[6] >> 4 | a->n[7] << 22; + r->n[6] = a->n[7] >> 10 | a->n[8] << 16; + r->n[7] = a->n[8] >> 16 | a->n[9] << 10; +} + +static SECP256K1_INLINE void secp256k1_fe_from_storage(secp256k1_fe *r, const secp256k1_fe_storage *a) { + r->n[0] = a->n[0] & 0x3FFFFFFUL; + r->n[1] = a->n[0] >> 26 | ((a->n[1] << 6) & 0x3FFFFFFUL); + r->n[2] = a->n[1] >> 20 | ((a->n[2] << 12) & 0x3FFFFFFUL); + r->n[3] = a->n[2] >> 14 | ((a->n[3] << 18) & 0x3FFFFFFUL); + r->n[4] = a->n[3] >> 8 | ((a->n[4] << 24) & 0x3FFFFFFUL); + r->n[5] = (a->n[4] >> 2) & 0x3FFFFFFUL; + r->n[6] = a->n[4] >> 28 | ((a->n[5] << 4) & 0x3FFFFFFUL); + r->n[7] = a->n[5] >> 22 | ((a->n[6] << 10) & 0x3FFFFFFUL); + r->n[8] = a->n[6] >> 16 | ((a->n[7] << 16) & 0x3FFFFFFUL); + r->n[9] = a->n[7] >> 10; +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 1; +#endif +} + +#endif /* SECP256K1_FIELD_REPR_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/field_5x52.h b/node_modules/secp256k1/src/secp256k1/src/field_5x52.h new file mode 100644 index 0000000..fc5bfe3 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/field_5x52.h @@ -0,0 +1,49 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_FIELD_REPR_H +#define SECP256K1_FIELD_REPR_H + +#include + +typedef struct { + /* X = sum(i=0..4, n[i]*2^(i*52)) mod p + * where p = 2^256 - 0x1000003D1 + */ + uint64_t n[5]; +#ifdef VERIFY + int magnitude; + int normalized; +#endif +} secp256k1_fe; + +/* Unpacks a constant into a overlapping multi-limbed FE element. */ +#define SECP256K1_FE_CONST_INNER(d7, d6, d5, d4, d3, d2, d1, d0) { \ + (d0) | (((uint64_t)(d1) & 0xFFFFFUL) << 32), \ + ((uint64_t)(d1) >> 20) | (((uint64_t)(d2)) << 12) | (((uint64_t)(d3) & 0xFFUL) << 44), \ + ((uint64_t)(d3) >> 8) | (((uint64_t)(d4) & 0xFFFFFFFUL) << 24), \ + ((uint64_t)(d4) >> 28) | (((uint64_t)(d5)) << 4) | (((uint64_t)(d6) & 0xFFFFUL) << 36), \ + ((uint64_t)(d6) >> 16) | (((uint64_t)(d7)) << 16) \ +} + +#ifdef VERIFY +#define SECP256K1_FE_CONST(d7, d6, d5, d4, d3, d2, d1, d0) {SECP256K1_FE_CONST_INNER((d7), (d6), (d5), (d4), (d3), (d2), (d1), (d0)), 1, 1} +#else +#define SECP256K1_FE_CONST(d7, d6, d5, d4, d3, d2, d1, d0) {SECP256K1_FE_CONST_INNER((d7), (d6), (d5), (d4), (d3), (d2), (d1), (d0))} +#endif + +typedef struct { + uint64_t n[4]; +} secp256k1_fe_storage; + +#define SECP256K1_FE_STORAGE_CONST(d7, d6, d5, d4, d3, d2, d1, d0) {{ \ + (d0) | (((uint64_t)(d1)) << 32), \ + (d2) | (((uint64_t)(d3)) << 32), \ + (d4) | (((uint64_t)(d5)) << 32), \ + (d6) | (((uint64_t)(d7)) << 32) \ +}} + +#endif /* SECP256K1_FIELD_REPR_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/field_5x52_asm_impl.h b/node_modules/secp256k1/src/secp256k1/src/field_5x52_asm_impl.h new file mode 100644 index 0000000..1fc3171 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/field_5x52_asm_impl.h @@ -0,0 +1,502 @@ +/********************************************************************** + * Copyright (c) 2013-2014 Diederik Huys, Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +/** + * Changelog: + * - March 2013, Diederik Huys: original version + * - November 2014, Pieter Wuille: updated to use Peter Dettman's parallel multiplication algorithm + * - December 2014, Pieter Wuille: converted from YASM to GCC inline assembly + */ + +#ifndef SECP256K1_FIELD_INNER5X52_IMPL_H +#define SECP256K1_FIELD_INNER5X52_IMPL_H + +SECP256K1_INLINE static void secp256k1_fe_mul_inner(uint64_t *r, const uint64_t *a, const uint64_t * SECP256K1_RESTRICT b) { +/** + * Registers: rdx:rax = multiplication accumulator + * r9:r8 = c + * r15:rcx = d + * r10-r14 = a0-a4 + * rbx = b + * rdi = r + * rsi = a / t? + */ + uint64_t tmp1, tmp2, tmp3; +__asm__ __volatile__( + "movq 0(%%rsi),%%r10\n" + "movq 8(%%rsi),%%r11\n" + "movq 16(%%rsi),%%r12\n" + "movq 24(%%rsi),%%r13\n" + "movq 32(%%rsi),%%r14\n" + + /* d += a3 * b0 */ + "movq 0(%%rbx),%%rax\n" + "mulq %%r13\n" + "movq %%rax,%%rcx\n" + "movq %%rdx,%%r15\n" + /* d += a2 * b1 */ + "movq 8(%%rbx),%%rax\n" + "mulq %%r12\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d += a1 * b2 */ + "movq 16(%%rbx),%%rax\n" + "mulq %%r11\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d = a0 * b3 */ + "movq 24(%%rbx),%%rax\n" + "mulq %%r10\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* c = a4 * b4 */ + "movq 32(%%rbx),%%rax\n" + "mulq %%r14\n" + "movq %%rax,%%r8\n" + "movq %%rdx,%%r9\n" + /* d += (c & M) * R */ + "movq $0xfffffffffffff,%%rdx\n" + "andq %%rdx,%%rax\n" + "movq $0x1000003d10,%%rdx\n" + "mulq %%rdx\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* c >>= 52 (%%r8 only) */ + "shrdq $52,%%r9,%%r8\n" + /* t3 (tmp1) = d & M */ + "movq %%rcx,%%rsi\n" + "movq $0xfffffffffffff,%%rdx\n" + "andq %%rdx,%%rsi\n" + "movq %%rsi,%q1\n" + /* d >>= 52 */ + "shrdq $52,%%r15,%%rcx\n" + "xorq %%r15,%%r15\n" + /* d += a4 * b0 */ + "movq 0(%%rbx),%%rax\n" + "mulq %%r14\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d += a3 * b1 */ + "movq 8(%%rbx),%%rax\n" + "mulq %%r13\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d += a2 * b2 */ + "movq 16(%%rbx),%%rax\n" + "mulq %%r12\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d += a1 * b3 */ + "movq 24(%%rbx),%%rax\n" + "mulq %%r11\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d += a0 * b4 */ + "movq 32(%%rbx),%%rax\n" + "mulq %%r10\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d += c * R */ + "movq %%r8,%%rax\n" + "movq $0x1000003d10,%%rdx\n" + "mulq %%rdx\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* t4 = d & M (%%rsi) */ + "movq %%rcx,%%rsi\n" + "movq $0xfffffffffffff,%%rdx\n" + "andq %%rdx,%%rsi\n" + /* d >>= 52 */ + "shrdq $52,%%r15,%%rcx\n" + "xorq %%r15,%%r15\n" + /* tx = t4 >> 48 (tmp3) */ + "movq %%rsi,%%rax\n" + "shrq $48,%%rax\n" + "movq %%rax,%q3\n" + /* t4 &= (M >> 4) (tmp2) */ + "movq $0xffffffffffff,%%rax\n" + "andq %%rax,%%rsi\n" + "movq %%rsi,%q2\n" + /* c = a0 * b0 */ + "movq 0(%%rbx),%%rax\n" + "mulq %%r10\n" + "movq %%rax,%%r8\n" + "movq %%rdx,%%r9\n" + /* d += a4 * b1 */ + "movq 8(%%rbx),%%rax\n" + "mulq %%r14\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d += a3 * b2 */ + "movq 16(%%rbx),%%rax\n" + "mulq %%r13\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d += a2 * b3 */ + "movq 24(%%rbx),%%rax\n" + "mulq %%r12\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d += a1 * b4 */ + "movq 32(%%rbx),%%rax\n" + "mulq %%r11\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* u0 = d & M (%%rsi) */ + "movq %%rcx,%%rsi\n" + "movq $0xfffffffffffff,%%rdx\n" + "andq %%rdx,%%rsi\n" + /* d >>= 52 */ + "shrdq $52,%%r15,%%rcx\n" + "xorq %%r15,%%r15\n" + /* u0 = (u0 << 4) | tx (%%rsi) */ + "shlq $4,%%rsi\n" + "movq %q3,%%rax\n" + "orq %%rax,%%rsi\n" + /* c += u0 * (R >> 4) */ + "movq $0x1000003d1,%%rax\n" + "mulq %%rsi\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* r[0] = c & M */ + "movq %%r8,%%rax\n" + "movq $0xfffffffffffff,%%rdx\n" + "andq %%rdx,%%rax\n" + "movq %%rax,0(%%rdi)\n" + /* c >>= 52 */ + "shrdq $52,%%r9,%%r8\n" + "xorq %%r9,%%r9\n" + /* c += a1 * b0 */ + "movq 0(%%rbx),%%rax\n" + "mulq %%r11\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* c += a0 * b1 */ + "movq 8(%%rbx),%%rax\n" + "mulq %%r10\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* d += a4 * b2 */ + "movq 16(%%rbx),%%rax\n" + "mulq %%r14\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d += a3 * b3 */ + "movq 24(%%rbx),%%rax\n" + "mulq %%r13\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d += a2 * b4 */ + "movq 32(%%rbx),%%rax\n" + "mulq %%r12\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* c += (d & M) * R */ + "movq %%rcx,%%rax\n" + "movq $0xfffffffffffff,%%rdx\n" + "andq %%rdx,%%rax\n" + "movq $0x1000003d10,%%rdx\n" + "mulq %%rdx\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* d >>= 52 */ + "shrdq $52,%%r15,%%rcx\n" + "xorq %%r15,%%r15\n" + /* r[1] = c & M */ + "movq %%r8,%%rax\n" + "movq $0xfffffffffffff,%%rdx\n" + "andq %%rdx,%%rax\n" + "movq %%rax,8(%%rdi)\n" + /* c >>= 52 */ + "shrdq $52,%%r9,%%r8\n" + "xorq %%r9,%%r9\n" + /* c += a2 * b0 */ + "movq 0(%%rbx),%%rax\n" + "mulq %%r12\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* c += a1 * b1 */ + "movq 8(%%rbx),%%rax\n" + "mulq %%r11\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* c += a0 * b2 (last use of %%r10 = a0) */ + "movq 16(%%rbx),%%rax\n" + "mulq %%r10\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* fetch t3 (%%r10, overwrites a0), t4 (%%rsi) */ + "movq %q2,%%rsi\n" + "movq %q1,%%r10\n" + /* d += a4 * b3 */ + "movq 24(%%rbx),%%rax\n" + "mulq %%r14\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* d += a3 * b4 */ + "movq 32(%%rbx),%%rax\n" + "mulq %%r13\n" + "addq %%rax,%%rcx\n" + "adcq %%rdx,%%r15\n" + /* c += (d & M) * R */ + "movq %%rcx,%%rax\n" + "movq $0xfffffffffffff,%%rdx\n" + "andq %%rdx,%%rax\n" + "movq $0x1000003d10,%%rdx\n" + "mulq %%rdx\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* d >>= 52 (%%rcx only) */ + "shrdq $52,%%r15,%%rcx\n" + /* r[2] = c & M */ + "movq %%r8,%%rax\n" + "movq $0xfffffffffffff,%%rdx\n" + "andq %%rdx,%%rax\n" + "movq %%rax,16(%%rdi)\n" + /* c >>= 52 */ + "shrdq $52,%%r9,%%r8\n" + "xorq %%r9,%%r9\n" + /* c += t3 */ + "addq %%r10,%%r8\n" + /* c += d * R */ + "movq %%rcx,%%rax\n" + "movq $0x1000003d10,%%rdx\n" + "mulq %%rdx\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* r[3] = c & M */ + "movq %%r8,%%rax\n" + "movq $0xfffffffffffff,%%rdx\n" + "andq %%rdx,%%rax\n" + "movq %%rax,24(%%rdi)\n" + /* c >>= 52 (%%r8 only) */ + "shrdq $52,%%r9,%%r8\n" + /* c += t4 (%%r8 only) */ + "addq %%rsi,%%r8\n" + /* r[4] = c */ + "movq %%r8,32(%%rdi)\n" +: "+S"(a), "=m"(tmp1), "=m"(tmp2), "=m"(tmp3) +: "b"(b), "D"(r) +: "%rax", "%rcx", "%rdx", "%r8", "%r9", "%r10", "%r11", "%r12", "%r13", "%r14", "%r15", "cc", "memory" +); +} + +SECP256K1_INLINE static void secp256k1_fe_sqr_inner(uint64_t *r, const uint64_t *a) { +/** + * Registers: rdx:rax = multiplication accumulator + * r9:r8 = c + * rcx:rbx = d + * r10-r14 = a0-a4 + * r15 = M (0xfffffffffffff) + * rdi = r + * rsi = a / t? + */ + uint64_t tmp1, tmp2, tmp3; +__asm__ __volatile__( + "movq 0(%%rsi),%%r10\n" + "movq 8(%%rsi),%%r11\n" + "movq 16(%%rsi),%%r12\n" + "movq 24(%%rsi),%%r13\n" + "movq 32(%%rsi),%%r14\n" + "movq $0xfffffffffffff,%%r15\n" + + /* d = (a0*2) * a3 */ + "leaq (%%r10,%%r10,1),%%rax\n" + "mulq %%r13\n" + "movq %%rax,%%rbx\n" + "movq %%rdx,%%rcx\n" + /* d += (a1*2) * a2 */ + "leaq (%%r11,%%r11,1),%%rax\n" + "mulq %%r12\n" + "addq %%rax,%%rbx\n" + "adcq %%rdx,%%rcx\n" + /* c = a4 * a4 */ + "movq %%r14,%%rax\n" + "mulq %%r14\n" + "movq %%rax,%%r8\n" + "movq %%rdx,%%r9\n" + /* d += (c & M) * R */ + "andq %%r15,%%rax\n" + "movq $0x1000003d10,%%rdx\n" + "mulq %%rdx\n" + "addq %%rax,%%rbx\n" + "adcq %%rdx,%%rcx\n" + /* c >>= 52 (%%r8 only) */ + "shrdq $52,%%r9,%%r8\n" + /* t3 (tmp1) = d & M */ + "movq %%rbx,%%rsi\n" + "andq %%r15,%%rsi\n" + "movq %%rsi,%q1\n" + /* d >>= 52 */ + "shrdq $52,%%rcx,%%rbx\n" + "xorq %%rcx,%%rcx\n" + /* a4 *= 2 */ + "addq %%r14,%%r14\n" + /* d += a0 * a4 */ + "movq %%r10,%%rax\n" + "mulq %%r14\n" + "addq %%rax,%%rbx\n" + "adcq %%rdx,%%rcx\n" + /* d+= (a1*2) * a3 */ + "leaq (%%r11,%%r11,1),%%rax\n" + "mulq %%r13\n" + "addq %%rax,%%rbx\n" + "adcq %%rdx,%%rcx\n" + /* d += a2 * a2 */ + "movq %%r12,%%rax\n" + "mulq %%r12\n" + "addq %%rax,%%rbx\n" + "adcq %%rdx,%%rcx\n" + /* d += c * R */ + "movq %%r8,%%rax\n" + "movq $0x1000003d10,%%rdx\n" + "mulq %%rdx\n" + "addq %%rax,%%rbx\n" + "adcq %%rdx,%%rcx\n" + /* t4 = d & M (%%rsi) */ + "movq %%rbx,%%rsi\n" + "andq %%r15,%%rsi\n" + /* d >>= 52 */ + "shrdq $52,%%rcx,%%rbx\n" + "xorq %%rcx,%%rcx\n" + /* tx = t4 >> 48 (tmp3) */ + "movq %%rsi,%%rax\n" + "shrq $48,%%rax\n" + "movq %%rax,%q3\n" + /* t4 &= (M >> 4) (tmp2) */ + "movq $0xffffffffffff,%%rax\n" + "andq %%rax,%%rsi\n" + "movq %%rsi,%q2\n" + /* c = a0 * a0 */ + "movq %%r10,%%rax\n" + "mulq %%r10\n" + "movq %%rax,%%r8\n" + "movq %%rdx,%%r9\n" + /* d += a1 * a4 */ + "movq %%r11,%%rax\n" + "mulq %%r14\n" + "addq %%rax,%%rbx\n" + "adcq %%rdx,%%rcx\n" + /* d += (a2*2) * a3 */ + "leaq (%%r12,%%r12,1),%%rax\n" + "mulq %%r13\n" + "addq %%rax,%%rbx\n" + "adcq %%rdx,%%rcx\n" + /* u0 = d & M (%%rsi) */ + "movq %%rbx,%%rsi\n" + "andq %%r15,%%rsi\n" + /* d >>= 52 */ + "shrdq $52,%%rcx,%%rbx\n" + "xorq %%rcx,%%rcx\n" + /* u0 = (u0 << 4) | tx (%%rsi) */ + "shlq $4,%%rsi\n" + "movq %q3,%%rax\n" + "orq %%rax,%%rsi\n" + /* c += u0 * (R >> 4) */ + "movq $0x1000003d1,%%rax\n" + "mulq %%rsi\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* r[0] = c & M */ + "movq %%r8,%%rax\n" + "andq %%r15,%%rax\n" + "movq %%rax,0(%%rdi)\n" + /* c >>= 52 */ + "shrdq $52,%%r9,%%r8\n" + "xorq %%r9,%%r9\n" + /* a0 *= 2 */ + "addq %%r10,%%r10\n" + /* c += a0 * a1 */ + "movq %%r10,%%rax\n" + "mulq %%r11\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* d += a2 * a4 */ + "movq %%r12,%%rax\n" + "mulq %%r14\n" + "addq %%rax,%%rbx\n" + "adcq %%rdx,%%rcx\n" + /* d += a3 * a3 */ + "movq %%r13,%%rax\n" + "mulq %%r13\n" + "addq %%rax,%%rbx\n" + "adcq %%rdx,%%rcx\n" + /* c += (d & M) * R */ + "movq %%rbx,%%rax\n" + "andq %%r15,%%rax\n" + "movq $0x1000003d10,%%rdx\n" + "mulq %%rdx\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* d >>= 52 */ + "shrdq $52,%%rcx,%%rbx\n" + "xorq %%rcx,%%rcx\n" + /* r[1] = c & M */ + "movq %%r8,%%rax\n" + "andq %%r15,%%rax\n" + "movq %%rax,8(%%rdi)\n" + /* c >>= 52 */ + "shrdq $52,%%r9,%%r8\n" + "xorq %%r9,%%r9\n" + /* c += a0 * a2 (last use of %%r10) */ + "movq %%r10,%%rax\n" + "mulq %%r12\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* fetch t3 (%%r10, overwrites a0),t4 (%%rsi) */ + "movq %q2,%%rsi\n" + "movq %q1,%%r10\n" + /* c += a1 * a1 */ + "movq %%r11,%%rax\n" + "mulq %%r11\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* d += a3 * a4 */ + "movq %%r13,%%rax\n" + "mulq %%r14\n" + "addq %%rax,%%rbx\n" + "adcq %%rdx,%%rcx\n" + /* c += (d & M) * R */ + "movq %%rbx,%%rax\n" + "andq %%r15,%%rax\n" + "movq $0x1000003d10,%%rdx\n" + "mulq %%rdx\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* d >>= 52 (%%rbx only) */ + "shrdq $52,%%rcx,%%rbx\n" + /* r[2] = c & M */ + "movq %%r8,%%rax\n" + "andq %%r15,%%rax\n" + "movq %%rax,16(%%rdi)\n" + /* c >>= 52 */ + "shrdq $52,%%r9,%%r8\n" + "xorq %%r9,%%r9\n" + /* c += t3 */ + "addq %%r10,%%r8\n" + /* c += d * R */ + "movq %%rbx,%%rax\n" + "movq $0x1000003d10,%%rdx\n" + "mulq %%rdx\n" + "addq %%rax,%%r8\n" + "adcq %%rdx,%%r9\n" + /* r[3] = c & M */ + "movq %%r8,%%rax\n" + "andq %%r15,%%rax\n" + "movq %%rax,24(%%rdi)\n" + /* c >>= 52 (%%r8 only) */ + "shrdq $52,%%r9,%%r8\n" + /* c += t4 (%%r8 only) */ + "addq %%rsi,%%r8\n" + /* r[4] = c */ + "movq %%r8,32(%%rdi)\n" +: "+S"(a), "=m"(tmp1), "=m"(tmp2), "=m"(tmp3) +: "D"(r) +: "%rax", "%rbx", "%rcx", "%rdx", "%r8", "%r9", "%r10", "%r11", "%r12", "%r13", "%r14", "%r15", "cc", "memory" +); +} + +#endif /* SECP256K1_FIELD_INNER5X52_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/field_5x52_impl.h b/node_modules/secp256k1/src/secp256k1/src/field_5x52_impl.h new file mode 100644 index 0000000..f426332 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/field_5x52_impl.h @@ -0,0 +1,496 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_FIELD_REPR_IMPL_H +#define SECP256K1_FIELD_REPR_IMPL_H + +#if defined HAVE_CONFIG_H +#include "libsecp256k1-config.h" +#endif + +#include "util.h" +#include "field.h" + +#if defined(USE_ASM_X86_64) +#include "field_5x52_asm_impl.h" +#else +#include "field_5x52_int128_impl.h" +#endif + +/** Implements arithmetic modulo FFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFE FFFFFC2F, + * represented as 5 uint64_t's in base 2^52. The values are allowed to contain >52 each. In particular, + * each FieldElem has a 'magnitude' associated with it. Internally, a magnitude M means each element + * is at most M*(2^53-1), except the most significant one, which is limited to M*(2^49-1). All operations + * accept any input with magnitude at most M, and have different rules for propagating magnitude to their + * output. + */ + +#ifdef VERIFY +static void secp256k1_fe_verify(const secp256k1_fe *a) { + const uint64_t *d = a->n; + int m = a->normalized ? 1 : 2 * a->magnitude, r = 1; + /* secp256k1 'p' value defined in "Standards for Efficient Cryptography" (SEC2) 2.7.1. */ + r &= (d[0] <= 0xFFFFFFFFFFFFFULL * m); + r &= (d[1] <= 0xFFFFFFFFFFFFFULL * m); + r &= (d[2] <= 0xFFFFFFFFFFFFFULL * m); + r &= (d[3] <= 0xFFFFFFFFFFFFFULL * m); + r &= (d[4] <= 0x0FFFFFFFFFFFFULL * m); + r &= (a->magnitude >= 0); + r &= (a->magnitude <= 2048); + if (a->normalized) { + r &= (a->magnitude <= 1); + if (r && (d[4] == 0x0FFFFFFFFFFFFULL) && ((d[3] & d[2] & d[1]) == 0xFFFFFFFFFFFFFULL)) { + r &= (d[0] < 0xFFFFEFFFFFC2FULL); + } + } + VERIFY_CHECK(r == 1); +} +#endif + +static void secp256k1_fe_normalize(secp256k1_fe *r) { + uint64_t t0 = r->n[0], t1 = r->n[1], t2 = r->n[2], t3 = r->n[3], t4 = r->n[4]; + + /* Reduce t4 at the start so there will be at most a single carry from the first pass */ + uint64_t m; + uint64_t x = t4 >> 48; t4 &= 0x0FFFFFFFFFFFFULL; + + /* The first pass ensures the magnitude is 1, ... */ + t0 += x * 0x1000003D1ULL; + t1 += (t0 >> 52); t0 &= 0xFFFFFFFFFFFFFULL; + t2 += (t1 >> 52); t1 &= 0xFFFFFFFFFFFFFULL; m = t1; + t3 += (t2 >> 52); t2 &= 0xFFFFFFFFFFFFFULL; m &= t2; + t4 += (t3 >> 52); t3 &= 0xFFFFFFFFFFFFFULL; m &= t3; + + /* ... except for a possible carry at bit 48 of t4 (i.e. bit 256 of the field element) */ + VERIFY_CHECK(t4 >> 49 == 0); + + /* At most a single final reduction is needed; check if the value is >= the field characteristic */ + x = (t4 >> 48) | ((t4 == 0x0FFFFFFFFFFFFULL) & (m == 0xFFFFFFFFFFFFFULL) + & (t0 >= 0xFFFFEFFFFFC2FULL)); + + /* Apply the final reduction (for constant-time behaviour, we do it always) */ + t0 += x * 0x1000003D1ULL; + t1 += (t0 >> 52); t0 &= 0xFFFFFFFFFFFFFULL; + t2 += (t1 >> 52); t1 &= 0xFFFFFFFFFFFFFULL; + t3 += (t2 >> 52); t2 &= 0xFFFFFFFFFFFFFULL; + t4 += (t3 >> 52); t3 &= 0xFFFFFFFFFFFFFULL; + + /* If t4 didn't carry to bit 48 already, then it should have after any final reduction */ + VERIFY_CHECK(t4 >> 48 == x); + + /* Mask off the possible multiple of 2^256 from the final reduction */ + t4 &= 0x0FFFFFFFFFFFFULL; + + r->n[0] = t0; r->n[1] = t1; r->n[2] = t2; r->n[3] = t3; r->n[4] = t4; + +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 1; + secp256k1_fe_verify(r); +#endif +} + +static void secp256k1_fe_normalize_weak(secp256k1_fe *r) { + uint64_t t0 = r->n[0], t1 = r->n[1], t2 = r->n[2], t3 = r->n[3], t4 = r->n[4]; + + /* Reduce t4 at the start so there will be at most a single carry from the first pass */ + uint64_t x = t4 >> 48; t4 &= 0x0FFFFFFFFFFFFULL; + + /* The first pass ensures the magnitude is 1, ... */ + t0 += x * 0x1000003D1ULL; + t1 += (t0 >> 52); t0 &= 0xFFFFFFFFFFFFFULL; + t2 += (t1 >> 52); t1 &= 0xFFFFFFFFFFFFFULL; + t3 += (t2 >> 52); t2 &= 0xFFFFFFFFFFFFFULL; + t4 += (t3 >> 52); t3 &= 0xFFFFFFFFFFFFFULL; + + /* ... except for a possible carry at bit 48 of t4 (i.e. bit 256 of the field element) */ + VERIFY_CHECK(t4 >> 49 == 0); + + r->n[0] = t0; r->n[1] = t1; r->n[2] = t2; r->n[3] = t3; r->n[4] = t4; + +#ifdef VERIFY + r->magnitude = 1; + secp256k1_fe_verify(r); +#endif +} + +static void secp256k1_fe_normalize_var(secp256k1_fe *r) { + uint64_t t0 = r->n[0], t1 = r->n[1], t2 = r->n[2], t3 = r->n[3], t4 = r->n[4]; + + /* Reduce t4 at the start so there will be at most a single carry from the first pass */ + uint64_t m; + uint64_t x = t4 >> 48; t4 &= 0x0FFFFFFFFFFFFULL; + + /* The first pass ensures the magnitude is 1, ... */ + t0 += x * 0x1000003D1ULL; + t1 += (t0 >> 52); t0 &= 0xFFFFFFFFFFFFFULL; + t2 += (t1 >> 52); t1 &= 0xFFFFFFFFFFFFFULL; m = t1; + t3 += (t2 >> 52); t2 &= 0xFFFFFFFFFFFFFULL; m &= t2; + t4 += (t3 >> 52); t3 &= 0xFFFFFFFFFFFFFULL; m &= t3; + + /* ... except for a possible carry at bit 48 of t4 (i.e. bit 256 of the field element) */ + VERIFY_CHECK(t4 >> 49 == 0); + + /* At most a single final reduction is needed; check if the value is >= the field characteristic */ + x = (t4 >> 48) | ((t4 == 0x0FFFFFFFFFFFFULL) & (m == 0xFFFFFFFFFFFFFULL) + & (t0 >= 0xFFFFEFFFFFC2FULL)); + + if (x) { + t0 += 0x1000003D1ULL; + t1 += (t0 >> 52); t0 &= 0xFFFFFFFFFFFFFULL; + t2 += (t1 >> 52); t1 &= 0xFFFFFFFFFFFFFULL; + t3 += (t2 >> 52); t2 &= 0xFFFFFFFFFFFFFULL; + t4 += (t3 >> 52); t3 &= 0xFFFFFFFFFFFFFULL; + + /* If t4 didn't carry to bit 48 already, then it should have after any final reduction */ + VERIFY_CHECK(t4 >> 48 == x); + + /* Mask off the possible multiple of 2^256 from the final reduction */ + t4 &= 0x0FFFFFFFFFFFFULL; + } + + r->n[0] = t0; r->n[1] = t1; r->n[2] = t2; r->n[3] = t3; r->n[4] = t4; + +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 1; + secp256k1_fe_verify(r); +#endif +} + +static int secp256k1_fe_normalizes_to_zero(secp256k1_fe *r) { + uint64_t t0 = r->n[0], t1 = r->n[1], t2 = r->n[2], t3 = r->n[3], t4 = r->n[4]; + + /* z0 tracks a possible raw value of 0, z1 tracks a possible raw value of P */ + uint64_t z0, z1; + + /* Reduce t4 at the start so there will be at most a single carry from the first pass */ + uint64_t x = t4 >> 48; t4 &= 0x0FFFFFFFFFFFFULL; + + /* The first pass ensures the magnitude is 1, ... */ + t0 += x * 0x1000003D1ULL; + t1 += (t0 >> 52); t0 &= 0xFFFFFFFFFFFFFULL; z0 = t0; z1 = t0 ^ 0x1000003D0ULL; + t2 += (t1 >> 52); t1 &= 0xFFFFFFFFFFFFFULL; z0 |= t1; z1 &= t1; + t3 += (t2 >> 52); t2 &= 0xFFFFFFFFFFFFFULL; z0 |= t2; z1 &= t2; + t4 += (t3 >> 52); t3 &= 0xFFFFFFFFFFFFFULL; z0 |= t3; z1 &= t3; + z0 |= t4; z1 &= t4 ^ 0xF000000000000ULL; + + /* ... except for a possible carry at bit 48 of t4 (i.e. bit 256 of the field element) */ + VERIFY_CHECK(t4 >> 49 == 0); + + return (z0 == 0) | (z1 == 0xFFFFFFFFFFFFFULL); +} + +static int secp256k1_fe_normalizes_to_zero_var(secp256k1_fe *r) { + uint64_t t0, t1, t2, t3, t4; + uint64_t z0, z1; + uint64_t x; + + t0 = r->n[0]; + t4 = r->n[4]; + + /* Reduce t4 at the start so there will be at most a single carry from the first pass */ + x = t4 >> 48; + + /* The first pass ensures the magnitude is 1, ... */ + t0 += x * 0x1000003D1ULL; + + /* z0 tracks a possible raw value of 0, z1 tracks a possible raw value of P */ + z0 = t0 & 0xFFFFFFFFFFFFFULL; + z1 = z0 ^ 0x1000003D0ULL; + + /* Fast return path should catch the majority of cases */ + if ((z0 != 0ULL) & (z1 != 0xFFFFFFFFFFFFFULL)) { + return 0; + } + + t1 = r->n[1]; + t2 = r->n[2]; + t3 = r->n[3]; + + t4 &= 0x0FFFFFFFFFFFFULL; + + t1 += (t0 >> 52); + t2 += (t1 >> 52); t1 &= 0xFFFFFFFFFFFFFULL; z0 |= t1; z1 &= t1; + t3 += (t2 >> 52); t2 &= 0xFFFFFFFFFFFFFULL; z0 |= t2; z1 &= t2; + t4 += (t3 >> 52); t3 &= 0xFFFFFFFFFFFFFULL; z0 |= t3; z1 &= t3; + z0 |= t4; z1 &= t4 ^ 0xF000000000000ULL; + + /* ... except for a possible carry at bit 48 of t4 (i.e. bit 256 of the field element) */ + VERIFY_CHECK(t4 >> 49 == 0); + + return (z0 == 0) | (z1 == 0xFFFFFFFFFFFFFULL); +} + +SECP256K1_INLINE static void secp256k1_fe_set_int(secp256k1_fe *r, int a) { + r->n[0] = a; + r->n[1] = r->n[2] = r->n[3] = r->n[4] = 0; +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 1; + secp256k1_fe_verify(r); +#endif +} + +SECP256K1_INLINE static int secp256k1_fe_is_zero(const secp256k1_fe *a) { + const uint64_t *t = a->n; +#ifdef VERIFY + VERIFY_CHECK(a->normalized); + secp256k1_fe_verify(a); +#endif + return (t[0] | t[1] | t[2] | t[3] | t[4]) == 0; +} + +SECP256K1_INLINE static int secp256k1_fe_is_odd(const secp256k1_fe *a) { +#ifdef VERIFY + VERIFY_CHECK(a->normalized); + secp256k1_fe_verify(a); +#endif + return a->n[0] & 1; +} + +SECP256K1_INLINE static void secp256k1_fe_clear(secp256k1_fe *a) { + int i; +#ifdef VERIFY + a->magnitude = 0; + a->normalized = 1; +#endif + for (i=0; i<5; i++) { + a->n[i] = 0; + } +} + +static int secp256k1_fe_cmp_var(const secp256k1_fe *a, const secp256k1_fe *b) { + int i; +#ifdef VERIFY + VERIFY_CHECK(a->normalized); + VERIFY_CHECK(b->normalized); + secp256k1_fe_verify(a); + secp256k1_fe_verify(b); +#endif + for (i = 4; i >= 0; i--) { + if (a->n[i] > b->n[i]) { + return 1; + } + if (a->n[i] < b->n[i]) { + return -1; + } + } + return 0; +} + +static int secp256k1_fe_set_b32(secp256k1_fe *r, const unsigned char *a) { + r->n[0] = (uint64_t)a[31] + | ((uint64_t)a[30] << 8) + | ((uint64_t)a[29] << 16) + | ((uint64_t)a[28] << 24) + | ((uint64_t)a[27] << 32) + | ((uint64_t)a[26] << 40) + | ((uint64_t)(a[25] & 0xF) << 48); + r->n[1] = (uint64_t)((a[25] >> 4) & 0xF) + | ((uint64_t)a[24] << 4) + | ((uint64_t)a[23] << 12) + | ((uint64_t)a[22] << 20) + | ((uint64_t)a[21] << 28) + | ((uint64_t)a[20] << 36) + | ((uint64_t)a[19] << 44); + r->n[2] = (uint64_t)a[18] + | ((uint64_t)a[17] << 8) + | ((uint64_t)a[16] << 16) + | ((uint64_t)a[15] << 24) + | ((uint64_t)a[14] << 32) + | ((uint64_t)a[13] << 40) + | ((uint64_t)(a[12] & 0xF) << 48); + r->n[3] = (uint64_t)((a[12] >> 4) & 0xF) + | ((uint64_t)a[11] << 4) + | ((uint64_t)a[10] << 12) + | ((uint64_t)a[9] << 20) + | ((uint64_t)a[8] << 28) + | ((uint64_t)a[7] << 36) + | ((uint64_t)a[6] << 44); + r->n[4] = (uint64_t)a[5] + | ((uint64_t)a[4] << 8) + | ((uint64_t)a[3] << 16) + | ((uint64_t)a[2] << 24) + | ((uint64_t)a[1] << 32) + | ((uint64_t)a[0] << 40); + if (r->n[4] == 0x0FFFFFFFFFFFFULL && (r->n[3] & r->n[2] & r->n[1]) == 0xFFFFFFFFFFFFFULL && r->n[0] >= 0xFFFFEFFFFFC2FULL) { + return 0; + } +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 1; + secp256k1_fe_verify(r); +#endif + return 1; +} + +/** Convert a field element to a 32-byte big endian value. Requires the input to be normalized */ +static void secp256k1_fe_get_b32(unsigned char *r, const secp256k1_fe *a) { +#ifdef VERIFY + VERIFY_CHECK(a->normalized); + secp256k1_fe_verify(a); +#endif + r[0] = (a->n[4] >> 40) & 0xFF; + r[1] = (a->n[4] >> 32) & 0xFF; + r[2] = (a->n[4] >> 24) & 0xFF; + r[3] = (a->n[4] >> 16) & 0xFF; + r[4] = (a->n[4] >> 8) & 0xFF; + r[5] = a->n[4] & 0xFF; + r[6] = (a->n[3] >> 44) & 0xFF; + r[7] = (a->n[3] >> 36) & 0xFF; + r[8] = (a->n[3] >> 28) & 0xFF; + r[9] = (a->n[3] >> 20) & 0xFF; + r[10] = (a->n[3] >> 12) & 0xFF; + r[11] = (a->n[3] >> 4) & 0xFF; + r[12] = ((a->n[2] >> 48) & 0xF) | ((a->n[3] & 0xF) << 4); + r[13] = (a->n[2] >> 40) & 0xFF; + r[14] = (a->n[2] >> 32) & 0xFF; + r[15] = (a->n[2] >> 24) & 0xFF; + r[16] = (a->n[2] >> 16) & 0xFF; + r[17] = (a->n[2] >> 8) & 0xFF; + r[18] = a->n[2] & 0xFF; + r[19] = (a->n[1] >> 44) & 0xFF; + r[20] = (a->n[1] >> 36) & 0xFF; + r[21] = (a->n[1] >> 28) & 0xFF; + r[22] = (a->n[1] >> 20) & 0xFF; + r[23] = (a->n[1] >> 12) & 0xFF; + r[24] = (a->n[1] >> 4) & 0xFF; + r[25] = ((a->n[0] >> 48) & 0xF) | ((a->n[1] & 0xF) << 4); + r[26] = (a->n[0] >> 40) & 0xFF; + r[27] = (a->n[0] >> 32) & 0xFF; + r[28] = (a->n[0] >> 24) & 0xFF; + r[29] = (a->n[0] >> 16) & 0xFF; + r[30] = (a->n[0] >> 8) & 0xFF; + r[31] = a->n[0] & 0xFF; +} + +SECP256K1_INLINE static void secp256k1_fe_negate(secp256k1_fe *r, const secp256k1_fe *a, int m) { +#ifdef VERIFY + VERIFY_CHECK(a->magnitude <= m); + secp256k1_fe_verify(a); +#endif + r->n[0] = 0xFFFFEFFFFFC2FULL * 2 * (m + 1) - a->n[0]; + r->n[1] = 0xFFFFFFFFFFFFFULL * 2 * (m + 1) - a->n[1]; + r->n[2] = 0xFFFFFFFFFFFFFULL * 2 * (m + 1) - a->n[2]; + r->n[3] = 0xFFFFFFFFFFFFFULL * 2 * (m + 1) - a->n[3]; + r->n[4] = 0x0FFFFFFFFFFFFULL * 2 * (m + 1) - a->n[4]; +#ifdef VERIFY + r->magnitude = m + 1; + r->normalized = 0; + secp256k1_fe_verify(r); +#endif +} + +SECP256K1_INLINE static void secp256k1_fe_mul_int(secp256k1_fe *r, int a) { + r->n[0] *= a; + r->n[1] *= a; + r->n[2] *= a; + r->n[3] *= a; + r->n[4] *= a; +#ifdef VERIFY + r->magnitude *= a; + r->normalized = 0; + secp256k1_fe_verify(r); +#endif +} + +SECP256K1_INLINE static void secp256k1_fe_add(secp256k1_fe *r, const secp256k1_fe *a) { +#ifdef VERIFY + secp256k1_fe_verify(a); +#endif + r->n[0] += a->n[0]; + r->n[1] += a->n[1]; + r->n[2] += a->n[2]; + r->n[3] += a->n[3]; + r->n[4] += a->n[4]; +#ifdef VERIFY + r->magnitude += a->magnitude; + r->normalized = 0; + secp256k1_fe_verify(r); +#endif +} + +static void secp256k1_fe_mul(secp256k1_fe *r, const secp256k1_fe *a, const secp256k1_fe * SECP256K1_RESTRICT b) { +#ifdef VERIFY + VERIFY_CHECK(a->magnitude <= 8); + VERIFY_CHECK(b->magnitude <= 8); + secp256k1_fe_verify(a); + secp256k1_fe_verify(b); + VERIFY_CHECK(r != b); + VERIFY_CHECK(a != b); +#endif + secp256k1_fe_mul_inner(r->n, a->n, b->n); +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 0; + secp256k1_fe_verify(r); +#endif +} + +static void secp256k1_fe_sqr(secp256k1_fe *r, const secp256k1_fe *a) { +#ifdef VERIFY + VERIFY_CHECK(a->magnitude <= 8); + secp256k1_fe_verify(a); +#endif + secp256k1_fe_sqr_inner(r->n, a->n); +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 0; + secp256k1_fe_verify(r); +#endif +} + +static SECP256K1_INLINE void secp256k1_fe_cmov(secp256k1_fe *r, const secp256k1_fe *a, int flag) { + uint64_t mask0, mask1; + mask0 = flag + ~((uint64_t)0); + mask1 = ~mask0; + r->n[0] = (r->n[0] & mask0) | (a->n[0] & mask1); + r->n[1] = (r->n[1] & mask0) | (a->n[1] & mask1); + r->n[2] = (r->n[2] & mask0) | (a->n[2] & mask1); + r->n[3] = (r->n[3] & mask0) | (a->n[3] & mask1); + r->n[4] = (r->n[4] & mask0) | (a->n[4] & mask1); +#ifdef VERIFY + if (a->magnitude > r->magnitude) { + r->magnitude = a->magnitude; + } + r->normalized &= a->normalized; +#endif +} + +static SECP256K1_INLINE void secp256k1_fe_storage_cmov(secp256k1_fe_storage *r, const secp256k1_fe_storage *a, int flag) { + uint64_t mask0, mask1; + mask0 = flag + ~((uint64_t)0); + mask1 = ~mask0; + r->n[0] = (r->n[0] & mask0) | (a->n[0] & mask1); + r->n[1] = (r->n[1] & mask0) | (a->n[1] & mask1); + r->n[2] = (r->n[2] & mask0) | (a->n[2] & mask1); + r->n[3] = (r->n[3] & mask0) | (a->n[3] & mask1); +} + +static void secp256k1_fe_to_storage(secp256k1_fe_storage *r, const secp256k1_fe *a) { +#ifdef VERIFY + VERIFY_CHECK(a->normalized); +#endif + r->n[0] = a->n[0] | a->n[1] << 52; + r->n[1] = a->n[1] >> 12 | a->n[2] << 40; + r->n[2] = a->n[2] >> 24 | a->n[3] << 28; + r->n[3] = a->n[3] >> 36 | a->n[4] << 16; +} + +static SECP256K1_INLINE void secp256k1_fe_from_storage(secp256k1_fe *r, const secp256k1_fe_storage *a) { + r->n[0] = a->n[0] & 0xFFFFFFFFFFFFFULL; + r->n[1] = a->n[0] >> 52 | ((a->n[1] << 12) & 0xFFFFFFFFFFFFFULL); + r->n[2] = a->n[1] >> 40 | ((a->n[2] << 24) & 0xFFFFFFFFFFFFFULL); + r->n[3] = a->n[2] >> 28 | ((a->n[3] << 36) & 0xFFFFFFFFFFFFFULL); + r->n[4] = a->n[3] >> 16; +#ifdef VERIFY + r->magnitude = 1; + r->normalized = 1; +#endif +} + +#endif /* SECP256K1_FIELD_REPR_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/field_5x52_int128_impl.h b/node_modules/secp256k1/src/secp256k1/src/field_5x52_int128_impl.h new file mode 100644 index 0000000..bcbfb92 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/field_5x52_int128_impl.h @@ -0,0 +1,279 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_FIELD_INNER5X52_IMPL_H +#define SECP256K1_FIELD_INNER5X52_IMPL_H + +#include + +#ifdef VERIFY +#define VERIFY_BITS(x, n) VERIFY_CHECK(((x) >> (n)) == 0) +#else +#define VERIFY_BITS(x, n) do { } while(0) +#endif + +SECP256K1_INLINE static void secp256k1_fe_mul_inner(uint64_t *r, const uint64_t *a, const uint64_t * SECP256K1_RESTRICT b) { + uint128_t c, d; + uint64_t t3, t4, tx, u0; + uint64_t a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3], a4 = a[4]; + const uint64_t M = 0xFFFFFFFFFFFFFULL, R = 0x1000003D10ULL; + + VERIFY_BITS(a[0], 56); + VERIFY_BITS(a[1], 56); + VERIFY_BITS(a[2], 56); + VERIFY_BITS(a[3], 56); + VERIFY_BITS(a[4], 52); + VERIFY_BITS(b[0], 56); + VERIFY_BITS(b[1], 56); + VERIFY_BITS(b[2], 56); + VERIFY_BITS(b[3], 56); + VERIFY_BITS(b[4], 52); + VERIFY_CHECK(r != b); + VERIFY_CHECK(a != b); + + /* [... a b c] is a shorthand for ... + a<<104 + b<<52 + c<<0 mod n. + * for 0 <= x <= 4, px is a shorthand for sum(a[i]*b[x-i], i=0..x). + * for 4 <= x <= 8, px is a shorthand for sum(a[i]*b[x-i], i=(x-4)..4) + * Note that [x 0 0 0 0 0] = [x*R]. + */ + + d = (uint128_t)a0 * b[3] + + (uint128_t)a1 * b[2] + + (uint128_t)a2 * b[1] + + (uint128_t)a3 * b[0]; + VERIFY_BITS(d, 114); + /* [d 0 0 0] = [p3 0 0 0] */ + c = (uint128_t)a4 * b[4]; + VERIFY_BITS(c, 112); + /* [c 0 0 0 0 d 0 0 0] = [p8 0 0 0 0 p3 0 0 0] */ + d += (c & M) * R; c >>= 52; + VERIFY_BITS(d, 115); + VERIFY_BITS(c, 60); + /* [c 0 0 0 0 0 d 0 0 0] = [p8 0 0 0 0 p3 0 0 0] */ + t3 = d & M; d >>= 52; + VERIFY_BITS(t3, 52); + VERIFY_BITS(d, 63); + /* [c 0 0 0 0 d t3 0 0 0] = [p8 0 0 0 0 p3 0 0 0] */ + + d += (uint128_t)a0 * b[4] + + (uint128_t)a1 * b[3] + + (uint128_t)a2 * b[2] + + (uint128_t)a3 * b[1] + + (uint128_t)a4 * b[0]; + VERIFY_BITS(d, 115); + /* [c 0 0 0 0 d t3 0 0 0] = [p8 0 0 0 p4 p3 0 0 0] */ + d += c * R; + VERIFY_BITS(d, 116); + /* [d t3 0 0 0] = [p8 0 0 0 p4 p3 0 0 0] */ + t4 = d & M; d >>= 52; + VERIFY_BITS(t4, 52); + VERIFY_BITS(d, 64); + /* [d t4 t3 0 0 0] = [p8 0 0 0 p4 p3 0 0 0] */ + tx = (t4 >> 48); t4 &= (M >> 4); + VERIFY_BITS(tx, 4); + VERIFY_BITS(t4, 48); + /* [d t4+(tx<<48) t3 0 0 0] = [p8 0 0 0 p4 p3 0 0 0] */ + + c = (uint128_t)a0 * b[0]; + VERIFY_BITS(c, 112); + /* [d t4+(tx<<48) t3 0 0 c] = [p8 0 0 0 p4 p3 0 0 p0] */ + d += (uint128_t)a1 * b[4] + + (uint128_t)a2 * b[3] + + (uint128_t)a3 * b[2] + + (uint128_t)a4 * b[1]; + VERIFY_BITS(d, 115); + /* [d t4+(tx<<48) t3 0 0 c] = [p8 0 0 p5 p4 p3 0 0 p0] */ + u0 = d & M; d >>= 52; + VERIFY_BITS(u0, 52); + VERIFY_BITS(d, 63); + /* [d u0 t4+(tx<<48) t3 0 0 c] = [p8 0 0 p5 p4 p3 0 0 p0] */ + /* [d 0 t4+(tx<<48)+(u0<<52) t3 0 0 c] = [p8 0 0 p5 p4 p3 0 0 p0] */ + u0 = (u0 << 4) | tx; + VERIFY_BITS(u0, 56); + /* [d 0 t4+(u0<<48) t3 0 0 c] = [p8 0 0 p5 p4 p3 0 0 p0] */ + c += (uint128_t)u0 * (R >> 4); + VERIFY_BITS(c, 115); + /* [d 0 t4 t3 0 0 c] = [p8 0 0 p5 p4 p3 0 0 p0] */ + r[0] = c & M; c >>= 52; + VERIFY_BITS(r[0], 52); + VERIFY_BITS(c, 61); + /* [d 0 t4 t3 0 c r0] = [p8 0 0 p5 p4 p3 0 0 p0] */ + + c += (uint128_t)a0 * b[1] + + (uint128_t)a1 * b[0]; + VERIFY_BITS(c, 114); + /* [d 0 t4 t3 0 c r0] = [p8 0 0 p5 p4 p3 0 p1 p0] */ + d += (uint128_t)a2 * b[4] + + (uint128_t)a3 * b[3] + + (uint128_t)a4 * b[2]; + VERIFY_BITS(d, 114); + /* [d 0 t4 t3 0 c r0] = [p8 0 p6 p5 p4 p3 0 p1 p0] */ + c += (d & M) * R; d >>= 52; + VERIFY_BITS(c, 115); + VERIFY_BITS(d, 62); + /* [d 0 0 t4 t3 0 c r0] = [p8 0 p6 p5 p4 p3 0 p1 p0] */ + r[1] = c & M; c >>= 52; + VERIFY_BITS(r[1], 52); + VERIFY_BITS(c, 63); + /* [d 0 0 t4 t3 c r1 r0] = [p8 0 p6 p5 p4 p3 0 p1 p0] */ + + c += (uint128_t)a0 * b[2] + + (uint128_t)a1 * b[1] + + (uint128_t)a2 * b[0]; + VERIFY_BITS(c, 114); + /* [d 0 0 t4 t3 c r1 r0] = [p8 0 p6 p5 p4 p3 p2 p1 p0] */ + d += (uint128_t)a3 * b[4] + + (uint128_t)a4 * b[3]; + VERIFY_BITS(d, 114); + /* [d 0 0 t4 t3 c t1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + c += (d & M) * R; d >>= 52; + VERIFY_BITS(c, 115); + VERIFY_BITS(d, 62); + /* [d 0 0 0 t4 t3 c r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + + /* [d 0 0 0 t4 t3 c r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[2] = c & M; c >>= 52; + VERIFY_BITS(r[2], 52); + VERIFY_BITS(c, 63); + /* [d 0 0 0 t4 t3+c r2 r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + c += d * R + t3; + VERIFY_BITS(c, 100); + /* [t4 c r2 r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[3] = c & M; c >>= 52; + VERIFY_BITS(r[3], 52); + VERIFY_BITS(c, 48); + /* [t4+c r3 r2 r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + c += t4; + VERIFY_BITS(c, 49); + /* [c r3 r2 r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[4] = c; + VERIFY_BITS(r[4], 49); + /* [r4 r3 r2 r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ +} + +SECP256K1_INLINE static void secp256k1_fe_sqr_inner(uint64_t *r, const uint64_t *a) { + uint128_t c, d; + uint64_t a0 = a[0], a1 = a[1], a2 = a[2], a3 = a[3], a4 = a[4]; + int64_t t3, t4, tx, u0; + const uint64_t M = 0xFFFFFFFFFFFFFULL, R = 0x1000003D10ULL; + + VERIFY_BITS(a[0], 56); + VERIFY_BITS(a[1], 56); + VERIFY_BITS(a[2], 56); + VERIFY_BITS(a[3], 56); + VERIFY_BITS(a[4], 52); + + /** [... a b c] is a shorthand for ... + a<<104 + b<<52 + c<<0 mod n. + * px is a shorthand for sum(a[i]*a[x-i], i=0..x). + * Note that [x 0 0 0 0 0] = [x*R]. + */ + + d = (uint128_t)(a0*2) * a3 + + (uint128_t)(a1*2) * a2; + VERIFY_BITS(d, 114); + /* [d 0 0 0] = [p3 0 0 0] */ + c = (uint128_t)a4 * a4; + VERIFY_BITS(c, 112); + /* [c 0 0 0 0 d 0 0 0] = [p8 0 0 0 0 p3 0 0 0] */ + d += (c & M) * R; c >>= 52; + VERIFY_BITS(d, 115); + VERIFY_BITS(c, 60); + /* [c 0 0 0 0 0 d 0 0 0] = [p8 0 0 0 0 p3 0 0 0] */ + t3 = d & M; d >>= 52; + VERIFY_BITS(t3, 52); + VERIFY_BITS(d, 63); + /* [c 0 0 0 0 d t3 0 0 0] = [p8 0 0 0 0 p3 0 0 0] */ + + a4 *= 2; + d += (uint128_t)a0 * a4 + + (uint128_t)(a1*2) * a3 + + (uint128_t)a2 * a2; + VERIFY_BITS(d, 115); + /* [c 0 0 0 0 d t3 0 0 0] = [p8 0 0 0 p4 p3 0 0 0] */ + d += c * R; + VERIFY_BITS(d, 116); + /* [d t3 0 0 0] = [p8 0 0 0 p4 p3 0 0 0] */ + t4 = d & M; d >>= 52; + VERIFY_BITS(t4, 52); + VERIFY_BITS(d, 64); + /* [d t4 t3 0 0 0] = [p8 0 0 0 p4 p3 0 0 0] */ + tx = (t4 >> 48); t4 &= (M >> 4); + VERIFY_BITS(tx, 4); + VERIFY_BITS(t4, 48); + /* [d t4+(tx<<48) t3 0 0 0] = [p8 0 0 0 p4 p3 0 0 0] */ + + c = (uint128_t)a0 * a0; + VERIFY_BITS(c, 112); + /* [d t4+(tx<<48) t3 0 0 c] = [p8 0 0 0 p4 p3 0 0 p0] */ + d += (uint128_t)a1 * a4 + + (uint128_t)(a2*2) * a3; + VERIFY_BITS(d, 114); + /* [d t4+(tx<<48) t3 0 0 c] = [p8 0 0 p5 p4 p3 0 0 p0] */ + u0 = d & M; d >>= 52; + VERIFY_BITS(u0, 52); + VERIFY_BITS(d, 62); + /* [d u0 t4+(tx<<48) t3 0 0 c] = [p8 0 0 p5 p4 p3 0 0 p0] */ + /* [d 0 t4+(tx<<48)+(u0<<52) t3 0 0 c] = [p8 0 0 p5 p4 p3 0 0 p0] */ + u0 = (u0 << 4) | tx; + VERIFY_BITS(u0, 56); + /* [d 0 t4+(u0<<48) t3 0 0 c] = [p8 0 0 p5 p4 p3 0 0 p0] */ + c += (uint128_t)u0 * (R >> 4); + VERIFY_BITS(c, 113); + /* [d 0 t4 t3 0 0 c] = [p8 0 0 p5 p4 p3 0 0 p0] */ + r[0] = c & M; c >>= 52; + VERIFY_BITS(r[0], 52); + VERIFY_BITS(c, 61); + /* [d 0 t4 t3 0 c r0] = [p8 0 0 p5 p4 p3 0 0 p0] */ + + a0 *= 2; + c += (uint128_t)a0 * a1; + VERIFY_BITS(c, 114); + /* [d 0 t4 t3 0 c r0] = [p8 0 0 p5 p4 p3 0 p1 p0] */ + d += (uint128_t)a2 * a4 + + (uint128_t)a3 * a3; + VERIFY_BITS(d, 114); + /* [d 0 t4 t3 0 c r0] = [p8 0 p6 p5 p4 p3 0 p1 p0] */ + c += (d & M) * R; d >>= 52; + VERIFY_BITS(c, 115); + VERIFY_BITS(d, 62); + /* [d 0 0 t4 t3 0 c r0] = [p8 0 p6 p5 p4 p3 0 p1 p0] */ + r[1] = c & M; c >>= 52; + VERIFY_BITS(r[1], 52); + VERIFY_BITS(c, 63); + /* [d 0 0 t4 t3 c r1 r0] = [p8 0 p6 p5 p4 p3 0 p1 p0] */ + + c += (uint128_t)a0 * a2 + + (uint128_t)a1 * a1; + VERIFY_BITS(c, 114); + /* [d 0 0 t4 t3 c r1 r0] = [p8 0 p6 p5 p4 p3 p2 p1 p0] */ + d += (uint128_t)a3 * a4; + VERIFY_BITS(d, 114); + /* [d 0 0 t4 t3 c r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + c += (d & M) * R; d >>= 52; + VERIFY_BITS(c, 115); + VERIFY_BITS(d, 62); + /* [d 0 0 0 t4 t3 c r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[2] = c & M; c >>= 52; + VERIFY_BITS(r[2], 52); + VERIFY_BITS(c, 63); + /* [d 0 0 0 t4 t3+c r2 r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + + c += d * R + t3; + VERIFY_BITS(c, 100); + /* [t4 c r2 r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[3] = c & M; c >>= 52; + VERIFY_BITS(r[3], 52); + VERIFY_BITS(c, 48); + /* [t4+c r3 r2 r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + c += t4; + VERIFY_BITS(c, 49); + /* [c r3 r2 r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ + r[4] = c; + VERIFY_BITS(r[4], 49); + /* [r4 r3 r2 r1 r0] = [p8 p7 p6 p5 p4 p3 p2 p1 p0] */ +} + +#endif /* SECP256K1_FIELD_INNER5X52_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/field_impl.h b/node_modules/secp256k1/src/secp256k1/src/field_impl.h new file mode 100644 index 0000000..6070cac --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/field_impl.h @@ -0,0 +1,318 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_FIELD_IMPL_H +#define SECP256K1_FIELD_IMPL_H + +#if defined HAVE_CONFIG_H +#include "libsecp256k1-config.h" +#endif + +#include "util.h" +#include "num.h" + +#if defined(USE_FIELD_10X26) +#include "field_10x26_impl.h" +#elif defined(USE_FIELD_5X52) +#include "field_5x52_impl.h" +#else +#error "Please select field implementation" +#endif + +SECP256K1_INLINE static int secp256k1_fe_equal(const secp256k1_fe *a, const secp256k1_fe *b) { + secp256k1_fe na; + secp256k1_fe_negate(&na, a, 1); + secp256k1_fe_add(&na, b); + return secp256k1_fe_normalizes_to_zero(&na); +} + +SECP256K1_INLINE static int secp256k1_fe_equal_var(const secp256k1_fe *a, const secp256k1_fe *b) { + secp256k1_fe na; + secp256k1_fe_negate(&na, a, 1); + secp256k1_fe_add(&na, b); + return secp256k1_fe_normalizes_to_zero_var(&na); +} + +static int secp256k1_fe_sqrt(secp256k1_fe *r, const secp256k1_fe *a) { + /** Given that p is congruent to 3 mod 4, we can compute the square root of + * a mod p as the (p+1)/4'th power of a. + * + * As (p+1)/4 is an even number, it will have the same result for a and for + * (-a). Only one of these two numbers actually has a square root however, + * so we test at the end by squaring and comparing to the input. + * Also because (p+1)/4 is an even number, the computed square root is + * itself always a square (a ** ((p+1)/4) is the square of a ** ((p+1)/8)). + */ + secp256k1_fe x2, x3, x6, x9, x11, x22, x44, x88, x176, x220, x223, t1; + int j; + + VERIFY_CHECK(r != a); + + /** The binary representation of (p + 1)/4 has 3 blocks of 1s, with lengths in + * { 2, 22, 223 }. Use an addition chain to calculate 2^n - 1 for each block: + * 1, [2], 3, 6, 9, 11, [22], 44, 88, 176, 220, [223] + */ + + secp256k1_fe_sqr(&x2, a); + secp256k1_fe_mul(&x2, &x2, a); + + secp256k1_fe_sqr(&x3, &x2); + secp256k1_fe_mul(&x3, &x3, a); + + x6 = x3; + for (j=0; j<3; j++) { + secp256k1_fe_sqr(&x6, &x6); + } + secp256k1_fe_mul(&x6, &x6, &x3); + + x9 = x6; + for (j=0; j<3; j++) { + secp256k1_fe_sqr(&x9, &x9); + } + secp256k1_fe_mul(&x9, &x9, &x3); + + x11 = x9; + for (j=0; j<2; j++) { + secp256k1_fe_sqr(&x11, &x11); + } + secp256k1_fe_mul(&x11, &x11, &x2); + + x22 = x11; + for (j=0; j<11; j++) { + secp256k1_fe_sqr(&x22, &x22); + } + secp256k1_fe_mul(&x22, &x22, &x11); + + x44 = x22; + for (j=0; j<22; j++) { + secp256k1_fe_sqr(&x44, &x44); + } + secp256k1_fe_mul(&x44, &x44, &x22); + + x88 = x44; + for (j=0; j<44; j++) { + secp256k1_fe_sqr(&x88, &x88); + } + secp256k1_fe_mul(&x88, &x88, &x44); + + x176 = x88; + for (j=0; j<88; j++) { + secp256k1_fe_sqr(&x176, &x176); + } + secp256k1_fe_mul(&x176, &x176, &x88); + + x220 = x176; + for (j=0; j<44; j++) { + secp256k1_fe_sqr(&x220, &x220); + } + secp256k1_fe_mul(&x220, &x220, &x44); + + x223 = x220; + for (j=0; j<3; j++) { + secp256k1_fe_sqr(&x223, &x223); + } + secp256k1_fe_mul(&x223, &x223, &x3); + + /* The final result is then assembled using a sliding window over the blocks. */ + + t1 = x223; + for (j=0; j<23; j++) { + secp256k1_fe_sqr(&t1, &t1); + } + secp256k1_fe_mul(&t1, &t1, &x22); + for (j=0; j<6; j++) { + secp256k1_fe_sqr(&t1, &t1); + } + secp256k1_fe_mul(&t1, &t1, &x2); + secp256k1_fe_sqr(&t1, &t1); + secp256k1_fe_sqr(r, &t1); + + /* Check that a square root was actually calculated */ + + secp256k1_fe_sqr(&t1, r); + return secp256k1_fe_equal(&t1, a); +} + +static void secp256k1_fe_inv(secp256k1_fe *r, const secp256k1_fe *a) { + secp256k1_fe x2, x3, x6, x9, x11, x22, x44, x88, x176, x220, x223, t1; + int j; + + /** The binary representation of (p - 2) has 5 blocks of 1s, with lengths in + * { 1, 2, 22, 223 }. Use an addition chain to calculate 2^n - 1 for each block: + * [1], [2], 3, 6, 9, 11, [22], 44, 88, 176, 220, [223] + */ + + secp256k1_fe_sqr(&x2, a); + secp256k1_fe_mul(&x2, &x2, a); + + secp256k1_fe_sqr(&x3, &x2); + secp256k1_fe_mul(&x3, &x3, a); + + x6 = x3; + for (j=0; j<3; j++) { + secp256k1_fe_sqr(&x6, &x6); + } + secp256k1_fe_mul(&x6, &x6, &x3); + + x9 = x6; + for (j=0; j<3; j++) { + secp256k1_fe_sqr(&x9, &x9); + } + secp256k1_fe_mul(&x9, &x9, &x3); + + x11 = x9; + for (j=0; j<2; j++) { + secp256k1_fe_sqr(&x11, &x11); + } + secp256k1_fe_mul(&x11, &x11, &x2); + + x22 = x11; + for (j=0; j<11; j++) { + secp256k1_fe_sqr(&x22, &x22); + } + secp256k1_fe_mul(&x22, &x22, &x11); + + x44 = x22; + for (j=0; j<22; j++) { + secp256k1_fe_sqr(&x44, &x44); + } + secp256k1_fe_mul(&x44, &x44, &x22); + + x88 = x44; + for (j=0; j<44; j++) { + secp256k1_fe_sqr(&x88, &x88); + } + secp256k1_fe_mul(&x88, &x88, &x44); + + x176 = x88; + for (j=0; j<88; j++) { + secp256k1_fe_sqr(&x176, &x176); + } + secp256k1_fe_mul(&x176, &x176, &x88); + + x220 = x176; + for (j=0; j<44; j++) { + secp256k1_fe_sqr(&x220, &x220); + } + secp256k1_fe_mul(&x220, &x220, &x44); + + x223 = x220; + for (j=0; j<3; j++) { + secp256k1_fe_sqr(&x223, &x223); + } + secp256k1_fe_mul(&x223, &x223, &x3); + + /* The final result is then assembled using a sliding window over the blocks. */ + + t1 = x223; + for (j=0; j<23; j++) { + secp256k1_fe_sqr(&t1, &t1); + } + secp256k1_fe_mul(&t1, &t1, &x22); + for (j=0; j<5; j++) { + secp256k1_fe_sqr(&t1, &t1); + } + secp256k1_fe_mul(&t1, &t1, a); + for (j=0; j<3; j++) { + secp256k1_fe_sqr(&t1, &t1); + } + secp256k1_fe_mul(&t1, &t1, &x2); + for (j=0; j<2; j++) { + secp256k1_fe_sqr(&t1, &t1); + } + secp256k1_fe_mul(r, a, &t1); +} + +static void secp256k1_fe_inv_var(secp256k1_fe *r, const secp256k1_fe *a) { +#if defined(USE_FIELD_INV_BUILTIN) + secp256k1_fe_inv(r, a); +#elif defined(USE_FIELD_INV_NUM) + secp256k1_num n, m; + static const secp256k1_fe negone = SECP256K1_FE_CONST( + 0xFFFFFFFFUL, 0xFFFFFFFFUL, 0xFFFFFFFFUL, 0xFFFFFFFFUL, + 0xFFFFFFFFUL, 0xFFFFFFFFUL, 0xFFFFFFFEUL, 0xFFFFFC2EUL + ); + /* secp256k1 field prime, value p defined in "Standards for Efficient Cryptography" (SEC2) 2.7.1. */ + static const unsigned char prime[32] = { + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFE,0xFF,0xFF,0xFC,0x2F + }; + unsigned char b[32]; + int res; + secp256k1_fe c = *a; + secp256k1_fe_normalize_var(&c); + secp256k1_fe_get_b32(b, &c); + secp256k1_num_set_bin(&n, b, 32); + secp256k1_num_set_bin(&m, prime, 32); + secp256k1_num_mod_inverse(&n, &n, &m); + secp256k1_num_get_bin(b, 32, &n); + res = secp256k1_fe_set_b32(r, b); + (void)res; + VERIFY_CHECK(res); + /* Verify the result is the (unique) valid inverse using non-GMP code. */ + secp256k1_fe_mul(&c, &c, r); + secp256k1_fe_add(&c, &negone); + CHECK(secp256k1_fe_normalizes_to_zero_var(&c)); +#else +#error "Please select field inverse implementation" +#endif +} + +static void secp256k1_fe_inv_all_var(secp256k1_fe *r, const secp256k1_fe *a, size_t len) { + secp256k1_fe u; + size_t i; + if (len < 1) { + return; + } + + VERIFY_CHECK((r + len <= a) || (a + len <= r)); + + r[0] = a[0]; + + i = 0; + while (++i < len) { + secp256k1_fe_mul(&r[i], &r[i - 1], &a[i]); + } + + secp256k1_fe_inv_var(&u, &r[--i]); + + while (i > 0) { + size_t j = i--; + secp256k1_fe_mul(&r[j], &r[i], &u); + secp256k1_fe_mul(&u, &u, &a[j]); + } + + r[0] = u; +} + +static int secp256k1_fe_is_quad_var(const secp256k1_fe *a) { +#ifndef USE_NUM_NONE + unsigned char b[32]; + secp256k1_num n; + secp256k1_num m; + /* secp256k1 field prime, value p defined in "Standards for Efficient Cryptography" (SEC2) 2.7.1. */ + static const unsigned char prime[32] = { + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFE,0xFF,0xFF,0xFC,0x2F + }; + + secp256k1_fe c = *a; + secp256k1_fe_normalize_var(&c); + secp256k1_fe_get_b32(b, &c); + secp256k1_num_set_bin(&n, b, 32); + secp256k1_num_set_bin(&m, prime, 32); + return secp256k1_num_jacobi(&n, &m) >= 0; +#else + secp256k1_fe r; + return secp256k1_fe_sqrt(&r, a); +#endif +} + +#endif /* SECP256K1_FIELD_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/gen_context.c b/node_modules/secp256k1/src/secp256k1/src/gen_context.c new file mode 100644 index 0000000..539f574 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/gen_context.c @@ -0,0 +1,87 @@ +/********************************************************************** + * Copyright (c) 2013, 2014, 2015 Thomas Daede, Cory Fields * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +// Autotools creates libsecp256k1-config.h, of which ECMULT_GEN_PREC_BITS is needed. +// ifndef guard so downstream users can define their own if they do not use autotools. +#if !defined(ECMULT_GEN_PREC_BITS) +#include "libsecp256k1-config.h" +#endif +#define USE_BASIC_CONFIG 1 +#include "basic-config.h" + +#include "include/secp256k1.h" +#include "util.h" +#include "field_impl.h" +#include "scalar_impl.h" +#include "group_impl.h" +#include "ecmult_gen_impl.h" + +static void default_error_callback_fn(const char* str, void* data) { + (void)data; + fprintf(stderr, "[libsecp256k1] internal consistency check failed: %s\n", str); + abort(); +} + +static const secp256k1_callback default_error_callback = { + default_error_callback_fn, + NULL +}; + +int main(int argc, char **argv) { + secp256k1_ecmult_gen_context ctx; + void *prealloc, *base; + int inner; + int outer; + FILE* fp; + + (void)argc; + (void)argv; + + fp = fopen("src/ecmult_static_context.h","w"); + if (fp == NULL) { + fprintf(stderr, "Could not open src/ecmult_static_context.h for writing!\n"); + return -1; + } + + fprintf(fp, "#ifndef _SECP256K1_ECMULT_STATIC_CONTEXT_\n"); + fprintf(fp, "#define _SECP256K1_ECMULT_STATIC_CONTEXT_\n"); + fprintf(fp, "#include \"src/group.h\"\n"); + fprintf(fp, "#define SC SECP256K1_GE_STORAGE_CONST\n"); + fprintf(fp, "#if ECMULT_GEN_PREC_N != %d || ECMULT_GEN_PREC_G != %d\n", ECMULT_GEN_PREC_N, ECMULT_GEN_PREC_G); + fprintf(fp, " #error configuration mismatch, invalid ECMULT_GEN_PREC_N, ECMULT_GEN_PREC_G. Try deleting ecmult_static_context.h before the build.\n"); + fprintf(fp, "#endif\n"); + fprintf(fp, "static const secp256k1_ge_storage secp256k1_ecmult_static_context[ECMULT_GEN_PREC_N][ECMULT_GEN_PREC_G] = {\n"); + + base = checked_malloc(&default_error_callback, SECP256K1_ECMULT_GEN_CONTEXT_PREALLOCATED_SIZE); + prealloc = base; + secp256k1_ecmult_gen_context_init(&ctx); + secp256k1_ecmult_gen_context_build(&ctx, &prealloc); + for(outer = 0; outer != ECMULT_GEN_PREC_N; outer++) { + fprintf(fp,"{\n"); + for(inner = 0; inner != ECMULT_GEN_PREC_G; inner++) { + fprintf(fp," SC(%uu, %uu, %uu, %uu, %uu, %uu, %uu, %uu, %uu, %uu, %uu, %uu, %uu, %uu, %uu, %uu)", SECP256K1_GE_STORAGE_CONST_GET((*ctx.prec)[outer][inner])); + if (inner != ECMULT_GEN_PREC_G - 1) { + fprintf(fp,",\n"); + } else { + fprintf(fp,"\n"); + } + } + if (outer != ECMULT_GEN_PREC_N - 1) { + fprintf(fp,"},\n"); + } else { + fprintf(fp,"}\n"); + } + } + fprintf(fp,"};\n"); + secp256k1_ecmult_gen_context_clear(&ctx); + free(base); + + fprintf(fp, "#undef SC\n"); + fprintf(fp, "#endif\n"); + fclose(fp); + + return 0; +} diff --git a/node_modules/secp256k1/src/secp256k1/src/group.h b/node_modules/secp256k1/src/secp256k1/src/group.h new file mode 100644 index 0000000..8e122ab --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/group.h @@ -0,0 +1,142 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_GROUP_H +#define SECP256K1_GROUP_H + +#include "num.h" +#include "field.h" + +/** A group element of the secp256k1 curve, in affine coordinates. */ +typedef struct { + secp256k1_fe x; + secp256k1_fe y; + int infinity; /* whether this represents the point at infinity */ +} secp256k1_ge; + +#define SECP256K1_GE_CONST(a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p) {SECP256K1_FE_CONST((a),(b),(c),(d),(e),(f),(g),(h)), SECP256K1_FE_CONST((i),(j),(k),(l),(m),(n),(o),(p)), 0} +#define SECP256K1_GE_CONST_INFINITY {SECP256K1_FE_CONST(0, 0, 0, 0, 0, 0, 0, 0), SECP256K1_FE_CONST(0, 0, 0, 0, 0, 0, 0, 0), 1} + +/** A group element of the secp256k1 curve, in jacobian coordinates. */ +typedef struct { + secp256k1_fe x; /* actual X: x/z^2 */ + secp256k1_fe y; /* actual Y: y/z^3 */ + secp256k1_fe z; + int infinity; /* whether this represents the point at infinity */ +} secp256k1_gej; + +#define SECP256K1_GEJ_CONST(a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p) {SECP256K1_FE_CONST((a),(b),(c),(d),(e),(f),(g),(h)), SECP256K1_FE_CONST((i),(j),(k),(l),(m),(n),(o),(p)), SECP256K1_FE_CONST(0, 0, 0, 0, 0, 0, 0, 1), 0} +#define SECP256K1_GEJ_CONST_INFINITY {SECP256K1_FE_CONST(0, 0, 0, 0, 0, 0, 0, 0), SECP256K1_FE_CONST(0, 0, 0, 0, 0, 0, 0, 0), SECP256K1_FE_CONST(0, 0, 0, 0, 0, 0, 0, 0), 1} + +typedef struct { + secp256k1_fe_storage x; + secp256k1_fe_storage y; +} secp256k1_ge_storage; + +#define SECP256K1_GE_STORAGE_CONST(a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p) {SECP256K1_FE_STORAGE_CONST((a),(b),(c),(d),(e),(f),(g),(h)), SECP256K1_FE_STORAGE_CONST((i),(j),(k),(l),(m),(n),(o),(p))} + +#define SECP256K1_GE_STORAGE_CONST_GET(t) SECP256K1_FE_STORAGE_CONST_GET(t.x), SECP256K1_FE_STORAGE_CONST_GET(t.y) + +/** Set a group element equal to the point with given X and Y coordinates */ +static void secp256k1_ge_set_xy(secp256k1_ge *r, const secp256k1_fe *x, const secp256k1_fe *y); + +/** Set a group element (affine) equal to the point with the given X coordinate + * and a Y coordinate that is a quadratic residue modulo p. The return value + * is true iff a coordinate with the given X coordinate exists. + */ +static int secp256k1_ge_set_xquad(secp256k1_ge *r, const secp256k1_fe *x); + +/** Set a group element (affine) equal to the point with the given X coordinate, and given oddness + * for Y. Return value indicates whether the result is valid. */ +static int secp256k1_ge_set_xo_var(secp256k1_ge *r, const secp256k1_fe *x, int odd); + +/** Check whether a group element is the point at infinity. */ +static int secp256k1_ge_is_infinity(const secp256k1_ge *a); + +/** Check whether a group element is valid (i.e., on the curve). */ +static int secp256k1_ge_is_valid_var(const secp256k1_ge *a); + +static void secp256k1_ge_neg(secp256k1_ge *r, const secp256k1_ge *a); + +/** Set a group element equal to another which is given in jacobian coordinates */ +static void secp256k1_ge_set_gej(secp256k1_ge *r, secp256k1_gej *a); + +/** Set a batch of group elements equal to the inputs given in jacobian coordinates */ +static void secp256k1_ge_set_all_gej_var(secp256k1_ge *r, const secp256k1_gej *a, size_t len); + +/** Bring a batch inputs given in jacobian coordinates (with known z-ratios) to + * the same global z "denominator". zr must contain the known z-ratios such + * that mul(a[i].z, zr[i+1]) == a[i+1].z. zr[0] is ignored. The x and y + * coordinates of the result are stored in r, the common z coordinate is + * stored in globalz. */ +static void secp256k1_ge_globalz_set_table_gej(size_t len, secp256k1_ge *r, secp256k1_fe *globalz, const secp256k1_gej *a, const secp256k1_fe *zr); + +/** Set a group element (affine) equal to the point at infinity. */ +static void secp256k1_ge_set_infinity(secp256k1_ge *r); + +/** Set a group element (jacobian) equal to the point at infinity. */ +static void secp256k1_gej_set_infinity(secp256k1_gej *r); + +/** Set a group element (jacobian) equal to another which is given in affine coordinates. */ +static void secp256k1_gej_set_ge(secp256k1_gej *r, const secp256k1_ge *a); + +/** Compare the X coordinate of a group element (jacobian). */ +static int secp256k1_gej_eq_x_var(const secp256k1_fe *x, const secp256k1_gej *a); + +/** Set r equal to the inverse of a (i.e., mirrored around the X axis) */ +static void secp256k1_gej_neg(secp256k1_gej *r, const secp256k1_gej *a); + +/** Check whether a group element is the point at infinity. */ +static int secp256k1_gej_is_infinity(const secp256k1_gej *a); + +/** Check whether a group element's y coordinate is a quadratic residue. */ +static int secp256k1_gej_has_quad_y_var(const secp256k1_gej *a); + +/** Set r equal to the double of a. If rzr is not-NULL, r->z = a->z * *rzr (where infinity means an implicit z = 0). + * a may not be zero. Constant time. */ +static void secp256k1_gej_double_nonzero(secp256k1_gej *r, const secp256k1_gej *a, secp256k1_fe *rzr); + +/** Set r equal to the double of a. If rzr is not-NULL, r->z = a->z * *rzr (where infinity means an implicit z = 0). */ +static void secp256k1_gej_double_var(secp256k1_gej *r, const secp256k1_gej *a, secp256k1_fe *rzr); + +/** Set r equal to the sum of a and b. If rzr is non-NULL, r->z = a->z * *rzr (a cannot be infinity in that case). */ +static void secp256k1_gej_add_var(secp256k1_gej *r, const secp256k1_gej *a, const secp256k1_gej *b, secp256k1_fe *rzr); + +/** Set r equal to the sum of a and b (with b given in affine coordinates, and not infinity). */ +static void secp256k1_gej_add_ge(secp256k1_gej *r, const secp256k1_gej *a, const secp256k1_ge *b); + +/** Set r equal to the sum of a and b (with b given in affine coordinates). This is more efficient + than secp256k1_gej_add_var. It is identical to secp256k1_gej_add_ge but without constant-time + guarantee, and b is allowed to be infinity. If rzr is non-NULL, r->z = a->z * *rzr (a cannot be infinity in that case). */ +static void secp256k1_gej_add_ge_var(secp256k1_gej *r, const secp256k1_gej *a, const secp256k1_ge *b, secp256k1_fe *rzr); + +/** Set r equal to the sum of a and b (with the inverse of b's Z coordinate passed as bzinv). */ +static void secp256k1_gej_add_zinv_var(secp256k1_gej *r, const secp256k1_gej *a, const secp256k1_ge *b, const secp256k1_fe *bzinv); + +#ifdef USE_ENDOMORPHISM +/** Set r to be equal to lambda times a, where lambda is chosen in a way such that this is very fast. */ +static void secp256k1_ge_mul_lambda(secp256k1_ge *r, const secp256k1_ge *a); +#endif + +/** Clear a secp256k1_gej to prevent leaking sensitive information. */ +static void secp256k1_gej_clear(secp256k1_gej *r); + +/** Clear a secp256k1_ge to prevent leaking sensitive information. */ +static void secp256k1_ge_clear(secp256k1_ge *r); + +/** Convert a group element to the storage type. */ +static void secp256k1_ge_to_storage(secp256k1_ge_storage *r, const secp256k1_ge *a); + +/** Convert a group element back from the storage type. */ +static void secp256k1_ge_from_storage(secp256k1_ge *r, const secp256k1_ge_storage *a); + +/** If flag is true, set *r equal to *a; otherwise leave it. Constant-time. */ +static void secp256k1_ge_storage_cmov(secp256k1_ge_storage *r, const secp256k1_ge_storage *a, int flag); + +/** Rescale a jacobian point by b which must be non-zero. Constant-time. */ +static void secp256k1_gej_rescale(secp256k1_gej *r, const secp256k1_fe *b); + +#endif /* SECP256K1_GROUP_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/group_impl.h b/node_modules/secp256k1/src/secp256k1/src/group_impl.h new file mode 100644 index 0000000..9b93c39 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/group_impl.h @@ -0,0 +1,705 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_GROUP_IMPL_H +#define SECP256K1_GROUP_IMPL_H + +#include "num.h" +#include "field.h" +#include "group.h" + +/* These points can be generated in sage as follows: + * + * 0. Setup a worksheet with the following parameters. + * b = 4 # whatever CURVE_B will be set to + * F = FiniteField (0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFC2F) + * C = EllipticCurve ([F (0), F (b)]) + * + * 1. Determine all the small orders available to you. (If there are + * no satisfactory ones, go back and change b.) + * print C.order().factor(limit=1000) + * + * 2. Choose an order as one of the prime factors listed in the above step. + * (You can also multiply some to get a composite order, though the + * tests will crash trying to invert scalars during signing.) We take a + * random point and scale it to drop its order to the desired value. + * There is some probability this won't work; just try again. + * order = 199 + * P = C.random_point() + * P = (int(P.order()) / int(order)) * P + * assert(P.order() == order) + * + * 3. Print the values. You'll need to use a vim macro or something to + * split the hex output into 4-byte chunks. + * print "%x %x" % P.xy() + */ +#if defined(EXHAUSTIVE_TEST_ORDER) +# if EXHAUSTIVE_TEST_ORDER == 199 +static const secp256k1_ge secp256k1_ge_const_g = SECP256K1_GE_CONST( + 0xFA7CC9A7, 0x0737F2DB, 0xA749DD39, 0x2B4FB069, + 0x3B017A7D, 0xA808C2F1, 0xFB12940C, 0x9EA66C18, + 0x78AC123A, 0x5ED8AEF3, 0x8732BC91, 0x1F3A2868, + 0x48DF246C, 0x808DAE72, 0xCFE52572, 0x7F0501ED +); + +static const int CURVE_B = 4; +# elif EXHAUSTIVE_TEST_ORDER == 13 +static const secp256k1_ge secp256k1_ge_const_g = SECP256K1_GE_CONST( + 0xedc60018, 0xa51a786b, 0x2ea91f4d, 0x4c9416c0, + 0x9de54c3b, 0xa1316554, 0x6cf4345c, 0x7277ef15, + 0x54cb1b6b, 0xdc8c1273, 0x087844ea, 0x43f4603e, + 0x0eaf9a43, 0xf6effe55, 0x939f806d, 0x37adf8ac +); +static const int CURVE_B = 2; +# else +# error No known generator for the specified exhaustive test group order. +# endif +#else +/** Generator for secp256k1, value 'g' defined in + * "Standards for Efficient Cryptography" (SEC2) 2.7.1. + */ +static const secp256k1_ge secp256k1_ge_const_g = SECP256K1_GE_CONST( + 0x79BE667EUL, 0xF9DCBBACUL, 0x55A06295UL, 0xCE870B07UL, + 0x029BFCDBUL, 0x2DCE28D9UL, 0x59F2815BUL, 0x16F81798UL, + 0x483ADA77UL, 0x26A3C465UL, 0x5DA4FBFCUL, 0x0E1108A8UL, + 0xFD17B448UL, 0xA6855419UL, 0x9C47D08FUL, 0xFB10D4B8UL +); + +static const int CURVE_B = 7; +#endif + +static void secp256k1_ge_set_gej_zinv(secp256k1_ge *r, const secp256k1_gej *a, const secp256k1_fe *zi) { + secp256k1_fe zi2; + secp256k1_fe zi3; + secp256k1_fe_sqr(&zi2, zi); + secp256k1_fe_mul(&zi3, &zi2, zi); + secp256k1_fe_mul(&r->x, &a->x, &zi2); + secp256k1_fe_mul(&r->y, &a->y, &zi3); + r->infinity = a->infinity; +} + +static void secp256k1_ge_set_xy(secp256k1_ge *r, const secp256k1_fe *x, const secp256k1_fe *y) { + r->infinity = 0; + r->x = *x; + r->y = *y; +} + +static int secp256k1_ge_is_infinity(const secp256k1_ge *a) { + return a->infinity; +} + +static void secp256k1_ge_neg(secp256k1_ge *r, const secp256k1_ge *a) { + *r = *a; + secp256k1_fe_normalize_weak(&r->y); + secp256k1_fe_negate(&r->y, &r->y, 1); +} + +static void secp256k1_ge_set_gej(secp256k1_ge *r, secp256k1_gej *a) { + secp256k1_fe z2, z3; + r->infinity = a->infinity; + secp256k1_fe_inv(&a->z, &a->z); + secp256k1_fe_sqr(&z2, &a->z); + secp256k1_fe_mul(&z3, &a->z, &z2); + secp256k1_fe_mul(&a->x, &a->x, &z2); + secp256k1_fe_mul(&a->y, &a->y, &z3); + secp256k1_fe_set_int(&a->z, 1); + r->x = a->x; + r->y = a->y; +} + +static void secp256k1_ge_set_gej_var(secp256k1_ge *r, secp256k1_gej *a) { + secp256k1_fe z2, z3; + r->infinity = a->infinity; + if (a->infinity) { + return; + } + secp256k1_fe_inv_var(&a->z, &a->z); + secp256k1_fe_sqr(&z2, &a->z); + secp256k1_fe_mul(&z3, &a->z, &z2); + secp256k1_fe_mul(&a->x, &a->x, &z2); + secp256k1_fe_mul(&a->y, &a->y, &z3); + secp256k1_fe_set_int(&a->z, 1); + r->x = a->x; + r->y = a->y; +} + +static void secp256k1_ge_set_all_gej_var(secp256k1_ge *r, const secp256k1_gej *a, size_t len) { + secp256k1_fe u; + size_t i; + size_t last_i = SIZE_MAX; + + for (i = 0; i < len; i++) { + if (!a[i].infinity) { + /* Use destination's x coordinates as scratch space */ + if (last_i == SIZE_MAX) { + r[i].x = a[i].z; + } else { + secp256k1_fe_mul(&r[i].x, &r[last_i].x, &a[i].z); + } + last_i = i; + } + } + if (last_i == SIZE_MAX) { + return; + } + secp256k1_fe_inv_var(&u, &r[last_i].x); + + i = last_i; + while (i > 0) { + i--; + if (!a[i].infinity) { + secp256k1_fe_mul(&r[last_i].x, &r[i].x, &u); + secp256k1_fe_mul(&u, &u, &a[last_i].z); + last_i = i; + } + } + VERIFY_CHECK(!a[last_i].infinity); + r[last_i].x = u; + + for (i = 0; i < len; i++) { + r[i].infinity = a[i].infinity; + if (!a[i].infinity) { + secp256k1_ge_set_gej_zinv(&r[i], &a[i], &r[i].x); + } + } +} + +static void secp256k1_ge_globalz_set_table_gej(size_t len, secp256k1_ge *r, secp256k1_fe *globalz, const secp256k1_gej *a, const secp256k1_fe *zr) { + size_t i = len - 1; + secp256k1_fe zs; + + if (len > 0) { + /* The z of the final point gives us the "global Z" for the table. */ + r[i].x = a[i].x; + r[i].y = a[i].y; + /* Ensure all y values are in weak normal form for fast negation of points */ + secp256k1_fe_normalize_weak(&r[i].y); + *globalz = a[i].z; + r[i].infinity = 0; + zs = zr[i]; + + /* Work our way backwards, using the z-ratios to scale the x/y values. */ + while (i > 0) { + if (i != len - 1) { + secp256k1_fe_mul(&zs, &zs, &zr[i]); + } + i--; + secp256k1_ge_set_gej_zinv(&r[i], &a[i], &zs); + } + } +} + +static void secp256k1_gej_set_infinity(secp256k1_gej *r) { + r->infinity = 1; + secp256k1_fe_clear(&r->x); + secp256k1_fe_clear(&r->y); + secp256k1_fe_clear(&r->z); +} + +static void secp256k1_ge_set_infinity(secp256k1_ge *r) { + r->infinity = 1; + secp256k1_fe_clear(&r->x); + secp256k1_fe_clear(&r->y); +} + +static void secp256k1_gej_clear(secp256k1_gej *r) { + r->infinity = 0; + secp256k1_fe_clear(&r->x); + secp256k1_fe_clear(&r->y); + secp256k1_fe_clear(&r->z); +} + +static void secp256k1_ge_clear(secp256k1_ge *r) { + r->infinity = 0; + secp256k1_fe_clear(&r->x); + secp256k1_fe_clear(&r->y); +} + +static int secp256k1_ge_set_xquad(secp256k1_ge *r, const secp256k1_fe *x) { + secp256k1_fe x2, x3, c; + r->x = *x; + secp256k1_fe_sqr(&x2, x); + secp256k1_fe_mul(&x3, x, &x2); + r->infinity = 0; + secp256k1_fe_set_int(&c, CURVE_B); + secp256k1_fe_add(&c, &x3); + return secp256k1_fe_sqrt(&r->y, &c); +} + +static int secp256k1_ge_set_xo_var(secp256k1_ge *r, const secp256k1_fe *x, int odd) { + if (!secp256k1_ge_set_xquad(r, x)) { + return 0; + } + secp256k1_fe_normalize_var(&r->y); + if (secp256k1_fe_is_odd(&r->y) != odd) { + secp256k1_fe_negate(&r->y, &r->y, 1); + } + return 1; + +} + +static void secp256k1_gej_set_ge(secp256k1_gej *r, const secp256k1_ge *a) { + r->infinity = a->infinity; + r->x = a->x; + r->y = a->y; + secp256k1_fe_set_int(&r->z, 1); +} + +static int secp256k1_gej_eq_x_var(const secp256k1_fe *x, const secp256k1_gej *a) { + secp256k1_fe r, r2; + VERIFY_CHECK(!a->infinity); + secp256k1_fe_sqr(&r, &a->z); secp256k1_fe_mul(&r, &r, x); + r2 = a->x; secp256k1_fe_normalize_weak(&r2); + return secp256k1_fe_equal_var(&r, &r2); +} + +static void secp256k1_gej_neg(secp256k1_gej *r, const secp256k1_gej *a) { + r->infinity = a->infinity; + r->x = a->x; + r->y = a->y; + r->z = a->z; + secp256k1_fe_normalize_weak(&r->y); + secp256k1_fe_negate(&r->y, &r->y, 1); +} + +static int secp256k1_gej_is_infinity(const secp256k1_gej *a) { + return a->infinity; +} + +static int secp256k1_gej_is_valid_var(const secp256k1_gej *a) { + secp256k1_fe y2, x3, z2, z6; + if (a->infinity) { + return 0; + } + /** y^2 = x^3 + 7 + * (Y/Z^3)^2 = (X/Z^2)^3 + 7 + * Y^2 / Z^6 = X^3 / Z^6 + 7 + * Y^2 = X^3 + 7*Z^6 + */ + secp256k1_fe_sqr(&y2, &a->y); + secp256k1_fe_sqr(&x3, &a->x); secp256k1_fe_mul(&x3, &x3, &a->x); + secp256k1_fe_sqr(&z2, &a->z); + secp256k1_fe_sqr(&z6, &z2); secp256k1_fe_mul(&z6, &z6, &z2); + secp256k1_fe_mul_int(&z6, CURVE_B); + secp256k1_fe_add(&x3, &z6); + secp256k1_fe_normalize_weak(&x3); + return secp256k1_fe_equal_var(&y2, &x3); +} + +static int secp256k1_ge_is_valid_var(const secp256k1_ge *a) { + secp256k1_fe y2, x3, c; + if (a->infinity) { + return 0; + } + /* y^2 = x^3 + 7 */ + secp256k1_fe_sqr(&y2, &a->y); + secp256k1_fe_sqr(&x3, &a->x); secp256k1_fe_mul(&x3, &x3, &a->x); + secp256k1_fe_set_int(&c, CURVE_B); + secp256k1_fe_add(&x3, &c); + secp256k1_fe_normalize_weak(&x3); + return secp256k1_fe_equal_var(&y2, &x3); +} + +static void secp256k1_gej_double_var(secp256k1_gej *r, const secp256k1_gej *a, secp256k1_fe *rzr) { + /* Operations: 3 mul, 4 sqr, 0 normalize, 12 mul_int/add/negate. + * + * Note that there is an implementation described at + * https://hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-0.html#doubling-dbl-2009-l + * which trades a multiply for a square, but in practice this is actually slower, + * mainly because it requires more normalizations. + */ + secp256k1_fe t1,t2,t3,t4; + /** For secp256k1, 2Q is infinity if and only if Q is infinity. This is because if 2Q = infinity, + * Q must equal -Q, or that Q.y == -(Q.y), or Q.y is 0. For a point on y^2 = x^3 + 7 to have + * y=0, x^3 must be -7 mod p. However, -7 has no cube root mod p. + * + * Having said this, if this function receives a point on a sextic twist, e.g. by + * a fault attack, it is possible for y to be 0. This happens for y^2 = x^3 + 6, + * since -6 does have a cube root mod p. For this point, this function will not set + * the infinity flag even though the point doubles to infinity, and the result + * point will be gibberish (z = 0 but infinity = 0). + */ + r->infinity = a->infinity; + if (r->infinity) { + if (rzr != NULL) { + secp256k1_fe_set_int(rzr, 1); + } + return; + } + + if (rzr != NULL) { + *rzr = a->y; + secp256k1_fe_normalize_weak(rzr); + secp256k1_fe_mul_int(rzr, 2); + } + + secp256k1_fe_mul(&r->z, &a->z, &a->y); + secp256k1_fe_mul_int(&r->z, 2); /* Z' = 2*Y*Z (2) */ + secp256k1_fe_sqr(&t1, &a->x); + secp256k1_fe_mul_int(&t1, 3); /* T1 = 3*X^2 (3) */ + secp256k1_fe_sqr(&t2, &t1); /* T2 = 9*X^4 (1) */ + secp256k1_fe_sqr(&t3, &a->y); + secp256k1_fe_mul_int(&t3, 2); /* T3 = 2*Y^2 (2) */ + secp256k1_fe_sqr(&t4, &t3); + secp256k1_fe_mul_int(&t4, 2); /* T4 = 8*Y^4 (2) */ + secp256k1_fe_mul(&t3, &t3, &a->x); /* T3 = 2*X*Y^2 (1) */ + r->x = t3; + secp256k1_fe_mul_int(&r->x, 4); /* X' = 8*X*Y^2 (4) */ + secp256k1_fe_negate(&r->x, &r->x, 4); /* X' = -8*X*Y^2 (5) */ + secp256k1_fe_add(&r->x, &t2); /* X' = 9*X^4 - 8*X*Y^2 (6) */ + secp256k1_fe_negate(&t2, &t2, 1); /* T2 = -9*X^4 (2) */ + secp256k1_fe_mul_int(&t3, 6); /* T3 = 12*X*Y^2 (6) */ + secp256k1_fe_add(&t3, &t2); /* T3 = 12*X*Y^2 - 9*X^4 (8) */ + secp256k1_fe_mul(&r->y, &t1, &t3); /* Y' = 36*X^3*Y^2 - 27*X^6 (1) */ + secp256k1_fe_negate(&t2, &t4, 2); /* T2 = -8*Y^4 (3) */ + secp256k1_fe_add(&r->y, &t2); /* Y' = 36*X^3*Y^2 - 27*X^6 - 8*Y^4 (4) */ +} + +static SECP256K1_INLINE void secp256k1_gej_double_nonzero(secp256k1_gej *r, const secp256k1_gej *a, secp256k1_fe *rzr) { + VERIFY_CHECK(!secp256k1_gej_is_infinity(a)); + secp256k1_gej_double_var(r, a, rzr); +} + +static void secp256k1_gej_add_var(secp256k1_gej *r, const secp256k1_gej *a, const secp256k1_gej *b, secp256k1_fe *rzr) { + /* Operations: 12 mul, 4 sqr, 2 normalize, 12 mul_int/add/negate */ + secp256k1_fe z22, z12, u1, u2, s1, s2, h, i, i2, h2, h3, t; + + if (a->infinity) { + VERIFY_CHECK(rzr == NULL); + *r = *b; + return; + } + + if (b->infinity) { + if (rzr != NULL) { + secp256k1_fe_set_int(rzr, 1); + } + *r = *a; + return; + } + + r->infinity = 0; + secp256k1_fe_sqr(&z22, &b->z); + secp256k1_fe_sqr(&z12, &a->z); + secp256k1_fe_mul(&u1, &a->x, &z22); + secp256k1_fe_mul(&u2, &b->x, &z12); + secp256k1_fe_mul(&s1, &a->y, &z22); secp256k1_fe_mul(&s1, &s1, &b->z); + secp256k1_fe_mul(&s2, &b->y, &z12); secp256k1_fe_mul(&s2, &s2, &a->z); + secp256k1_fe_negate(&h, &u1, 1); secp256k1_fe_add(&h, &u2); + secp256k1_fe_negate(&i, &s1, 1); secp256k1_fe_add(&i, &s2); + if (secp256k1_fe_normalizes_to_zero_var(&h)) { + if (secp256k1_fe_normalizes_to_zero_var(&i)) { + secp256k1_gej_double_var(r, a, rzr); + } else { + if (rzr != NULL) { + secp256k1_fe_set_int(rzr, 0); + } + r->infinity = 1; + } + return; + } + secp256k1_fe_sqr(&i2, &i); + secp256k1_fe_sqr(&h2, &h); + secp256k1_fe_mul(&h3, &h, &h2); + secp256k1_fe_mul(&h, &h, &b->z); + if (rzr != NULL) { + *rzr = h; + } + secp256k1_fe_mul(&r->z, &a->z, &h); + secp256k1_fe_mul(&t, &u1, &h2); + r->x = t; secp256k1_fe_mul_int(&r->x, 2); secp256k1_fe_add(&r->x, &h3); secp256k1_fe_negate(&r->x, &r->x, 3); secp256k1_fe_add(&r->x, &i2); + secp256k1_fe_negate(&r->y, &r->x, 5); secp256k1_fe_add(&r->y, &t); secp256k1_fe_mul(&r->y, &r->y, &i); + secp256k1_fe_mul(&h3, &h3, &s1); secp256k1_fe_negate(&h3, &h3, 1); + secp256k1_fe_add(&r->y, &h3); +} + +static void secp256k1_gej_add_ge_var(secp256k1_gej *r, const secp256k1_gej *a, const secp256k1_ge *b, secp256k1_fe *rzr) { + /* 8 mul, 3 sqr, 4 normalize, 12 mul_int/add/negate */ + secp256k1_fe z12, u1, u2, s1, s2, h, i, i2, h2, h3, t; + if (a->infinity) { + VERIFY_CHECK(rzr == NULL); + secp256k1_gej_set_ge(r, b); + return; + } + if (b->infinity) { + if (rzr != NULL) { + secp256k1_fe_set_int(rzr, 1); + } + *r = *a; + return; + } + r->infinity = 0; + + secp256k1_fe_sqr(&z12, &a->z); + u1 = a->x; secp256k1_fe_normalize_weak(&u1); + secp256k1_fe_mul(&u2, &b->x, &z12); + s1 = a->y; secp256k1_fe_normalize_weak(&s1); + secp256k1_fe_mul(&s2, &b->y, &z12); secp256k1_fe_mul(&s2, &s2, &a->z); + secp256k1_fe_negate(&h, &u1, 1); secp256k1_fe_add(&h, &u2); + secp256k1_fe_negate(&i, &s1, 1); secp256k1_fe_add(&i, &s2); + if (secp256k1_fe_normalizes_to_zero_var(&h)) { + if (secp256k1_fe_normalizes_to_zero_var(&i)) { + secp256k1_gej_double_var(r, a, rzr); + } else { + if (rzr != NULL) { + secp256k1_fe_set_int(rzr, 0); + } + r->infinity = 1; + } + return; + } + secp256k1_fe_sqr(&i2, &i); + secp256k1_fe_sqr(&h2, &h); + secp256k1_fe_mul(&h3, &h, &h2); + if (rzr != NULL) { + *rzr = h; + } + secp256k1_fe_mul(&r->z, &a->z, &h); + secp256k1_fe_mul(&t, &u1, &h2); + r->x = t; secp256k1_fe_mul_int(&r->x, 2); secp256k1_fe_add(&r->x, &h3); secp256k1_fe_negate(&r->x, &r->x, 3); secp256k1_fe_add(&r->x, &i2); + secp256k1_fe_negate(&r->y, &r->x, 5); secp256k1_fe_add(&r->y, &t); secp256k1_fe_mul(&r->y, &r->y, &i); + secp256k1_fe_mul(&h3, &h3, &s1); secp256k1_fe_negate(&h3, &h3, 1); + secp256k1_fe_add(&r->y, &h3); +} + +static void secp256k1_gej_add_zinv_var(secp256k1_gej *r, const secp256k1_gej *a, const secp256k1_ge *b, const secp256k1_fe *bzinv) { + /* 9 mul, 3 sqr, 4 normalize, 12 mul_int/add/negate */ + secp256k1_fe az, z12, u1, u2, s1, s2, h, i, i2, h2, h3, t; + + if (b->infinity) { + *r = *a; + return; + } + if (a->infinity) { + secp256k1_fe bzinv2, bzinv3; + r->infinity = b->infinity; + secp256k1_fe_sqr(&bzinv2, bzinv); + secp256k1_fe_mul(&bzinv3, &bzinv2, bzinv); + secp256k1_fe_mul(&r->x, &b->x, &bzinv2); + secp256k1_fe_mul(&r->y, &b->y, &bzinv3); + secp256k1_fe_set_int(&r->z, 1); + return; + } + r->infinity = 0; + + /** We need to calculate (rx,ry,rz) = (ax,ay,az) + (bx,by,1/bzinv). Due to + * secp256k1's isomorphism we can multiply the Z coordinates on both sides + * by bzinv, and get: (rx,ry,rz*bzinv) = (ax,ay,az*bzinv) + (bx,by,1). + * This means that (rx,ry,rz) can be calculated as + * (ax,ay,az*bzinv) + (bx,by,1), when not applying the bzinv factor to rz. + * The variable az below holds the modified Z coordinate for a, which is used + * for the computation of rx and ry, but not for rz. + */ + secp256k1_fe_mul(&az, &a->z, bzinv); + + secp256k1_fe_sqr(&z12, &az); + u1 = a->x; secp256k1_fe_normalize_weak(&u1); + secp256k1_fe_mul(&u2, &b->x, &z12); + s1 = a->y; secp256k1_fe_normalize_weak(&s1); + secp256k1_fe_mul(&s2, &b->y, &z12); secp256k1_fe_mul(&s2, &s2, &az); + secp256k1_fe_negate(&h, &u1, 1); secp256k1_fe_add(&h, &u2); + secp256k1_fe_negate(&i, &s1, 1); secp256k1_fe_add(&i, &s2); + if (secp256k1_fe_normalizes_to_zero_var(&h)) { + if (secp256k1_fe_normalizes_to_zero_var(&i)) { + secp256k1_gej_double_var(r, a, NULL); + } else { + r->infinity = 1; + } + return; + } + secp256k1_fe_sqr(&i2, &i); + secp256k1_fe_sqr(&h2, &h); + secp256k1_fe_mul(&h3, &h, &h2); + r->z = a->z; secp256k1_fe_mul(&r->z, &r->z, &h); + secp256k1_fe_mul(&t, &u1, &h2); + r->x = t; secp256k1_fe_mul_int(&r->x, 2); secp256k1_fe_add(&r->x, &h3); secp256k1_fe_negate(&r->x, &r->x, 3); secp256k1_fe_add(&r->x, &i2); + secp256k1_fe_negate(&r->y, &r->x, 5); secp256k1_fe_add(&r->y, &t); secp256k1_fe_mul(&r->y, &r->y, &i); + secp256k1_fe_mul(&h3, &h3, &s1); secp256k1_fe_negate(&h3, &h3, 1); + secp256k1_fe_add(&r->y, &h3); +} + + +static void secp256k1_gej_add_ge(secp256k1_gej *r, const secp256k1_gej *a, const secp256k1_ge *b) { + /* Operations: 7 mul, 5 sqr, 4 normalize, 21 mul_int/add/negate/cmov */ + static const secp256k1_fe fe_1 = SECP256K1_FE_CONST(0, 0, 0, 0, 0, 0, 0, 1); + secp256k1_fe zz, u1, u2, s1, s2, t, tt, m, n, q, rr; + secp256k1_fe m_alt, rr_alt; + int infinity, degenerate; + VERIFY_CHECK(!b->infinity); + VERIFY_CHECK(a->infinity == 0 || a->infinity == 1); + + /** In: + * Eric Brier and Marc Joye, Weierstrass Elliptic Curves and Side-Channel Attacks. + * In D. Naccache and P. Paillier, Eds., Public Key Cryptography, vol. 2274 of Lecture Notes in Computer Science, pages 335-345. Springer-Verlag, 2002. + * we find as solution for a unified addition/doubling formula: + * lambda = ((x1 + x2)^2 - x1 * x2 + a) / (y1 + y2), with a = 0 for secp256k1's curve equation. + * x3 = lambda^2 - (x1 + x2) + * 2*y3 = lambda * (x1 + x2 - 2 * x3) - (y1 + y2). + * + * Substituting x_i = Xi / Zi^2 and yi = Yi / Zi^3, for i=1,2,3, gives: + * U1 = X1*Z2^2, U2 = X2*Z1^2 + * S1 = Y1*Z2^3, S2 = Y2*Z1^3 + * Z = Z1*Z2 + * T = U1+U2 + * M = S1+S2 + * Q = T*M^2 + * R = T^2-U1*U2 + * X3 = 4*(R^2-Q) + * Y3 = 4*(R*(3*Q-2*R^2)-M^4) + * Z3 = 2*M*Z + * (Note that the paper uses xi = Xi / Zi and yi = Yi / Zi instead.) + * + * This formula has the benefit of being the same for both addition + * of distinct points and doubling. However, it breaks down in the + * case that either point is infinity, or that y1 = -y2. We handle + * these cases in the following ways: + * + * - If b is infinity we simply bail by means of a VERIFY_CHECK. + * + * - If a is infinity, we detect this, and at the end of the + * computation replace the result (which will be meaningless, + * but we compute to be constant-time) with b.x : b.y : 1. + * + * - If a = -b, we have y1 = -y2, which is a degenerate case. + * But here the answer is infinity, so we simply set the + * infinity flag of the result, overriding the computed values + * without even needing to cmov. + * + * - If y1 = -y2 but x1 != x2, which does occur thanks to certain + * properties of our curve (specifically, 1 has nontrivial cube + * roots in our field, and the curve equation has no x coefficient) + * then the answer is not infinity but also not given by the above + * equation. In this case, we cmov in place an alternate expression + * for lambda. Specifically (y1 - y2)/(x1 - x2). Where both these + * expressions for lambda are defined, they are equal, and can be + * obtained from each other by multiplication by (y1 + y2)/(y1 + y2) + * then substitution of x^3 + 7 for y^2 (using the curve equation). + * For all pairs of nonzero points (a, b) at least one is defined, + * so this covers everything. + */ + + secp256k1_fe_sqr(&zz, &a->z); /* z = Z1^2 */ + u1 = a->x; secp256k1_fe_normalize_weak(&u1); /* u1 = U1 = X1*Z2^2 (1) */ + secp256k1_fe_mul(&u2, &b->x, &zz); /* u2 = U2 = X2*Z1^2 (1) */ + s1 = a->y; secp256k1_fe_normalize_weak(&s1); /* s1 = S1 = Y1*Z2^3 (1) */ + secp256k1_fe_mul(&s2, &b->y, &zz); /* s2 = Y2*Z1^2 (1) */ + secp256k1_fe_mul(&s2, &s2, &a->z); /* s2 = S2 = Y2*Z1^3 (1) */ + t = u1; secp256k1_fe_add(&t, &u2); /* t = T = U1+U2 (2) */ + m = s1; secp256k1_fe_add(&m, &s2); /* m = M = S1+S2 (2) */ + secp256k1_fe_sqr(&rr, &t); /* rr = T^2 (1) */ + secp256k1_fe_negate(&m_alt, &u2, 1); /* Malt = -X2*Z1^2 */ + secp256k1_fe_mul(&tt, &u1, &m_alt); /* tt = -U1*U2 (2) */ + secp256k1_fe_add(&rr, &tt); /* rr = R = T^2-U1*U2 (3) */ + /** If lambda = R/M = 0/0 we have a problem (except in the "trivial" + * case that Z = z1z2 = 0, and this is special-cased later on). */ + degenerate = secp256k1_fe_normalizes_to_zero(&m) & + secp256k1_fe_normalizes_to_zero(&rr); + /* This only occurs when y1 == -y2 and x1^3 == x2^3, but x1 != x2. + * This means either x1 == beta*x2 or beta*x1 == x2, where beta is + * a nontrivial cube root of one. In either case, an alternate + * non-indeterminate expression for lambda is (y1 - y2)/(x1 - x2), + * so we set R/M equal to this. */ + rr_alt = s1; + secp256k1_fe_mul_int(&rr_alt, 2); /* rr = Y1*Z2^3 - Y2*Z1^3 (2) */ + secp256k1_fe_add(&m_alt, &u1); /* Malt = X1*Z2^2 - X2*Z1^2 */ + + secp256k1_fe_cmov(&rr_alt, &rr, !degenerate); + secp256k1_fe_cmov(&m_alt, &m, !degenerate); + /* Now Ralt / Malt = lambda and is guaranteed not to be 0/0. + * From here on out Ralt and Malt represent the numerator + * and denominator of lambda; R and M represent the explicit + * expressions x1^2 + x2^2 + x1x2 and y1 + y2. */ + secp256k1_fe_sqr(&n, &m_alt); /* n = Malt^2 (1) */ + secp256k1_fe_mul(&q, &n, &t); /* q = Q = T*Malt^2 (1) */ + /* These two lines use the observation that either M == Malt or M == 0, + * so M^3 * Malt is either Malt^4 (which is computed by squaring), or + * zero (which is "computed" by cmov). So the cost is one squaring + * versus two multiplications. */ + secp256k1_fe_sqr(&n, &n); + secp256k1_fe_cmov(&n, &m, degenerate); /* n = M^3 * Malt (2) */ + secp256k1_fe_sqr(&t, &rr_alt); /* t = Ralt^2 (1) */ + secp256k1_fe_mul(&r->z, &a->z, &m_alt); /* r->z = Malt*Z (1) */ + infinity = secp256k1_fe_normalizes_to_zero(&r->z) * (1 - a->infinity); + secp256k1_fe_mul_int(&r->z, 2); /* r->z = Z3 = 2*Malt*Z (2) */ + secp256k1_fe_negate(&q, &q, 1); /* q = -Q (2) */ + secp256k1_fe_add(&t, &q); /* t = Ralt^2-Q (3) */ + secp256k1_fe_normalize_weak(&t); + r->x = t; /* r->x = Ralt^2-Q (1) */ + secp256k1_fe_mul_int(&t, 2); /* t = 2*x3 (2) */ + secp256k1_fe_add(&t, &q); /* t = 2*x3 - Q: (4) */ + secp256k1_fe_mul(&t, &t, &rr_alt); /* t = Ralt*(2*x3 - Q) (1) */ + secp256k1_fe_add(&t, &n); /* t = Ralt*(2*x3 - Q) + M^3*Malt (3) */ + secp256k1_fe_negate(&r->y, &t, 3); /* r->y = Ralt*(Q - 2x3) - M^3*Malt (4) */ + secp256k1_fe_normalize_weak(&r->y); + secp256k1_fe_mul_int(&r->x, 4); /* r->x = X3 = 4*(Ralt^2-Q) */ + secp256k1_fe_mul_int(&r->y, 4); /* r->y = Y3 = 4*Ralt*(Q - 2x3) - 4*M^3*Malt (4) */ + + /** In case a->infinity == 1, replace r with (b->x, b->y, 1). */ + secp256k1_fe_cmov(&r->x, &b->x, a->infinity); + secp256k1_fe_cmov(&r->y, &b->y, a->infinity); + secp256k1_fe_cmov(&r->z, &fe_1, a->infinity); + r->infinity = infinity; +} + +static void secp256k1_gej_rescale(secp256k1_gej *r, const secp256k1_fe *s) { + /* Operations: 4 mul, 1 sqr */ + secp256k1_fe zz; + VERIFY_CHECK(!secp256k1_fe_is_zero(s)); + secp256k1_fe_sqr(&zz, s); + secp256k1_fe_mul(&r->x, &r->x, &zz); /* r->x *= s^2 */ + secp256k1_fe_mul(&r->y, &r->y, &zz); + secp256k1_fe_mul(&r->y, &r->y, s); /* r->y *= s^3 */ + secp256k1_fe_mul(&r->z, &r->z, s); /* r->z *= s */ +} + +static void secp256k1_ge_to_storage(secp256k1_ge_storage *r, const secp256k1_ge *a) { + secp256k1_fe x, y; + VERIFY_CHECK(!a->infinity); + x = a->x; + secp256k1_fe_normalize(&x); + y = a->y; + secp256k1_fe_normalize(&y); + secp256k1_fe_to_storage(&r->x, &x); + secp256k1_fe_to_storage(&r->y, &y); +} + +static void secp256k1_ge_from_storage(secp256k1_ge *r, const secp256k1_ge_storage *a) { + secp256k1_fe_from_storage(&r->x, &a->x); + secp256k1_fe_from_storage(&r->y, &a->y); + r->infinity = 0; +} + +static SECP256K1_INLINE void secp256k1_ge_storage_cmov(secp256k1_ge_storage *r, const secp256k1_ge_storage *a, int flag) { + secp256k1_fe_storage_cmov(&r->x, &a->x, flag); + secp256k1_fe_storage_cmov(&r->y, &a->y, flag); +} + +#ifdef USE_ENDOMORPHISM +static void secp256k1_ge_mul_lambda(secp256k1_ge *r, const secp256k1_ge *a) { + static const secp256k1_fe beta = SECP256K1_FE_CONST( + 0x7ae96a2bul, 0x657c0710ul, 0x6e64479eul, 0xac3434e9ul, + 0x9cf04975ul, 0x12f58995ul, 0xc1396c28ul, 0x719501eeul + ); + *r = *a; + secp256k1_fe_mul(&r->x, &r->x, &beta); +} +#endif + +static int secp256k1_gej_has_quad_y_var(const secp256k1_gej *a) { + secp256k1_fe yz; + + if (a->infinity) { + return 0; + } + + /* We rely on the fact that the Jacobi symbol of 1 / a->z^3 is the same as + * that of a->z. Thus a->y / a->z^3 is a quadratic residue iff a->y * a->z + is */ + secp256k1_fe_mul(&yz, &a->y, &a->z); + return secp256k1_fe_is_quad_var(&yz); +} + +#endif /* SECP256K1_GROUP_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/hash.h b/node_modules/secp256k1/src/secp256k1/src/hash.h new file mode 100644 index 0000000..de26e4b --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/hash.h @@ -0,0 +1,41 @@ +/********************************************************************** + * Copyright (c) 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_HASH_H +#define SECP256K1_HASH_H + +#include +#include + +typedef struct { + uint32_t s[8]; + uint32_t buf[16]; /* In big endian */ + size_t bytes; +} secp256k1_sha256; + +static void secp256k1_sha256_initialize(secp256k1_sha256 *hash); +static void secp256k1_sha256_write(secp256k1_sha256 *hash, const unsigned char *data, size_t size); +static void secp256k1_sha256_finalize(secp256k1_sha256 *hash, unsigned char *out32); + +typedef struct { + secp256k1_sha256 inner, outer; +} secp256k1_hmac_sha256; + +static void secp256k1_hmac_sha256_initialize(secp256k1_hmac_sha256 *hash, const unsigned char *key, size_t size); +static void secp256k1_hmac_sha256_write(secp256k1_hmac_sha256 *hash, const unsigned char *data, size_t size); +static void secp256k1_hmac_sha256_finalize(secp256k1_hmac_sha256 *hash, unsigned char *out32); + +typedef struct { + unsigned char v[32]; + unsigned char k[32]; + int retry; +} secp256k1_rfc6979_hmac_sha256; + +static void secp256k1_rfc6979_hmac_sha256_initialize(secp256k1_rfc6979_hmac_sha256 *rng, const unsigned char *key, size_t keylen); +static void secp256k1_rfc6979_hmac_sha256_generate(secp256k1_rfc6979_hmac_sha256 *rng, unsigned char *out, size_t outlen); +static void secp256k1_rfc6979_hmac_sha256_finalize(secp256k1_rfc6979_hmac_sha256 *rng); + +#endif /* SECP256K1_HASH_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/hash_impl.h b/node_modules/secp256k1/src/secp256k1/src/hash_impl.h new file mode 100644 index 0000000..782f972 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/hash_impl.h @@ -0,0 +1,283 @@ +/********************************************************************** + * Copyright (c) 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_HASH_IMPL_H +#define SECP256K1_HASH_IMPL_H + +#include "hash.h" + +#include +#include +#include + +#define Ch(x,y,z) ((z) ^ ((x) & ((y) ^ (z)))) +#define Maj(x,y,z) (((x) & (y)) | ((z) & ((x) | (y)))) +#define Sigma0(x) (((x) >> 2 | (x) << 30) ^ ((x) >> 13 | (x) << 19) ^ ((x) >> 22 | (x) << 10)) +#define Sigma1(x) (((x) >> 6 | (x) << 26) ^ ((x) >> 11 | (x) << 21) ^ ((x) >> 25 | (x) << 7)) +#define sigma0(x) (((x) >> 7 | (x) << 25) ^ ((x) >> 18 | (x) << 14) ^ ((x) >> 3)) +#define sigma1(x) (((x) >> 17 | (x) << 15) ^ ((x) >> 19 | (x) << 13) ^ ((x) >> 10)) + +#define Round(a,b,c,d,e,f,g,h,k,w) do { \ + uint32_t t1 = (h) + Sigma1(e) + Ch((e), (f), (g)) + (k) + (w); \ + uint32_t t2 = Sigma0(a) + Maj((a), (b), (c)); \ + (d) += t1; \ + (h) = t1 + t2; \ +} while(0) + +#ifdef WORDS_BIGENDIAN +#define BE32(x) (x) +#else +#define BE32(p) ((((p) & 0xFF) << 24) | (((p) & 0xFF00) << 8) | (((p) & 0xFF0000) >> 8) | (((p) & 0xFF000000) >> 24)) +#endif + +static void secp256k1_sha256_initialize(secp256k1_sha256 *hash) { + hash->s[0] = 0x6a09e667ul; + hash->s[1] = 0xbb67ae85ul; + hash->s[2] = 0x3c6ef372ul; + hash->s[3] = 0xa54ff53aul; + hash->s[4] = 0x510e527ful; + hash->s[5] = 0x9b05688cul; + hash->s[6] = 0x1f83d9abul; + hash->s[7] = 0x5be0cd19ul; + hash->bytes = 0; +} + +/** Perform one SHA-256 transformation, processing 16 big endian 32-bit words. */ +static void secp256k1_sha256_transform(uint32_t* s, const uint32_t* chunk) { + uint32_t a = s[0], b = s[1], c = s[2], d = s[3], e = s[4], f = s[5], g = s[6], h = s[7]; + uint32_t w0, w1, w2, w3, w4, w5, w6, w7, w8, w9, w10, w11, w12, w13, w14, w15; + + Round(a, b, c, d, e, f, g, h, 0x428a2f98, w0 = BE32(chunk[0])); + Round(h, a, b, c, d, e, f, g, 0x71374491, w1 = BE32(chunk[1])); + Round(g, h, a, b, c, d, e, f, 0xb5c0fbcf, w2 = BE32(chunk[2])); + Round(f, g, h, a, b, c, d, e, 0xe9b5dba5, w3 = BE32(chunk[3])); + Round(e, f, g, h, a, b, c, d, 0x3956c25b, w4 = BE32(chunk[4])); + Round(d, e, f, g, h, a, b, c, 0x59f111f1, w5 = BE32(chunk[5])); + Round(c, d, e, f, g, h, a, b, 0x923f82a4, w6 = BE32(chunk[6])); + Round(b, c, d, e, f, g, h, a, 0xab1c5ed5, w7 = BE32(chunk[7])); + Round(a, b, c, d, e, f, g, h, 0xd807aa98, w8 = BE32(chunk[8])); + Round(h, a, b, c, d, e, f, g, 0x12835b01, w9 = BE32(chunk[9])); + Round(g, h, a, b, c, d, e, f, 0x243185be, w10 = BE32(chunk[10])); + Round(f, g, h, a, b, c, d, e, 0x550c7dc3, w11 = BE32(chunk[11])); + Round(e, f, g, h, a, b, c, d, 0x72be5d74, w12 = BE32(chunk[12])); + Round(d, e, f, g, h, a, b, c, 0x80deb1fe, w13 = BE32(chunk[13])); + Round(c, d, e, f, g, h, a, b, 0x9bdc06a7, w14 = BE32(chunk[14])); + Round(b, c, d, e, f, g, h, a, 0xc19bf174, w15 = BE32(chunk[15])); + + Round(a, b, c, d, e, f, g, h, 0xe49b69c1, w0 += sigma1(w14) + w9 + sigma0(w1)); + Round(h, a, b, c, d, e, f, g, 0xefbe4786, w1 += sigma1(w15) + w10 + sigma0(w2)); + Round(g, h, a, b, c, d, e, f, 0x0fc19dc6, w2 += sigma1(w0) + w11 + sigma0(w3)); + Round(f, g, h, a, b, c, d, e, 0x240ca1cc, w3 += sigma1(w1) + w12 + sigma0(w4)); + Round(e, f, g, h, a, b, c, d, 0x2de92c6f, w4 += sigma1(w2) + w13 + sigma0(w5)); + Round(d, e, f, g, h, a, b, c, 0x4a7484aa, w5 += sigma1(w3) + w14 + sigma0(w6)); + Round(c, d, e, f, g, h, a, b, 0x5cb0a9dc, w6 += sigma1(w4) + w15 + sigma0(w7)); + Round(b, c, d, e, f, g, h, a, 0x76f988da, w7 += sigma1(w5) + w0 + sigma0(w8)); + Round(a, b, c, d, e, f, g, h, 0x983e5152, w8 += sigma1(w6) + w1 + sigma0(w9)); + Round(h, a, b, c, d, e, f, g, 0xa831c66d, w9 += sigma1(w7) + w2 + sigma0(w10)); + Round(g, h, a, b, c, d, e, f, 0xb00327c8, w10 += sigma1(w8) + w3 + sigma0(w11)); + Round(f, g, h, a, b, c, d, e, 0xbf597fc7, w11 += sigma1(w9) + w4 + sigma0(w12)); + Round(e, f, g, h, a, b, c, d, 0xc6e00bf3, w12 += sigma1(w10) + w5 + sigma0(w13)); + Round(d, e, f, g, h, a, b, c, 0xd5a79147, w13 += sigma1(w11) + w6 + sigma0(w14)); + Round(c, d, e, f, g, h, a, b, 0x06ca6351, w14 += sigma1(w12) + w7 + sigma0(w15)); + Round(b, c, d, e, f, g, h, a, 0x14292967, w15 += sigma1(w13) + w8 + sigma0(w0)); + + Round(a, b, c, d, e, f, g, h, 0x27b70a85, w0 += sigma1(w14) + w9 + sigma0(w1)); + Round(h, a, b, c, d, e, f, g, 0x2e1b2138, w1 += sigma1(w15) + w10 + sigma0(w2)); + Round(g, h, a, b, c, d, e, f, 0x4d2c6dfc, w2 += sigma1(w0) + w11 + sigma0(w3)); + Round(f, g, h, a, b, c, d, e, 0x53380d13, w3 += sigma1(w1) + w12 + sigma0(w4)); + Round(e, f, g, h, a, b, c, d, 0x650a7354, w4 += sigma1(w2) + w13 + sigma0(w5)); + Round(d, e, f, g, h, a, b, c, 0x766a0abb, w5 += sigma1(w3) + w14 + sigma0(w6)); + Round(c, d, e, f, g, h, a, b, 0x81c2c92e, w6 += sigma1(w4) + w15 + sigma0(w7)); + Round(b, c, d, e, f, g, h, a, 0x92722c85, w7 += sigma1(w5) + w0 + sigma0(w8)); + Round(a, b, c, d, e, f, g, h, 0xa2bfe8a1, w8 += sigma1(w6) + w1 + sigma0(w9)); + Round(h, a, b, c, d, e, f, g, 0xa81a664b, w9 += sigma1(w7) + w2 + sigma0(w10)); + Round(g, h, a, b, c, d, e, f, 0xc24b8b70, w10 += sigma1(w8) + w3 + sigma0(w11)); + Round(f, g, h, a, b, c, d, e, 0xc76c51a3, w11 += sigma1(w9) + w4 + sigma0(w12)); + Round(e, f, g, h, a, b, c, d, 0xd192e819, w12 += sigma1(w10) + w5 + sigma0(w13)); + Round(d, e, f, g, h, a, b, c, 0xd6990624, w13 += sigma1(w11) + w6 + sigma0(w14)); + Round(c, d, e, f, g, h, a, b, 0xf40e3585, w14 += sigma1(w12) + w7 + sigma0(w15)); + Round(b, c, d, e, f, g, h, a, 0x106aa070, w15 += sigma1(w13) + w8 + sigma0(w0)); + + Round(a, b, c, d, e, f, g, h, 0x19a4c116, w0 += sigma1(w14) + w9 + sigma0(w1)); + Round(h, a, b, c, d, e, f, g, 0x1e376c08, w1 += sigma1(w15) + w10 + sigma0(w2)); + Round(g, h, a, b, c, d, e, f, 0x2748774c, w2 += sigma1(w0) + w11 + sigma0(w3)); + Round(f, g, h, a, b, c, d, e, 0x34b0bcb5, w3 += sigma1(w1) + w12 + sigma0(w4)); + Round(e, f, g, h, a, b, c, d, 0x391c0cb3, w4 += sigma1(w2) + w13 + sigma0(w5)); + Round(d, e, f, g, h, a, b, c, 0x4ed8aa4a, w5 += sigma1(w3) + w14 + sigma0(w6)); + Round(c, d, e, f, g, h, a, b, 0x5b9cca4f, w6 += sigma1(w4) + w15 + sigma0(w7)); + Round(b, c, d, e, f, g, h, a, 0x682e6ff3, w7 += sigma1(w5) + w0 + sigma0(w8)); + Round(a, b, c, d, e, f, g, h, 0x748f82ee, w8 += sigma1(w6) + w1 + sigma0(w9)); + Round(h, a, b, c, d, e, f, g, 0x78a5636f, w9 += sigma1(w7) + w2 + sigma0(w10)); + Round(g, h, a, b, c, d, e, f, 0x84c87814, w10 += sigma1(w8) + w3 + sigma0(w11)); + Round(f, g, h, a, b, c, d, e, 0x8cc70208, w11 += sigma1(w9) + w4 + sigma0(w12)); + Round(e, f, g, h, a, b, c, d, 0x90befffa, w12 += sigma1(w10) + w5 + sigma0(w13)); + Round(d, e, f, g, h, a, b, c, 0xa4506ceb, w13 += sigma1(w11) + w6 + sigma0(w14)); + Round(c, d, e, f, g, h, a, b, 0xbef9a3f7, w14 + sigma1(w12) + w7 + sigma0(w15)); + Round(b, c, d, e, f, g, h, a, 0xc67178f2, w15 + sigma1(w13) + w8 + sigma0(w0)); + + s[0] += a; + s[1] += b; + s[2] += c; + s[3] += d; + s[4] += e; + s[5] += f; + s[6] += g; + s[7] += h; +} + +static void secp256k1_sha256_write(secp256k1_sha256 *hash, const unsigned char *data, size_t len) { + size_t bufsize = hash->bytes & 0x3F; + hash->bytes += len; + VERIFY_CHECK(hash->bytes >= len); + while (len >= 64 - bufsize) { + /* Fill the buffer, and process it. */ + size_t chunk_len = 64 - bufsize; + memcpy(((unsigned char*)hash->buf) + bufsize, data, chunk_len); + data += chunk_len; + len -= chunk_len; + secp256k1_sha256_transform(hash->s, hash->buf); + bufsize = 0; + } + if (len) { + /* Fill the buffer with what remains. */ + memcpy(((unsigned char*)hash->buf) + bufsize, data, len); + } +} + +static void secp256k1_sha256_finalize(secp256k1_sha256 *hash, unsigned char *out32) { + static const unsigned char pad[64] = {0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; + uint32_t sizedesc[2]; + uint32_t out[8]; + int i = 0; + sizedesc[0] = BE32(hash->bytes >> 29); + sizedesc[1] = BE32(hash->bytes << 3); + secp256k1_sha256_write(hash, pad, 1 + ((119 - (hash->bytes % 64)) % 64)); + secp256k1_sha256_write(hash, (const unsigned char*)sizedesc, 8); + for (i = 0; i < 8; i++) { + out[i] = BE32(hash->s[i]); + hash->s[i] = 0; + } + memcpy(out32, (const unsigned char*)out, 32); +} + +static void secp256k1_hmac_sha256_initialize(secp256k1_hmac_sha256 *hash, const unsigned char *key, size_t keylen) { + size_t n; + unsigned char rkey[64]; + if (keylen <= sizeof(rkey)) { + memcpy(rkey, key, keylen); + memset(rkey + keylen, 0, sizeof(rkey) - keylen); + } else { + secp256k1_sha256 sha256; + secp256k1_sha256_initialize(&sha256); + secp256k1_sha256_write(&sha256, key, keylen); + secp256k1_sha256_finalize(&sha256, rkey); + memset(rkey + 32, 0, 32); + } + + secp256k1_sha256_initialize(&hash->outer); + for (n = 0; n < sizeof(rkey); n++) { + rkey[n] ^= 0x5c; + } + secp256k1_sha256_write(&hash->outer, rkey, sizeof(rkey)); + + secp256k1_sha256_initialize(&hash->inner); + for (n = 0; n < sizeof(rkey); n++) { + rkey[n] ^= 0x5c ^ 0x36; + } + secp256k1_sha256_write(&hash->inner, rkey, sizeof(rkey)); + memset(rkey, 0, sizeof(rkey)); +} + +static void secp256k1_hmac_sha256_write(secp256k1_hmac_sha256 *hash, const unsigned char *data, size_t size) { + secp256k1_sha256_write(&hash->inner, data, size); +} + +static void secp256k1_hmac_sha256_finalize(secp256k1_hmac_sha256 *hash, unsigned char *out32) { + unsigned char temp[32]; + secp256k1_sha256_finalize(&hash->inner, temp); + secp256k1_sha256_write(&hash->outer, temp, 32); + memset(temp, 0, 32); + secp256k1_sha256_finalize(&hash->outer, out32); +} + + +static void secp256k1_rfc6979_hmac_sha256_initialize(secp256k1_rfc6979_hmac_sha256 *rng, const unsigned char *key, size_t keylen) { + secp256k1_hmac_sha256 hmac; + static const unsigned char zero[1] = {0x00}; + static const unsigned char one[1] = {0x01}; + + memset(rng->v, 0x01, 32); /* RFC6979 3.2.b. */ + memset(rng->k, 0x00, 32); /* RFC6979 3.2.c. */ + + /* RFC6979 3.2.d. */ + secp256k1_hmac_sha256_initialize(&hmac, rng->k, 32); + secp256k1_hmac_sha256_write(&hmac, rng->v, 32); + secp256k1_hmac_sha256_write(&hmac, zero, 1); + secp256k1_hmac_sha256_write(&hmac, key, keylen); + secp256k1_hmac_sha256_finalize(&hmac, rng->k); + secp256k1_hmac_sha256_initialize(&hmac, rng->k, 32); + secp256k1_hmac_sha256_write(&hmac, rng->v, 32); + secp256k1_hmac_sha256_finalize(&hmac, rng->v); + + /* RFC6979 3.2.f. */ + secp256k1_hmac_sha256_initialize(&hmac, rng->k, 32); + secp256k1_hmac_sha256_write(&hmac, rng->v, 32); + secp256k1_hmac_sha256_write(&hmac, one, 1); + secp256k1_hmac_sha256_write(&hmac, key, keylen); + secp256k1_hmac_sha256_finalize(&hmac, rng->k); + secp256k1_hmac_sha256_initialize(&hmac, rng->k, 32); + secp256k1_hmac_sha256_write(&hmac, rng->v, 32); + secp256k1_hmac_sha256_finalize(&hmac, rng->v); + rng->retry = 0; +} + +static void secp256k1_rfc6979_hmac_sha256_generate(secp256k1_rfc6979_hmac_sha256 *rng, unsigned char *out, size_t outlen) { + /* RFC6979 3.2.h. */ + static const unsigned char zero[1] = {0x00}; + if (rng->retry) { + secp256k1_hmac_sha256 hmac; + secp256k1_hmac_sha256_initialize(&hmac, rng->k, 32); + secp256k1_hmac_sha256_write(&hmac, rng->v, 32); + secp256k1_hmac_sha256_write(&hmac, zero, 1); + secp256k1_hmac_sha256_finalize(&hmac, rng->k); + secp256k1_hmac_sha256_initialize(&hmac, rng->k, 32); + secp256k1_hmac_sha256_write(&hmac, rng->v, 32); + secp256k1_hmac_sha256_finalize(&hmac, rng->v); + } + + while (outlen > 0) { + secp256k1_hmac_sha256 hmac; + int now = outlen; + secp256k1_hmac_sha256_initialize(&hmac, rng->k, 32); + secp256k1_hmac_sha256_write(&hmac, rng->v, 32); + secp256k1_hmac_sha256_finalize(&hmac, rng->v); + if (now > 32) { + now = 32; + } + memcpy(out, rng->v, now); + out += now; + outlen -= now; + } + + rng->retry = 1; +} + +static void secp256k1_rfc6979_hmac_sha256_finalize(secp256k1_rfc6979_hmac_sha256 *rng) { + memset(rng->k, 0, 32); + memset(rng->v, 0, 32); + rng->retry = 0; +} + +#undef BE32 +#undef Round +#undef sigma1 +#undef sigma0 +#undef Sigma1 +#undef Sigma0 +#undef Maj +#undef Ch + +#endif /* SECP256K1_HASH_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/java/org/bitcoin/NativeSecp256k1.java b/node_modules/secp256k1/src/secp256k1/src/java/org/bitcoin/NativeSecp256k1.java new file mode 100644 index 0000000..1c67802 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/java/org/bitcoin/NativeSecp256k1.java @@ -0,0 +1,446 @@ +/* + * Copyright 2013 Google Inc. + * Copyright 2014-2016 the libsecp256k1 contributors + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.bitcoin; + +import java.nio.ByteBuffer; +import java.nio.ByteOrder; + +import java.math.BigInteger; +import com.google.common.base.Preconditions; +import java.util.concurrent.locks.Lock; +import java.util.concurrent.locks.ReentrantReadWriteLock; +import static org.bitcoin.NativeSecp256k1Util.*; + +/** + *

This class holds native methods to handle ECDSA verification.

+ * + *

You can find an example library that can be used for this at https://github.com/bitcoin/secp256k1

+ * + *

To build secp256k1 for use with bitcoinj, run + * `./configure --enable-jni --enable-experimental --enable-module-ecdh` + * and `make` then copy `.libs/libsecp256k1.so` to your system library path + * or point the JVM to the folder containing it with -Djava.library.path + *

+ */ +public class NativeSecp256k1 { + + private static final ReentrantReadWriteLock rwl = new ReentrantReadWriteLock(); + private static final Lock r = rwl.readLock(); + private static final Lock w = rwl.writeLock(); + private static ThreadLocal nativeECDSABuffer = new ThreadLocal(); + /** + * Verifies the given secp256k1 signature in native code. + * Calling when enabled == false is undefined (probably library not loaded) + * + * @param data The data which was signed, must be exactly 32 bytes + * @param signature The signature + * @param pub The public key which did the signing + */ + public static boolean verify(byte[] data, byte[] signature, byte[] pub) throws AssertFailException{ + Preconditions.checkArgument(data.length == 32 && signature.length <= 520 && pub.length <= 520); + + ByteBuffer byteBuff = nativeECDSABuffer.get(); + if (byteBuff == null || byteBuff.capacity() < 520) { + byteBuff = ByteBuffer.allocateDirect(520); + byteBuff.order(ByteOrder.nativeOrder()); + nativeECDSABuffer.set(byteBuff); + } + byteBuff.rewind(); + byteBuff.put(data); + byteBuff.put(signature); + byteBuff.put(pub); + + byte[][] retByteArray; + + r.lock(); + try { + return secp256k1_ecdsa_verify(byteBuff, Secp256k1Context.getContext(), signature.length, pub.length) == 1; + } finally { + r.unlock(); + } + } + + /** + * libsecp256k1 Create an ECDSA signature. + * + * @param data Message hash, 32 bytes + * @param key Secret key, 32 bytes + * + * Return values + * @param sig byte array of signature + */ + public static byte[] sign(byte[] data, byte[] sec) throws AssertFailException{ + Preconditions.checkArgument(data.length == 32 && sec.length <= 32); + + ByteBuffer byteBuff = nativeECDSABuffer.get(); + if (byteBuff == null || byteBuff.capacity() < 32 + 32) { + byteBuff = ByteBuffer.allocateDirect(32 + 32); + byteBuff.order(ByteOrder.nativeOrder()); + nativeECDSABuffer.set(byteBuff); + } + byteBuff.rewind(); + byteBuff.put(data); + byteBuff.put(sec); + + byte[][] retByteArray; + + r.lock(); + try { + retByteArray = secp256k1_ecdsa_sign(byteBuff, Secp256k1Context.getContext()); + } finally { + r.unlock(); + } + + byte[] sigArr = retByteArray[0]; + int sigLen = new BigInteger(new byte[] { retByteArray[1][0] }).intValue(); + int retVal = new BigInteger(new byte[] { retByteArray[1][1] }).intValue(); + + assertEquals(sigArr.length, sigLen, "Got bad signature length."); + + return retVal == 0 ? new byte[0] : sigArr; + } + + /** + * libsecp256k1 Seckey Verify - returns 1 if valid, 0 if invalid + * + * @param seckey ECDSA Secret key, 32 bytes + */ + public static boolean secKeyVerify(byte[] seckey) { + Preconditions.checkArgument(seckey.length == 32); + + ByteBuffer byteBuff = nativeECDSABuffer.get(); + if (byteBuff == null || byteBuff.capacity() < seckey.length) { + byteBuff = ByteBuffer.allocateDirect(seckey.length); + byteBuff.order(ByteOrder.nativeOrder()); + nativeECDSABuffer.set(byteBuff); + } + byteBuff.rewind(); + byteBuff.put(seckey); + + r.lock(); + try { + return secp256k1_ec_seckey_verify(byteBuff,Secp256k1Context.getContext()) == 1; + } finally { + r.unlock(); + } + } + + + /** + * libsecp256k1 Compute Pubkey - computes public key from secret key + * + * @param seckey ECDSA Secret key, 32 bytes + * + * Return values + * @param pubkey ECDSA Public key, 33 or 65 bytes + */ + //TODO add a 'compressed' arg + public static byte[] computePubkey(byte[] seckey) throws AssertFailException{ + Preconditions.checkArgument(seckey.length == 32); + + ByteBuffer byteBuff = nativeECDSABuffer.get(); + if (byteBuff == null || byteBuff.capacity() < seckey.length) { + byteBuff = ByteBuffer.allocateDirect(seckey.length); + byteBuff.order(ByteOrder.nativeOrder()); + nativeECDSABuffer.set(byteBuff); + } + byteBuff.rewind(); + byteBuff.put(seckey); + + byte[][] retByteArray; + + r.lock(); + try { + retByteArray = secp256k1_ec_pubkey_create(byteBuff, Secp256k1Context.getContext()); + } finally { + r.unlock(); + } + + byte[] pubArr = retByteArray[0]; + int pubLen = new BigInteger(new byte[] { retByteArray[1][0] }).intValue(); + int retVal = new BigInteger(new byte[] { retByteArray[1][1] }).intValue(); + + assertEquals(pubArr.length, pubLen, "Got bad pubkey length."); + + return retVal == 0 ? new byte[0]: pubArr; + } + + /** + * libsecp256k1 Cleanup - This destroys the secp256k1 context object + * This should be called at the end of the program for proper cleanup of the context. + */ + public static synchronized void cleanup() { + w.lock(); + try { + secp256k1_destroy_context(Secp256k1Context.getContext()); + } finally { + w.unlock(); + } + } + + public static long cloneContext() { + r.lock(); + try { + return secp256k1_ctx_clone(Secp256k1Context.getContext()); + } finally { r.unlock(); } + } + + /** + * libsecp256k1 PrivKey Tweak-Mul - Tweak privkey by multiplying to it + * + * @param tweak some bytes to tweak with + * @param seckey 32-byte seckey + */ + public static byte[] privKeyTweakMul(byte[] privkey, byte[] tweak) throws AssertFailException{ + Preconditions.checkArgument(privkey.length == 32); + + ByteBuffer byteBuff = nativeECDSABuffer.get(); + if (byteBuff == null || byteBuff.capacity() < privkey.length + tweak.length) { + byteBuff = ByteBuffer.allocateDirect(privkey.length + tweak.length); + byteBuff.order(ByteOrder.nativeOrder()); + nativeECDSABuffer.set(byteBuff); + } + byteBuff.rewind(); + byteBuff.put(privkey); + byteBuff.put(tweak); + + byte[][] retByteArray; + r.lock(); + try { + retByteArray = secp256k1_privkey_tweak_mul(byteBuff,Secp256k1Context.getContext()); + } finally { + r.unlock(); + } + + byte[] privArr = retByteArray[0]; + + int privLen = (byte) new BigInteger(new byte[] { retByteArray[1][0] }).intValue() & 0xFF; + int retVal = new BigInteger(new byte[] { retByteArray[1][1] }).intValue(); + + assertEquals(privArr.length, privLen, "Got bad pubkey length."); + + assertEquals(retVal, 1, "Failed return value check."); + + return privArr; + } + + /** + * libsecp256k1 PrivKey Tweak-Add - Tweak privkey by adding to it + * + * @param tweak some bytes to tweak with + * @param seckey 32-byte seckey + */ + public static byte[] privKeyTweakAdd(byte[] privkey, byte[] tweak) throws AssertFailException{ + Preconditions.checkArgument(privkey.length == 32); + + ByteBuffer byteBuff = nativeECDSABuffer.get(); + if (byteBuff == null || byteBuff.capacity() < privkey.length + tweak.length) { + byteBuff = ByteBuffer.allocateDirect(privkey.length + tweak.length); + byteBuff.order(ByteOrder.nativeOrder()); + nativeECDSABuffer.set(byteBuff); + } + byteBuff.rewind(); + byteBuff.put(privkey); + byteBuff.put(tweak); + + byte[][] retByteArray; + r.lock(); + try { + retByteArray = secp256k1_privkey_tweak_add(byteBuff,Secp256k1Context.getContext()); + } finally { + r.unlock(); + } + + byte[] privArr = retByteArray[0]; + + int privLen = (byte) new BigInteger(new byte[] { retByteArray[1][0] }).intValue() & 0xFF; + int retVal = new BigInteger(new byte[] { retByteArray[1][1] }).intValue(); + + assertEquals(privArr.length, privLen, "Got bad pubkey length."); + + assertEquals(retVal, 1, "Failed return value check."); + + return privArr; + } + + /** + * libsecp256k1 PubKey Tweak-Add - Tweak pubkey by adding to it + * + * @param tweak some bytes to tweak with + * @param pubkey 32-byte seckey + */ + public static byte[] pubKeyTweakAdd(byte[] pubkey, byte[] tweak) throws AssertFailException{ + Preconditions.checkArgument(pubkey.length == 33 || pubkey.length == 65); + + ByteBuffer byteBuff = nativeECDSABuffer.get(); + if (byteBuff == null || byteBuff.capacity() < pubkey.length + tweak.length) { + byteBuff = ByteBuffer.allocateDirect(pubkey.length + tweak.length); + byteBuff.order(ByteOrder.nativeOrder()); + nativeECDSABuffer.set(byteBuff); + } + byteBuff.rewind(); + byteBuff.put(pubkey); + byteBuff.put(tweak); + + byte[][] retByteArray; + r.lock(); + try { + retByteArray = secp256k1_pubkey_tweak_add(byteBuff,Secp256k1Context.getContext(), pubkey.length); + } finally { + r.unlock(); + } + + byte[] pubArr = retByteArray[0]; + + int pubLen = (byte) new BigInteger(new byte[] { retByteArray[1][0] }).intValue() & 0xFF; + int retVal = new BigInteger(new byte[] { retByteArray[1][1] }).intValue(); + + assertEquals(pubArr.length, pubLen, "Got bad pubkey length."); + + assertEquals(retVal, 1, "Failed return value check."); + + return pubArr; + } + + /** + * libsecp256k1 PubKey Tweak-Mul - Tweak pubkey by multiplying to it + * + * @param tweak some bytes to tweak with + * @param pubkey 32-byte seckey + */ + public static byte[] pubKeyTweakMul(byte[] pubkey, byte[] tweak) throws AssertFailException{ + Preconditions.checkArgument(pubkey.length == 33 || pubkey.length == 65); + + ByteBuffer byteBuff = nativeECDSABuffer.get(); + if (byteBuff == null || byteBuff.capacity() < pubkey.length + tweak.length) { + byteBuff = ByteBuffer.allocateDirect(pubkey.length + tweak.length); + byteBuff.order(ByteOrder.nativeOrder()); + nativeECDSABuffer.set(byteBuff); + } + byteBuff.rewind(); + byteBuff.put(pubkey); + byteBuff.put(tweak); + + byte[][] retByteArray; + r.lock(); + try { + retByteArray = secp256k1_pubkey_tweak_mul(byteBuff,Secp256k1Context.getContext(), pubkey.length); + } finally { + r.unlock(); + } + + byte[] pubArr = retByteArray[0]; + + int pubLen = (byte) new BigInteger(new byte[] { retByteArray[1][0] }).intValue() & 0xFF; + int retVal = new BigInteger(new byte[] { retByteArray[1][1] }).intValue(); + + assertEquals(pubArr.length, pubLen, "Got bad pubkey length."); + + assertEquals(retVal, 1, "Failed return value check."); + + return pubArr; + } + + /** + * libsecp256k1 create ECDH secret - constant time ECDH calculation + * + * @param seckey byte array of secret key used in exponentiaion + * @param pubkey byte array of public key used in exponentiaion + */ + public static byte[] createECDHSecret(byte[] seckey, byte[] pubkey) throws AssertFailException{ + Preconditions.checkArgument(seckey.length <= 32 && pubkey.length <= 65); + + ByteBuffer byteBuff = nativeECDSABuffer.get(); + if (byteBuff == null || byteBuff.capacity() < 32 + pubkey.length) { + byteBuff = ByteBuffer.allocateDirect(32 + pubkey.length); + byteBuff.order(ByteOrder.nativeOrder()); + nativeECDSABuffer.set(byteBuff); + } + byteBuff.rewind(); + byteBuff.put(seckey); + byteBuff.put(pubkey); + + byte[][] retByteArray; + r.lock(); + try { + retByteArray = secp256k1_ecdh(byteBuff, Secp256k1Context.getContext(), pubkey.length); + } finally { + r.unlock(); + } + + byte[] resArr = retByteArray[0]; + int retVal = new BigInteger(new byte[] { retByteArray[1][0] }).intValue(); + + assertEquals(resArr.length, 32, "Got bad result length."); + assertEquals(retVal, 1, "Failed return value check."); + + return resArr; + } + + /** + * libsecp256k1 randomize - updates the context randomization + * + * @param seed 32-byte random seed + */ + public static synchronized boolean randomize(byte[] seed) throws AssertFailException{ + Preconditions.checkArgument(seed.length == 32 || seed == null); + + ByteBuffer byteBuff = nativeECDSABuffer.get(); + if (byteBuff == null || byteBuff.capacity() < seed.length) { + byteBuff = ByteBuffer.allocateDirect(seed.length); + byteBuff.order(ByteOrder.nativeOrder()); + nativeECDSABuffer.set(byteBuff); + } + byteBuff.rewind(); + byteBuff.put(seed); + + w.lock(); + try { + return secp256k1_context_randomize(byteBuff, Secp256k1Context.getContext()) == 1; + } finally { + w.unlock(); + } + } + + private static native long secp256k1_ctx_clone(long context); + + private static native int secp256k1_context_randomize(ByteBuffer byteBuff, long context); + + private static native byte[][] secp256k1_privkey_tweak_add(ByteBuffer byteBuff, long context); + + private static native byte[][] secp256k1_privkey_tweak_mul(ByteBuffer byteBuff, long context); + + private static native byte[][] secp256k1_pubkey_tweak_add(ByteBuffer byteBuff, long context, int pubLen); + + private static native byte[][] secp256k1_pubkey_tweak_mul(ByteBuffer byteBuff, long context, int pubLen); + + private static native void secp256k1_destroy_context(long context); + + private static native int secp256k1_ecdsa_verify(ByteBuffer byteBuff, long context, int sigLen, int pubLen); + + private static native byte[][] secp256k1_ecdsa_sign(ByteBuffer byteBuff, long context); + + private static native int secp256k1_ec_seckey_verify(ByteBuffer byteBuff, long context); + + private static native byte[][] secp256k1_ec_pubkey_create(ByteBuffer byteBuff, long context); + + private static native byte[][] secp256k1_ec_pubkey_parse(ByteBuffer byteBuff, long context, int inputLen); + + private static native byte[][] secp256k1_ecdh(ByteBuffer byteBuff, long context, int inputLen); + +} diff --git a/node_modules/secp256k1/src/secp256k1/src/java/org/bitcoin/NativeSecp256k1Test.java b/node_modules/secp256k1/src/secp256k1/src/java/org/bitcoin/NativeSecp256k1Test.java new file mode 100644 index 0000000..710d9f0 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/java/org/bitcoin/NativeSecp256k1Test.java @@ -0,0 +1,225 @@ +package org.bitcoin; + +import com.google.common.io.BaseEncoding; +import java.util.Arrays; +import java.math.BigInteger; +import static org.bitcoin.NativeSecp256k1Util.*; + +/** + * This class holds test cases defined for testing this library. + */ +public class NativeSecp256k1Test { + + //TODO improve comments/add more tests + /** + * This tests verify() for a valid signature + */ + public static void testVerifyPos() throws AssertFailException{ + boolean result = false; + byte[] data = BaseEncoding.base16().lowerCase().decode("CF80CD8AED482D5D1527D7DC72FCEFF84E6326592848447D2DC0B0E87DFC9A90".toLowerCase()); //sha256hash of "testing" + byte[] sig = BaseEncoding.base16().lowerCase().decode("3044022079BE667EF9DCBBAC55A06295CE870B07029BFCDB2DCE28D959F2815B16F817980220294F14E883B3F525B5367756C2A11EF6CF84B730B36C17CB0C56F0AAB2C98589".toLowerCase()); + byte[] pub = BaseEncoding.base16().lowerCase().decode("040A629506E1B65CD9D2E0BA9C75DF9C4FED0DB16DC9625ED14397F0AFC836FAE595DC53F8B0EFE61E703075BD9B143BAC75EC0E19F82A2208CAEB32BE53414C40".toLowerCase()); + + result = NativeSecp256k1.verify( data, sig, pub); + assertEquals( result, true , "testVerifyPos"); + } + + /** + * This tests verify() for a non-valid signature + */ + public static void testVerifyNeg() throws AssertFailException{ + boolean result = false; + byte[] data = BaseEncoding.base16().lowerCase().decode("CF80CD8AED482D5D1527D7DC72FCEFF84E6326592848447D2DC0B0E87DFC9A91".toLowerCase()); //sha256hash of "testing" + byte[] sig = BaseEncoding.base16().lowerCase().decode("3044022079BE667EF9DCBBAC55A06295CE870B07029BFCDB2DCE28D959F2815B16F817980220294F14E883B3F525B5367756C2A11EF6CF84B730B36C17CB0C56F0AAB2C98589".toLowerCase()); + byte[] pub = BaseEncoding.base16().lowerCase().decode("040A629506E1B65CD9D2E0BA9C75DF9C4FED0DB16DC9625ED14397F0AFC836FAE595DC53F8B0EFE61E703075BD9B143BAC75EC0E19F82A2208CAEB32BE53414C40".toLowerCase()); + + result = NativeSecp256k1.verify( data, sig, pub); + //System.out.println(" TEST " + new BigInteger(1, resultbytes).toString(16)); + assertEquals( result, false , "testVerifyNeg"); + } + + /** + * This tests secret key verify() for a valid secretkey + */ + public static void testSecKeyVerifyPos() throws AssertFailException{ + boolean result = false; + byte[] sec = BaseEncoding.base16().lowerCase().decode("67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530".toLowerCase()); + + result = NativeSecp256k1.secKeyVerify( sec ); + //System.out.println(" TEST " + new BigInteger(1, resultbytes).toString(16)); + assertEquals( result, true , "testSecKeyVerifyPos"); + } + + /** + * This tests secret key verify() for an invalid secretkey + */ + public static void testSecKeyVerifyNeg() throws AssertFailException{ + boolean result = false; + byte[] sec = BaseEncoding.base16().lowerCase().decode("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF".toLowerCase()); + + result = NativeSecp256k1.secKeyVerify( sec ); + //System.out.println(" TEST " + new BigInteger(1, resultbytes).toString(16)); + assertEquals( result, false , "testSecKeyVerifyNeg"); + } + + /** + * This tests public key create() for a valid secretkey + */ + public static void testPubKeyCreatePos() throws AssertFailException{ + byte[] sec = BaseEncoding.base16().lowerCase().decode("67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530".toLowerCase()); + + byte[] resultArr = NativeSecp256k1.computePubkey( sec); + String pubkeyString = BaseEncoding.base16().encode(resultArr); + assertEquals( pubkeyString , "04C591A8FF19AC9C4E4E5793673B83123437E975285E7B442F4EE2654DFFCA5E2D2103ED494718C697AC9AEBCFD19612E224DB46661011863ED2FC54E71861E2A6" , "testPubKeyCreatePos"); + } + + /** + * This tests public key create() for a invalid secretkey + */ + public static void testPubKeyCreateNeg() throws AssertFailException{ + byte[] sec = BaseEncoding.base16().lowerCase().decode("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF".toLowerCase()); + + byte[] resultArr = NativeSecp256k1.computePubkey( sec); + String pubkeyString = BaseEncoding.base16().encode(resultArr); + assertEquals( pubkeyString, "" , "testPubKeyCreateNeg"); + } + + /** + * This tests sign() for a valid secretkey + */ + public static void testSignPos() throws AssertFailException{ + + byte[] data = BaseEncoding.base16().lowerCase().decode("CF80CD8AED482D5D1527D7DC72FCEFF84E6326592848447D2DC0B0E87DFC9A90".toLowerCase()); //sha256hash of "testing" + byte[] sec = BaseEncoding.base16().lowerCase().decode("67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530".toLowerCase()); + + byte[] resultArr = NativeSecp256k1.sign(data, sec); + String sigString = BaseEncoding.base16().encode(resultArr); + assertEquals( sigString, "30440220182A108E1448DC8F1FB467D06A0F3BB8EA0533584CB954EF8DA112F1D60E39A202201C66F36DA211C087F3AF88B50EDF4F9BDAA6CF5FD6817E74DCA34DB12390C6E9" , "testSignPos"); + } + + /** + * This tests sign() for a invalid secretkey + */ + public static void testSignNeg() throws AssertFailException{ + byte[] data = BaseEncoding.base16().lowerCase().decode("CF80CD8AED482D5D1527D7DC72FCEFF84E6326592848447D2DC0B0E87DFC9A90".toLowerCase()); //sha256hash of "testing" + byte[] sec = BaseEncoding.base16().lowerCase().decode("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF".toLowerCase()); + + byte[] resultArr = NativeSecp256k1.sign(data, sec); + String sigString = BaseEncoding.base16().encode(resultArr); + assertEquals( sigString, "" , "testSignNeg"); + } + + /** + * This tests private key tweak-add + */ + public static void testPrivKeyTweakAdd_1() throws AssertFailException { + byte[] sec = BaseEncoding.base16().lowerCase().decode("67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530".toLowerCase()); + byte[] data = BaseEncoding.base16().lowerCase().decode("3982F19BEF1615BCCFBB05E321C10E1D4CBA3DF0E841C2E41EEB6016347653C3".toLowerCase()); //sha256hash of "tweak" + + byte[] resultArr = NativeSecp256k1.privKeyTweakAdd( sec , data ); + String sigString = BaseEncoding.base16().encode(resultArr); + assertEquals( sigString , "A168571E189E6F9A7E2D657A4B53AE99B909F7E712D1C23CED28093CD57C88F3" , "testPrivKeyAdd_1"); + } + + /** + * This tests private key tweak-mul + */ + public static void testPrivKeyTweakMul_1() throws AssertFailException { + byte[] sec = BaseEncoding.base16().lowerCase().decode("67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530".toLowerCase()); + byte[] data = BaseEncoding.base16().lowerCase().decode("3982F19BEF1615BCCFBB05E321C10E1D4CBA3DF0E841C2E41EEB6016347653C3".toLowerCase()); //sha256hash of "tweak" + + byte[] resultArr = NativeSecp256k1.privKeyTweakMul( sec , data ); + String sigString = BaseEncoding.base16().encode(resultArr); + assertEquals( sigString , "97F8184235F101550F3C71C927507651BD3F1CDB4A5A33B8986ACF0DEE20FFFC" , "testPrivKeyMul_1"); + } + + /** + * This tests private key tweak-add uncompressed + */ + public static void testPrivKeyTweakAdd_2() throws AssertFailException { + byte[] pub = BaseEncoding.base16().lowerCase().decode("040A629506E1B65CD9D2E0BA9C75DF9C4FED0DB16DC9625ED14397F0AFC836FAE595DC53F8B0EFE61E703075BD9B143BAC75EC0E19F82A2208CAEB32BE53414C40".toLowerCase()); + byte[] data = BaseEncoding.base16().lowerCase().decode("3982F19BEF1615BCCFBB05E321C10E1D4CBA3DF0E841C2E41EEB6016347653C3".toLowerCase()); //sha256hash of "tweak" + + byte[] resultArr = NativeSecp256k1.pubKeyTweakAdd( pub , data ); + String sigString = BaseEncoding.base16().encode(resultArr); + assertEquals( sigString , "0411C6790F4B663CCE607BAAE08C43557EDC1A4D11D88DFCB3D841D0C6A941AF525A268E2A863C148555C48FB5FBA368E88718A46E205FABC3DBA2CCFFAB0796EF" , "testPrivKeyAdd_2"); + } + + /** + * This tests private key tweak-mul uncompressed + */ + public static void testPrivKeyTweakMul_2() throws AssertFailException { + byte[] pub = BaseEncoding.base16().lowerCase().decode("040A629506E1B65CD9D2E0BA9C75DF9C4FED0DB16DC9625ED14397F0AFC836FAE595DC53F8B0EFE61E703075BD9B143BAC75EC0E19F82A2208CAEB32BE53414C40".toLowerCase()); + byte[] data = BaseEncoding.base16().lowerCase().decode("3982F19BEF1615BCCFBB05E321C10E1D4CBA3DF0E841C2E41EEB6016347653C3".toLowerCase()); //sha256hash of "tweak" + + byte[] resultArr = NativeSecp256k1.pubKeyTweakMul( pub , data ); + String sigString = BaseEncoding.base16().encode(resultArr); + assertEquals( sigString , "04E0FE6FE55EBCA626B98A807F6CAF654139E14E5E3698F01A9A658E21DC1D2791EC060D4F412A794D5370F672BC94B722640B5F76914151CFCA6E712CA48CC589" , "testPrivKeyMul_2"); + } + + /** + * This tests seed randomization + */ + public static void testRandomize() throws AssertFailException { + byte[] seed = BaseEncoding.base16().lowerCase().decode("A441B15FE9A3CF56661190A0B93B9DEC7D04127288CC87250967CF3B52894D11".toLowerCase()); //sha256hash of "random" + boolean result = NativeSecp256k1.randomize(seed); + assertEquals( result, true, "testRandomize"); + } + + public static void testCreateECDHSecret() throws AssertFailException{ + + byte[] sec = BaseEncoding.base16().lowerCase().decode("67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530".toLowerCase()); + byte[] pub = BaseEncoding.base16().lowerCase().decode("040A629506E1B65CD9D2E0BA9C75DF9C4FED0DB16DC9625ED14397F0AFC836FAE595DC53F8B0EFE61E703075BD9B143BAC75EC0E19F82A2208CAEB32BE53414C40".toLowerCase()); + + byte[] resultArr = NativeSecp256k1.createECDHSecret(sec, pub); + String ecdhString = BaseEncoding.base16().encode(resultArr); + assertEquals( ecdhString, "2A2A67007A926E6594AF3EB564FC74005B37A9C8AEF2033C4552051B5C87F043" , "testCreateECDHSecret"); + } + + public static void main(String[] args) throws AssertFailException{ + + + System.out.println("\n libsecp256k1 enabled: " + Secp256k1Context.isEnabled() + "\n"); + + assertEquals( Secp256k1Context.isEnabled(), true, "isEnabled" ); + + //Test verify() success/fail + testVerifyPos(); + testVerifyNeg(); + + //Test secKeyVerify() success/fail + testSecKeyVerifyPos(); + testSecKeyVerifyNeg(); + + //Test computePubkey() success/fail + testPubKeyCreatePos(); + testPubKeyCreateNeg(); + + //Test sign() success/fail + testSignPos(); + testSignNeg(); + + //Test privKeyTweakAdd() 1 + testPrivKeyTweakAdd_1(); + + //Test privKeyTweakMul() 2 + testPrivKeyTweakMul_1(); + + //Test privKeyTweakAdd() 3 + testPrivKeyTweakAdd_2(); + + //Test privKeyTweakMul() 4 + testPrivKeyTweakMul_2(); + + //Test randomize() + testRandomize(); + + //Test ECDH + testCreateECDHSecret(); + + NativeSecp256k1.cleanup(); + + System.out.println(" All tests passed." ); + + } +} diff --git a/node_modules/secp256k1/src/secp256k1/src/java/org/bitcoin/NativeSecp256k1Util.java b/node_modules/secp256k1/src/secp256k1/src/java/org/bitcoin/NativeSecp256k1Util.java new file mode 100644 index 0000000..04732ba --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/java/org/bitcoin/NativeSecp256k1Util.java @@ -0,0 +1,45 @@ +/* + * Copyright 2014-2016 the libsecp256k1 contributors + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.bitcoin; + +public class NativeSecp256k1Util{ + + public static void assertEquals( int val, int val2, String message ) throws AssertFailException{ + if( val != val2 ) + throw new AssertFailException("FAIL: " + message); + } + + public static void assertEquals( boolean val, boolean val2, String message ) throws AssertFailException{ + if( val != val2 ) + throw new AssertFailException("FAIL: " + message); + else + System.out.println("PASS: " + message); + } + + public static void assertEquals( String val, String val2, String message ) throws AssertFailException{ + if( !val.equals(val2) ) + throw new AssertFailException("FAIL: " + message); + else + System.out.println("PASS: " + message); + } + + public static class AssertFailException extends Exception { + public AssertFailException(String message) { + super( message ); + } + } +} diff --git a/node_modules/secp256k1/src/secp256k1/src/java/org/bitcoin/Secp256k1Context.java b/node_modules/secp256k1/src/secp256k1/src/java/org/bitcoin/Secp256k1Context.java new file mode 100644 index 0000000..216c986 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/java/org/bitcoin/Secp256k1Context.java @@ -0,0 +1,51 @@ +/* + * Copyright 2014-2016 the libsecp256k1 contributors + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.bitcoin; + +/** + * This class holds the context reference used in native methods + * to handle ECDSA operations. + */ +public class Secp256k1Context { + private static final boolean enabled; //true if the library is loaded + private static final long context; //ref to pointer to context obj + + static { //static initializer + boolean isEnabled = true; + long contextRef = -1; + try { + System.loadLibrary("secp256k1"); + contextRef = secp256k1_init_context(); + } catch (UnsatisfiedLinkError e) { + System.out.println("UnsatisfiedLinkError: " + e.toString()); + isEnabled = false; + } + enabled = isEnabled; + context = contextRef; + } + + public static boolean isEnabled() { + return enabled; + } + + public static long getContext() { + if(!enabled) return -1; //sanity check + return context; + } + + private static native long secp256k1_init_context(); +} diff --git a/node_modules/secp256k1/src/secp256k1/src/java/org_bitcoin_NativeSecp256k1.c b/node_modules/secp256k1/src/secp256k1/src/java/org_bitcoin_NativeSecp256k1.c new file mode 100644 index 0000000..b590256 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/java/org_bitcoin_NativeSecp256k1.c @@ -0,0 +1,379 @@ +#include +#include +#include +#include "org_bitcoin_NativeSecp256k1.h" +#include "include/secp256k1.h" +#include "include/secp256k1_ecdh.h" +#include "include/secp256k1_recovery.h" + + +SECP256K1_API jlong JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ctx_1clone + (JNIEnv* env, jclass classObject, jlong ctx_l) +{ + const secp256k1_context *ctx = (secp256k1_context*)(uintptr_t)ctx_l; + + jlong ctx_clone_l = (uintptr_t) secp256k1_context_clone(ctx); + + (void)classObject;(void)env; + + return ctx_clone_l; + +} + +SECP256K1_API jint JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1context_1randomize + (JNIEnv* env, jclass classObject, jobject byteBufferObject, jlong ctx_l) +{ + secp256k1_context *ctx = (secp256k1_context*)(uintptr_t)ctx_l; + + const unsigned char* seed = (unsigned char*) (*env)->GetDirectBufferAddress(env, byteBufferObject); + + (void)classObject; + + return secp256k1_context_randomize(ctx, seed); + +} + +SECP256K1_API void JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1destroy_1context + (JNIEnv* env, jclass classObject, jlong ctx_l) +{ + secp256k1_context *ctx = (secp256k1_context*)(uintptr_t)ctx_l; + + secp256k1_context_destroy(ctx); + + (void)classObject;(void)env; +} + +SECP256K1_API jint JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ecdsa_1verify + (JNIEnv* env, jclass classObject, jobject byteBufferObject, jlong ctx_l, jint siglen, jint publen) +{ + secp256k1_context *ctx = (secp256k1_context*)(uintptr_t)ctx_l; + + unsigned char* data = (unsigned char*) (*env)->GetDirectBufferAddress(env, byteBufferObject); + const unsigned char* sigdata = { (unsigned char*) (data + 32) }; + const unsigned char* pubdata = { (unsigned char*) (data + siglen + 32) }; + + secp256k1_ecdsa_signature sig; + secp256k1_pubkey pubkey; + + int ret = secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigdata, siglen); + + if( ret ) { + ret = secp256k1_ec_pubkey_parse(ctx, &pubkey, pubdata, publen); + + if( ret ) { + ret = secp256k1_ecdsa_verify(ctx, &sig, data, &pubkey); + } + } + + (void)classObject; + + return ret; +} + +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ecdsa_1sign + (JNIEnv* env, jclass classObject, jobject byteBufferObject, jlong ctx_l) +{ + secp256k1_context *ctx = (secp256k1_context*)(uintptr_t)ctx_l; + unsigned char* data = (unsigned char*) (*env)->GetDirectBufferAddress(env, byteBufferObject); + unsigned char* secKey = (unsigned char*) (data + 32); + + jobjectArray retArray; + jbyteArray sigArray, intsByteArray; + unsigned char intsarray[2]; + + secp256k1_ecdsa_signature sig; + + int ret = secp256k1_ecdsa_sign(ctx, &sig, data, secKey, NULL, NULL); + + unsigned char outputSer[72]; + size_t outputLen = 72; + + if( ret ) { + int ret2 = secp256k1_ecdsa_signature_serialize_der(ctx,outputSer, &outputLen, &sig ); (void)ret2; + } + + intsarray[0] = outputLen; + intsarray[1] = ret; + + retArray = (*env)->NewObjectArray(env, 2, + (*env)->FindClass(env, "[B"), + (*env)->NewByteArray(env, 1)); + + sigArray = (*env)->NewByteArray(env, outputLen); + (*env)->SetByteArrayRegion(env, sigArray, 0, outputLen, (jbyte*)outputSer); + (*env)->SetObjectArrayElement(env, retArray, 0, sigArray); + + intsByteArray = (*env)->NewByteArray(env, 2); + (*env)->SetByteArrayRegion(env, intsByteArray, 0, 2, (jbyte*)intsarray); + (*env)->SetObjectArrayElement(env, retArray, 1, intsByteArray); + + (void)classObject; + + return retArray; +} + +SECP256K1_API jint JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ec_1seckey_1verify + (JNIEnv* env, jclass classObject, jobject byteBufferObject, jlong ctx_l) +{ + secp256k1_context *ctx = (secp256k1_context*)(uintptr_t)ctx_l; + unsigned char* secKey = (unsigned char*) (*env)->GetDirectBufferAddress(env, byteBufferObject); + + (void)classObject; + + return secp256k1_ec_seckey_verify(ctx, secKey); +} + +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ec_1pubkey_1create + (JNIEnv* env, jclass classObject, jobject byteBufferObject, jlong ctx_l) +{ + secp256k1_context *ctx = (secp256k1_context*)(uintptr_t)ctx_l; + const unsigned char* secKey = (unsigned char*) (*env)->GetDirectBufferAddress(env, byteBufferObject); + + secp256k1_pubkey pubkey; + + jobjectArray retArray; + jbyteArray pubkeyArray, intsByteArray; + unsigned char intsarray[2]; + + int ret = secp256k1_ec_pubkey_create(ctx, &pubkey, secKey); + + unsigned char outputSer[65]; + size_t outputLen = 65; + + if( ret ) { + int ret2 = secp256k1_ec_pubkey_serialize(ctx,outputSer, &outputLen, &pubkey,SECP256K1_EC_UNCOMPRESSED );(void)ret2; + } + + intsarray[0] = outputLen; + intsarray[1] = ret; + + retArray = (*env)->NewObjectArray(env, 2, + (*env)->FindClass(env, "[B"), + (*env)->NewByteArray(env, 1)); + + pubkeyArray = (*env)->NewByteArray(env, outputLen); + (*env)->SetByteArrayRegion(env, pubkeyArray, 0, outputLen, (jbyte*)outputSer); + (*env)->SetObjectArrayElement(env, retArray, 0, pubkeyArray); + + intsByteArray = (*env)->NewByteArray(env, 2); + (*env)->SetByteArrayRegion(env, intsByteArray, 0, 2, (jbyte*)intsarray); + (*env)->SetObjectArrayElement(env, retArray, 1, intsByteArray); + + (void)classObject; + + return retArray; + +} + +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1privkey_1tweak_1add + (JNIEnv* env, jclass classObject, jobject byteBufferObject, jlong ctx_l) +{ + secp256k1_context *ctx = (secp256k1_context*)(uintptr_t)ctx_l; + unsigned char* privkey = (unsigned char*) (*env)->GetDirectBufferAddress(env, byteBufferObject); + const unsigned char* tweak = (unsigned char*) (privkey + 32); + + jobjectArray retArray; + jbyteArray privArray, intsByteArray; + unsigned char intsarray[2]; + + int privkeylen = 32; + + int ret = secp256k1_ec_privkey_tweak_add(ctx, privkey, tweak); + + intsarray[0] = privkeylen; + intsarray[1] = ret; + + retArray = (*env)->NewObjectArray(env, 2, + (*env)->FindClass(env, "[B"), + (*env)->NewByteArray(env, 1)); + + privArray = (*env)->NewByteArray(env, privkeylen); + (*env)->SetByteArrayRegion(env, privArray, 0, privkeylen, (jbyte*)privkey); + (*env)->SetObjectArrayElement(env, retArray, 0, privArray); + + intsByteArray = (*env)->NewByteArray(env, 2); + (*env)->SetByteArrayRegion(env, intsByteArray, 0, 2, (jbyte*)intsarray); + (*env)->SetObjectArrayElement(env, retArray, 1, intsByteArray); + + (void)classObject; + + return retArray; +} + +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1privkey_1tweak_1mul + (JNIEnv* env, jclass classObject, jobject byteBufferObject, jlong ctx_l) +{ + secp256k1_context *ctx = (secp256k1_context*)(uintptr_t)ctx_l; + unsigned char* privkey = (unsigned char*) (*env)->GetDirectBufferAddress(env, byteBufferObject); + const unsigned char* tweak = (unsigned char*) (privkey + 32); + + jobjectArray retArray; + jbyteArray privArray, intsByteArray; + unsigned char intsarray[2]; + + int privkeylen = 32; + + int ret = secp256k1_ec_privkey_tweak_mul(ctx, privkey, tweak); + + intsarray[0] = privkeylen; + intsarray[1] = ret; + + retArray = (*env)->NewObjectArray(env, 2, + (*env)->FindClass(env, "[B"), + (*env)->NewByteArray(env, 1)); + + privArray = (*env)->NewByteArray(env, privkeylen); + (*env)->SetByteArrayRegion(env, privArray, 0, privkeylen, (jbyte*)privkey); + (*env)->SetObjectArrayElement(env, retArray, 0, privArray); + + intsByteArray = (*env)->NewByteArray(env, 2); + (*env)->SetByteArrayRegion(env, intsByteArray, 0, 2, (jbyte*)intsarray); + (*env)->SetObjectArrayElement(env, retArray, 1, intsByteArray); + + (void)classObject; + + return retArray; +} + +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1pubkey_1tweak_1add + (JNIEnv* env, jclass classObject, jobject byteBufferObject, jlong ctx_l, jint publen) +{ + secp256k1_context *ctx = (secp256k1_context*)(uintptr_t)ctx_l; +/* secp256k1_pubkey* pubkey = (secp256k1_pubkey*) (*env)->GetDirectBufferAddress(env, byteBufferObject);*/ + unsigned char* pkey = (*env)->GetDirectBufferAddress(env, byteBufferObject); + const unsigned char* tweak = (unsigned char*) (pkey + publen); + + jobjectArray retArray; + jbyteArray pubArray, intsByteArray; + unsigned char intsarray[2]; + unsigned char outputSer[65]; + size_t outputLen = 65; + + secp256k1_pubkey pubkey; + int ret = secp256k1_ec_pubkey_parse(ctx, &pubkey, pkey, publen); + + if( ret ) { + ret = secp256k1_ec_pubkey_tweak_add(ctx, &pubkey, tweak); + } + + if( ret ) { + int ret2 = secp256k1_ec_pubkey_serialize(ctx,outputSer, &outputLen, &pubkey,SECP256K1_EC_UNCOMPRESSED );(void)ret2; + } + + intsarray[0] = outputLen; + intsarray[1] = ret; + + retArray = (*env)->NewObjectArray(env, 2, + (*env)->FindClass(env, "[B"), + (*env)->NewByteArray(env, 1)); + + pubArray = (*env)->NewByteArray(env, outputLen); + (*env)->SetByteArrayRegion(env, pubArray, 0, outputLen, (jbyte*)outputSer); + (*env)->SetObjectArrayElement(env, retArray, 0, pubArray); + + intsByteArray = (*env)->NewByteArray(env, 2); + (*env)->SetByteArrayRegion(env, intsByteArray, 0, 2, (jbyte*)intsarray); + (*env)->SetObjectArrayElement(env, retArray, 1, intsByteArray); + + (void)classObject; + + return retArray; +} + +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1pubkey_1tweak_1mul + (JNIEnv* env, jclass classObject, jobject byteBufferObject, jlong ctx_l, jint publen) +{ + secp256k1_context *ctx = (secp256k1_context*)(uintptr_t)ctx_l; + unsigned char* pkey = (*env)->GetDirectBufferAddress(env, byteBufferObject); + const unsigned char* tweak = (unsigned char*) (pkey + publen); + + jobjectArray retArray; + jbyteArray pubArray, intsByteArray; + unsigned char intsarray[2]; + unsigned char outputSer[65]; + size_t outputLen = 65; + + secp256k1_pubkey pubkey; + int ret = secp256k1_ec_pubkey_parse(ctx, &pubkey, pkey, publen); + + if ( ret ) { + ret = secp256k1_ec_pubkey_tweak_mul(ctx, &pubkey, tweak); + } + + if( ret ) { + int ret2 = secp256k1_ec_pubkey_serialize(ctx,outputSer, &outputLen, &pubkey,SECP256K1_EC_UNCOMPRESSED );(void)ret2; + } + + intsarray[0] = outputLen; + intsarray[1] = ret; + + retArray = (*env)->NewObjectArray(env, 2, + (*env)->FindClass(env, "[B"), + (*env)->NewByteArray(env, 1)); + + pubArray = (*env)->NewByteArray(env, outputLen); + (*env)->SetByteArrayRegion(env, pubArray, 0, outputLen, (jbyte*)outputSer); + (*env)->SetObjectArrayElement(env, retArray, 0, pubArray); + + intsByteArray = (*env)->NewByteArray(env, 2); + (*env)->SetByteArrayRegion(env, intsByteArray, 0, 2, (jbyte*)intsarray); + (*env)->SetObjectArrayElement(env, retArray, 1, intsByteArray); + + (void)classObject; + + return retArray; +} + +SECP256K1_API jlong JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ecdsa_1pubkey_1combine + (JNIEnv * env, jclass classObject, jobject byteBufferObject, jlong ctx_l, jint numkeys) +{ + (void)classObject;(void)env;(void)byteBufferObject;(void)ctx_l;(void)numkeys; + + return 0; +} + +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ecdh + (JNIEnv* env, jclass classObject, jobject byteBufferObject, jlong ctx_l, jint publen) +{ + secp256k1_context *ctx = (secp256k1_context*)(uintptr_t)ctx_l; + const unsigned char* secdata = (*env)->GetDirectBufferAddress(env, byteBufferObject); + const unsigned char* pubdata = (const unsigned char*) (secdata + 32); + + jobjectArray retArray; + jbyteArray outArray, intsByteArray; + unsigned char intsarray[1]; + secp256k1_pubkey pubkey; + unsigned char nonce_res[32]; + size_t outputLen = 32; + + int ret = secp256k1_ec_pubkey_parse(ctx, &pubkey, pubdata, publen); + + if (ret) { + ret = secp256k1_ecdh( + ctx, + nonce_res, + &pubkey, + secdata, + NULL, + NULL + ); + } + + intsarray[0] = ret; + + retArray = (*env)->NewObjectArray(env, 2, + (*env)->FindClass(env, "[B"), + (*env)->NewByteArray(env, 1)); + + outArray = (*env)->NewByteArray(env, outputLen); + (*env)->SetByteArrayRegion(env, outArray, 0, 32, (jbyte*)nonce_res); + (*env)->SetObjectArrayElement(env, retArray, 0, outArray); + + intsByteArray = (*env)->NewByteArray(env, 1); + (*env)->SetByteArrayRegion(env, intsByteArray, 0, 1, (jbyte*)intsarray); + (*env)->SetObjectArrayElement(env, retArray, 1, intsByteArray); + + (void)classObject; + + return retArray; +} diff --git a/node_modules/secp256k1/src/secp256k1/src/java/org_bitcoin_NativeSecp256k1.h b/node_modules/secp256k1/src/secp256k1/src/java/org_bitcoin_NativeSecp256k1.h new file mode 100644 index 0000000..fe613c9 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/java/org_bitcoin_NativeSecp256k1.h @@ -0,0 +1,119 @@ +/* DO NOT EDIT THIS FILE - it is machine generated */ +#include +#include "include/secp256k1.h" +/* Header for class org_bitcoin_NativeSecp256k1 */ + +#ifndef _Included_org_bitcoin_NativeSecp256k1 +#define _Included_org_bitcoin_NativeSecp256k1 +#ifdef __cplusplus +extern "C" { +#endif +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_ctx_clone + * Signature: (J)J + */ +SECP256K1_API jlong JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ctx_1clone + (JNIEnv *, jclass, jlong); + +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_context_randomize + * Signature: (Ljava/nio/ByteBuffer;J)I + */ +SECP256K1_API jint JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1context_1randomize + (JNIEnv *, jclass, jobject, jlong); + +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_privkey_tweak_add + * Signature: (Ljava/nio/ByteBuffer;J)[[B + */ +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1privkey_1tweak_1add + (JNIEnv *, jclass, jobject, jlong); + +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_privkey_tweak_mul + * Signature: (Ljava/nio/ByteBuffer;J)[[B + */ +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1privkey_1tweak_1mul + (JNIEnv *, jclass, jobject, jlong); + +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_pubkey_tweak_add + * Signature: (Ljava/nio/ByteBuffer;JI)[[B + */ +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1pubkey_1tweak_1add + (JNIEnv *, jclass, jobject, jlong, jint); + +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_pubkey_tweak_mul + * Signature: (Ljava/nio/ByteBuffer;JI)[[B + */ +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1pubkey_1tweak_1mul + (JNIEnv *, jclass, jobject, jlong, jint); + +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_destroy_context + * Signature: (J)V + */ +SECP256K1_API void JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1destroy_1context + (JNIEnv *, jclass, jlong); + +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_ecdsa_verify + * Signature: (Ljava/nio/ByteBuffer;JII)I + */ +SECP256K1_API jint JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ecdsa_1verify + (JNIEnv *, jclass, jobject, jlong, jint, jint); + +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_ecdsa_sign + * Signature: (Ljava/nio/ByteBuffer;J)[[B + */ +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ecdsa_1sign + (JNIEnv *, jclass, jobject, jlong); + +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_ec_seckey_verify + * Signature: (Ljava/nio/ByteBuffer;J)I + */ +SECP256K1_API jint JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ec_1seckey_1verify + (JNIEnv *, jclass, jobject, jlong); + +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_ec_pubkey_create + * Signature: (Ljava/nio/ByteBuffer;J)[[B + */ +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ec_1pubkey_1create + (JNIEnv *, jclass, jobject, jlong); + +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_ec_pubkey_parse + * Signature: (Ljava/nio/ByteBuffer;JI)[[B + */ +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ec_1pubkey_1parse + (JNIEnv *, jclass, jobject, jlong, jint); + +/* + * Class: org_bitcoin_NativeSecp256k1 + * Method: secp256k1_ecdh + * Signature: (Ljava/nio/ByteBuffer;JI)[[B + */ +SECP256K1_API jobjectArray JNICALL Java_org_bitcoin_NativeSecp256k1_secp256k1_1ecdh + (JNIEnv* env, jclass classObject, jobject byteBufferObject, jlong ctx_l, jint publen); + + +#ifdef __cplusplus +} +#endif +#endif diff --git a/node_modules/secp256k1/src/secp256k1/src/java/org_bitcoin_Secp256k1Context.c b/node_modules/secp256k1/src/secp256k1/src/java/org_bitcoin_Secp256k1Context.c new file mode 100644 index 0000000..a52939e --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/java/org_bitcoin_Secp256k1Context.c @@ -0,0 +1,15 @@ +#include +#include +#include "org_bitcoin_Secp256k1Context.h" +#include "include/secp256k1.h" + +SECP256K1_API jlong JNICALL Java_org_bitcoin_Secp256k1Context_secp256k1_1init_1context + (JNIEnv* env, jclass classObject) +{ + secp256k1_context *ctx = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY); + + (void)classObject;(void)env; + + return (uintptr_t)ctx; +} + diff --git a/node_modules/secp256k1/src/secp256k1/src/java/org_bitcoin_Secp256k1Context.h b/node_modules/secp256k1/src/secp256k1/src/java/org_bitcoin_Secp256k1Context.h new file mode 100644 index 0000000..0d2bc84 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/java/org_bitcoin_Secp256k1Context.h @@ -0,0 +1,22 @@ +/* DO NOT EDIT THIS FILE - it is machine generated */ +#include +#include "include/secp256k1.h" +/* Header for class org_bitcoin_Secp256k1Context */ + +#ifndef _Included_org_bitcoin_Secp256k1Context +#define _Included_org_bitcoin_Secp256k1Context +#ifdef __cplusplus +extern "C" { +#endif +/* + * Class: org_bitcoin_Secp256k1Context + * Method: secp256k1_init_context + * Signature: ()J + */ +SECP256K1_API jlong JNICALL Java_org_bitcoin_Secp256k1Context_secp256k1_1init_1context + (JNIEnv *, jclass); + +#ifdef __cplusplus +} +#endif +#endif diff --git a/node_modules/secp256k1/src/secp256k1/src/modules/ecdh/Makefile.am.include b/node_modules/secp256k1/src/secp256k1/src/modules/ecdh/Makefile.am.include new file mode 100644 index 0000000..e3088b4 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/modules/ecdh/Makefile.am.include @@ -0,0 +1,8 @@ +include_HEADERS += include/secp256k1_ecdh.h +noinst_HEADERS += src/modules/ecdh/main_impl.h +noinst_HEADERS += src/modules/ecdh/tests_impl.h +if USE_BENCHMARK +noinst_PROGRAMS += bench_ecdh +bench_ecdh_SOURCES = src/bench_ecdh.c +bench_ecdh_LDADD = libsecp256k1.la $(SECP_LIBS) $(COMMON_LIB) +endif diff --git a/node_modules/secp256k1/src/secp256k1/src/modules/ecdh/main_impl.h b/node_modules/secp256k1/src/secp256k1/src/modules/ecdh/main_impl.h new file mode 100644 index 0000000..44cb68e --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/modules/ecdh/main_impl.h @@ -0,0 +1,67 @@ +/********************************************************************** + * Copyright (c) 2015 Andrew Poelstra * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_MODULE_ECDH_MAIN_H +#define SECP256K1_MODULE_ECDH_MAIN_H + +#include "include/secp256k1_ecdh.h" +#include "ecmult_const_impl.h" + +static int ecdh_hash_function_sha256(unsigned char *output, const unsigned char *x, const unsigned char *y, void *data) { + unsigned char version = (y[31] & 0x01) | 0x02; + secp256k1_sha256 sha; + (void)data; + + secp256k1_sha256_initialize(&sha); + secp256k1_sha256_write(&sha, &version, 1); + secp256k1_sha256_write(&sha, x, 32); + secp256k1_sha256_finalize(&sha, output); + + return 1; +} + +const secp256k1_ecdh_hash_function secp256k1_ecdh_hash_function_sha256 = ecdh_hash_function_sha256; +const secp256k1_ecdh_hash_function secp256k1_ecdh_hash_function_default = ecdh_hash_function_sha256; + +int secp256k1_ecdh(const secp256k1_context* ctx, unsigned char *output, const secp256k1_pubkey *point, const unsigned char *scalar, secp256k1_ecdh_hash_function hashfp, void *data) { + int ret = 0; + int overflow = 0; + secp256k1_gej res; + secp256k1_ge pt; + secp256k1_scalar s; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(output != NULL); + ARG_CHECK(point != NULL); + ARG_CHECK(scalar != NULL); + if (hashfp == NULL) { + hashfp = secp256k1_ecdh_hash_function_default; + } + + secp256k1_pubkey_load(ctx, &pt, point); + secp256k1_scalar_set_b32(&s, scalar, &overflow); + if (overflow || secp256k1_scalar_is_zero(&s)) { + ret = 0; + } else { + unsigned char x[32]; + unsigned char y[32]; + + secp256k1_ecmult_const(&res, &pt, &s, 256); + secp256k1_ge_set_gej(&pt, &res); + + /* Compute a hash of the point */ + secp256k1_fe_normalize(&pt.x); + secp256k1_fe_normalize(&pt.y); + secp256k1_fe_get_b32(x, &pt.x); + secp256k1_fe_get_b32(y, &pt.y); + + ret = hashfp(output, x, y, data); + } + + secp256k1_scalar_clear(&s); + return ret; +} + +#endif /* SECP256K1_MODULE_ECDH_MAIN_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/modules/ecdh/tests_impl.h b/node_modules/secp256k1/src/secp256k1/src/modules/ecdh/tests_impl.h new file mode 100644 index 0000000..fe26e8f --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/modules/ecdh/tests_impl.h @@ -0,0 +1,132 @@ +/********************************************************************** + * Copyright (c) 2015 Andrew Poelstra * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_MODULE_ECDH_TESTS_H +#define SECP256K1_MODULE_ECDH_TESTS_H + +int ecdh_hash_function_test_fail(unsigned char *output, const unsigned char *x, const unsigned char *y, void *data) { + (void)output; + (void)x; + (void)y; + (void)data; + return 0; +} + +int ecdh_hash_function_custom(unsigned char *output, const unsigned char *x, const unsigned char *y, void *data) { + (void)data; + /* Save x and y as uncompressed public key */ + output[0] = 0x04; + memcpy(output + 1, x, 32); + memcpy(output + 33, y, 32); + return 1; +} + +void test_ecdh_api(void) { + /* Setup context that just counts errors */ + secp256k1_context *tctx = secp256k1_context_create(SECP256K1_CONTEXT_SIGN); + secp256k1_pubkey point; + unsigned char res[32]; + unsigned char s_one[32] = { 0 }; + int32_t ecount = 0; + s_one[31] = 1; + + secp256k1_context_set_error_callback(tctx, counting_illegal_callback_fn, &ecount); + secp256k1_context_set_illegal_callback(tctx, counting_illegal_callback_fn, &ecount); + CHECK(secp256k1_ec_pubkey_create(tctx, &point, s_one) == 1); + + /* Check all NULLs are detected */ + CHECK(secp256k1_ecdh(tctx, res, &point, s_one, NULL, NULL) == 1); + CHECK(ecount == 0); + CHECK(secp256k1_ecdh(tctx, NULL, &point, s_one, NULL, NULL) == 0); + CHECK(ecount == 1); + CHECK(secp256k1_ecdh(tctx, res, NULL, s_one, NULL, NULL) == 0); + CHECK(ecount == 2); + CHECK(secp256k1_ecdh(tctx, res, &point, NULL, NULL, NULL) == 0); + CHECK(ecount == 3); + CHECK(secp256k1_ecdh(tctx, res, &point, s_one, NULL, NULL) == 1); + CHECK(ecount == 3); + + /* Cleanup */ + secp256k1_context_destroy(tctx); +} + +void test_ecdh_generator_basepoint(void) { + unsigned char s_one[32] = { 0 }; + secp256k1_pubkey point[2]; + int i; + + s_one[31] = 1; + /* Check against pubkey creation when the basepoint is the generator */ + for (i = 0; i < 100; ++i) { + secp256k1_sha256 sha; + unsigned char s_b32[32]; + unsigned char output_ecdh[65]; + unsigned char output_ser[32]; + unsigned char point_ser[65]; + size_t point_ser_len = sizeof(point_ser); + secp256k1_scalar s; + + random_scalar_order(&s); + secp256k1_scalar_get_b32(s_b32, &s); + + CHECK(secp256k1_ec_pubkey_create(ctx, &point[0], s_one) == 1); + CHECK(secp256k1_ec_pubkey_create(ctx, &point[1], s_b32) == 1); + + /* compute using ECDH function with custom hash function */ + CHECK(secp256k1_ecdh(ctx, output_ecdh, &point[0], s_b32, ecdh_hash_function_custom, NULL) == 1); + /* compute "explicitly" */ + CHECK(secp256k1_ec_pubkey_serialize(ctx, point_ser, &point_ser_len, &point[1], SECP256K1_EC_UNCOMPRESSED) == 1); + /* compare */ + CHECK(memcmp(output_ecdh, point_ser, 65) == 0); + + /* compute using ECDH function with default hash function */ + CHECK(secp256k1_ecdh(ctx, output_ecdh, &point[0], s_b32, NULL, NULL) == 1); + /* compute "explicitly" */ + CHECK(secp256k1_ec_pubkey_serialize(ctx, point_ser, &point_ser_len, &point[1], SECP256K1_EC_COMPRESSED) == 1); + secp256k1_sha256_initialize(&sha); + secp256k1_sha256_write(&sha, point_ser, point_ser_len); + secp256k1_sha256_finalize(&sha, output_ser); + /* compare */ + CHECK(memcmp(output_ecdh, output_ser, 32) == 0); + } +} + +void test_bad_scalar(void) { + unsigned char s_zero[32] = { 0 }; + unsigned char s_overflow[32] = { + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, + 0xba, 0xae, 0xdc, 0xe6, 0xaf, 0x48, 0xa0, 0x3b, + 0xbf, 0xd2, 0x5e, 0x8c, 0xd0, 0x36, 0x41, 0x41 + }; + unsigned char s_rand[32] = { 0 }; + unsigned char output[32]; + secp256k1_scalar rand; + secp256k1_pubkey point; + + /* Create random point */ + random_scalar_order(&rand); + secp256k1_scalar_get_b32(s_rand, &rand); + CHECK(secp256k1_ec_pubkey_create(ctx, &point, s_rand) == 1); + + /* Try to multiply it by bad values */ + CHECK(secp256k1_ecdh(ctx, output, &point, s_zero, NULL, NULL) == 0); + CHECK(secp256k1_ecdh(ctx, output, &point, s_overflow, NULL, NULL) == 0); + /* ...and a good one */ + s_overflow[31] -= 1; + CHECK(secp256k1_ecdh(ctx, output, &point, s_overflow, NULL, NULL) == 1); + + /* Hash function failure results in ecdh failure */ + CHECK(secp256k1_ecdh(ctx, output, &point, s_overflow, ecdh_hash_function_test_fail, NULL) == 0); +} + +void run_ecdh_tests(void) { + test_ecdh_api(); + test_ecdh_generator_basepoint(); + test_bad_scalar(); +} + +#endif /* SECP256K1_MODULE_ECDH_TESTS_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/modules/recovery/Makefile.am.include b/node_modules/secp256k1/src/secp256k1/src/modules/recovery/Makefile.am.include new file mode 100644 index 0000000..bf23c26 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/modules/recovery/Makefile.am.include @@ -0,0 +1,8 @@ +include_HEADERS += include/secp256k1_recovery.h +noinst_HEADERS += src/modules/recovery/main_impl.h +noinst_HEADERS += src/modules/recovery/tests_impl.h +if USE_BENCHMARK +noinst_PROGRAMS += bench_recover +bench_recover_SOURCES = src/bench_recover.c +bench_recover_LDADD = libsecp256k1.la $(SECP_LIBS) $(COMMON_LIB) +endif diff --git a/node_modules/secp256k1/src/secp256k1/src/modules/recovery/main_impl.h b/node_modules/secp256k1/src/secp256k1/src/modules/recovery/main_impl.h new file mode 100755 index 0000000..ed356e5 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/modules/recovery/main_impl.h @@ -0,0 +1,193 @@ +/********************************************************************** + * Copyright (c) 2013-2015 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_MODULE_RECOVERY_MAIN_H +#define SECP256K1_MODULE_RECOVERY_MAIN_H + +#include "include/secp256k1_recovery.h" + +static void secp256k1_ecdsa_recoverable_signature_load(const secp256k1_context* ctx, secp256k1_scalar* r, secp256k1_scalar* s, int* recid, const secp256k1_ecdsa_recoverable_signature* sig) { + (void)ctx; + if (sizeof(secp256k1_scalar) == 32) { + /* When the secp256k1_scalar type is exactly 32 byte, use its + * representation inside secp256k1_ecdsa_signature, as conversion is very fast. + * Note that secp256k1_ecdsa_signature_save must use the same representation. */ + memcpy(r, &sig->data[0], 32); + memcpy(s, &sig->data[32], 32); + } else { + secp256k1_scalar_set_b32(r, &sig->data[0], NULL); + secp256k1_scalar_set_b32(s, &sig->data[32], NULL); + } + *recid = sig->data[64]; +} + +static void secp256k1_ecdsa_recoverable_signature_save(secp256k1_ecdsa_recoverable_signature* sig, const secp256k1_scalar* r, const secp256k1_scalar* s, int recid) { + if (sizeof(secp256k1_scalar) == 32) { + memcpy(&sig->data[0], r, 32); + memcpy(&sig->data[32], s, 32); + } else { + secp256k1_scalar_get_b32(&sig->data[0], r); + secp256k1_scalar_get_b32(&sig->data[32], s); + } + sig->data[64] = recid; +} + +int secp256k1_ecdsa_recoverable_signature_parse_compact(const secp256k1_context* ctx, secp256k1_ecdsa_recoverable_signature* sig, const unsigned char *input64, int recid) { + secp256k1_scalar r, s; + int ret = 1; + int overflow = 0; + + (void)ctx; + ARG_CHECK(sig != NULL); + ARG_CHECK(input64 != NULL); + ARG_CHECK(recid >= 0 && recid <= 3); + + secp256k1_scalar_set_b32(&r, &input64[0], &overflow); + ret &= !overflow; + secp256k1_scalar_set_b32(&s, &input64[32], &overflow); + ret &= !overflow; + if (ret) { + secp256k1_ecdsa_recoverable_signature_save(sig, &r, &s, recid); + } else { + memset(sig, 0, sizeof(*sig)); + } + return ret; +} + +int secp256k1_ecdsa_recoverable_signature_serialize_compact(const secp256k1_context* ctx, unsigned char *output64, int *recid, const secp256k1_ecdsa_recoverable_signature* sig) { + secp256k1_scalar r, s; + + (void)ctx; + ARG_CHECK(output64 != NULL); + ARG_CHECK(sig != NULL); + ARG_CHECK(recid != NULL); + + secp256k1_ecdsa_recoverable_signature_load(ctx, &r, &s, recid, sig); + secp256k1_scalar_get_b32(&output64[0], &r); + secp256k1_scalar_get_b32(&output64[32], &s); + return 1; +} + +int secp256k1_ecdsa_recoverable_signature_convert(const secp256k1_context* ctx, secp256k1_ecdsa_signature* sig, const secp256k1_ecdsa_recoverable_signature* sigin) { + secp256k1_scalar r, s; + int recid; + + (void)ctx; + ARG_CHECK(sig != NULL); + ARG_CHECK(sigin != NULL); + + secp256k1_ecdsa_recoverable_signature_load(ctx, &r, &s, &recid, sigin); + secp256k1_ecdsa_signature_save(sig, &r, &s); + return 1; +} + +static int secp256k1_ecdsa_sig_recover(const secp256k1_ecmult_context *ctx, const secp256k1_scalar *sigr, const secp256k1_scalar* sigs, secp256k1_ge *pubkey, const secp256k1_scalar *message, int recid) { + unsigned char brx[32]; + secp256k1_fe fx; + secp256k1_ge x; + secp256k1_gej xj; + secp256k1_scalar rn, u1, u2; + secp256k1_gej qj; + int r; + + if (secp256k1_scalar_is_zero(sigr) || secp256k1_scalar_is_zero(sigs)) { + return 0; + } + + secp256k1_scalar_get_b32(brx, sigr); + r = secp256k1_fe_set_b32(&fx, brx); + (void)r; + VERIFY_CHECK(r); /* brx comes from a scalar, so is less than the order; certainly less than p */ + if (recid & 2) { + if (secp256k1_fe_cmp_var(&fx, &secp256k1_ecdsa_const_p_minus_order) >= 0) { + return 0; + } + secp256k1_fe_add(&fx, &secp256k1_ecdsa_const_order_as_fe); + } + if (!secp256k1_ge_set_xo_var(&x, &fx, recid & 1)) { + return 0; + } + secp256k1_gej_set_ge(&xj, &x); + secp256k1_scalar_inverse_var(&rn, sigr); + secp256k1_scalar_mul(&u1, &rn, message); + secp256k1_scalar_negate(&u1, &u1); + secp256k1_scalar_mul(&u2, &rn, sigs); + secp256k1_ecmult(ctx, &qj, &xj, &u2, &u1); + secp256k1_ge_set_gej_var(pubkey, &qj); + return !secp256k1_gej_is_infinity(&qj); +} + +int secp256k1_ecdsa_sign_recoverable(const secp256k1_context* ctx, secp256k1_ecdsa_recoverable_signature *signature, const unsigned char *msg32, const unsigned char *seckey, secp256k1_nonce_function noncefp, const void* noncedata) { + secp256k1_scalar r, s; + secp256k1_scalar sec, non, msg; + int recid; + int ret = 0; + int overflow = 0; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(secp256k1_ecmult_gen_context_is_built(&ctx->ecmult_gen_ctx)); + ARG_CHECK(msg32 != NULL); + ARG_CHECK(signature != NULL); + ARG_CHECK(seckey != NULL); + if (noncefp == NULL) { + noncefp = secp256k1_nonce_function_default; + } + + secp256k1_scalar_set_b32(&sec, seckey, &overflow); + /* Fail if the secret key is invalid. */ + if (!overflow && !secp256k1_scalar_is_zero(&sec)) { + unsigned char nonce32[32]; + unsigned int count = 0; + secp256k1_scalar_set_b32(&msg, msg32, NULL); + while (1) { + ret = noncefp(nonce32, msg32, seckey, NULL, (void*)noncedata, count); + if (!ret) { + break; + } + secp256k1_scalar_set_b32(&non, nonce32, &overflow); + if (!overflow && !secp256k1_scalar_is_zero(&non)) { + if (secp256k1_ecdsa_sig_sign(&ctx->ecmult_gen_ctx, &r, &s, &sec, &msg, &non, &recid)) { + break; + } + } + count++; + } + memset(nonce32, 0, 32); + secp256k1_scalar_clear(&msg); + secp256k1_scalar_clear(&non); + secp256k1_scalar_clear(&sec); + } + if (ret) { + secp256k1_ecdsa_recoverable_signature_save(signature, &r, &s, recid); + } else { + memset(signature, 0, sizeof(*signature)); + } + return ret; +} + +int secp256k1_ecdsa_recover(const secp256k1_context* ctx, secp256k1_pubkey *pubkey, const secp256k1_ecdsa_recoverable_signature *signature, const unsigned char *msg32) { + secp256k1_ge q; + secp256k1_scalar r, s; + secp256k1_scalar m; + int recid; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(secp256k1_ecmult_context_is_built(&ctx->ecmult_ctx)); + ARG_CHECK(msg32 != NULL); + ARG_CHECK(signature != NULL); + ARG_CHECK(pubkey != NULL); + + secp256k1_ecdsa_recoverable_signature_load(ctx, &r, &s, &recid, signature); + VERIFY_CHECK(recid >= 0 && recid < 4); /* should have been caught in parse_compact */ + secp256k1_scalar_set_b32(&m, msg32, NULL); + if (secp256k1_ecdsa_sig_recover(&ctx->ecmult_ctx, &r, &s, &q, &m, recid)) { + secp256k1_pubkey_save(pubkey, &q); + return 1; + } else { + memset(pubkey, 0, sizeof(*pubkey)); + return 0; + } +} + +#endif /* SECP256K1_MODULE_RECOVERY_MAIN_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/modules/recovery/tests_impl.h b/node_modules/secp256k1/src/secp256k1/src/modules/recovery/tests_impl.h new file mode 100644 index 0000000..38a533a --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/modules/recovery/tests_impl.h @@ -0,0 +1,393 @@ +/********************************************************************** + * Copyright (c) 2013-2015 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_MODULE_RECOVERY_TESTS_H +#define SECP256K1_MODULE_RECOVERY_TESTS_H + +static int recovery_test_nonce_function(unsigned char *nonce32, const unsigned char *msg32, const unsigned char *key32, const unsigned char *algo16, void *data, unsigned int counter) { + (void) msg32; + (void) key32; + (void) algo16; + (void) data; + + /* On the first run, return 0 to force a second run */ + if (counter == 0) { + memset(nonce32, 0, 32); + return 1; + } + /* On the second run, return an overflow to force a third run */ + if (counter == 1) { + memset(nonce32, 0xff, 32); + return 1; + } + /* On the next run, return a valid nonce, but flip a coin as to whether or not to fail signing. */ + memset(nonce32, 1, 32); + return secp256k1_rand_bits(1); +} + +void test_ecdsa_recovery_api(void) { + /* Setup contexts that just count errors */ + secp256k1_context *none = secp256k1_context_create(SECP256K1_CONTEXT_NONE); + secp256k1_context *sign = secp256k1_context_create(SECP256K1_CONTEXT_SIGN); + secp256k1_context *vrfy = secp256k1_context_create(SECP256K1_CONTEXT_VERIFY); + secp256k1_context *both = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY); + secp256k1_pubkey pubkey; + secp256k1_pubkey recpubkey; + secp256k1_ecdsa_signature normal_sig; + secp256k1_ecdsa_recoverable_signature recsig; + unsigned char privkey[32] = { 1 }; + unsigned char message[32] = { 2 }; + int32_t ecount = 0; + int recid = 0; + unsigned char sig[74]; + unsigned char zero_privkey[32] = { 0 }; + unsigned char over_privkey[32] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff }; + + secp256k1_context_set_error_callback(none, counting_illegal_callback_fn, &ecount); + secp256k1_context_set_error_callback(sign, counting_illegal_callback_fn, &ecount); + secp256k1_context_set_error_callback(vrfy, counting_illegal_callback_fn, &ecount); + secp256k1_context_set_error_callback(both, counting_illegal_callback_fn, &ecount); + secp256k1_context_set_illegal_callback(none, counting_illegal_callback_fn, &ecount); + secp256k1_context_set_illegal_callback(sign, counting_illegal_callback_fn, &ecount); + secp256k1_context_set_illegal_callback(vrfy, counting_illegal_callback_fn, &ecount); + secp256k1_context_set_illegal_callback(both, counting_illegal_callback_fn, &ecount); + + /* Construct and verify corresponding public key. */ + CHECK(secp256k1_ec_seckey_verify(ctx, privkey) == 1); + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey, privkey) == 1); + + /* Check bad contexts and NULLs for signing */ + ecount = 0; + CHECK(secp256k1_ecdsa_sign_recoverable(none, &recsig, message, privkey, NULL, NULL) == 0); + CHECK(ecount == 1); + CHECK(secp256k1_ecdsa_sign_recoverable(sign, &recsig, message, privkey, NULL, NULL) == 1); + CHECK(ecount == 1); + CHECK(secp256k1_ecdsa_sign_recoverable(vrfy, &recsig, message, privkey, NULL, NULL) == 0); + CHECK(ecount == 2); + CHECK(secp256k1_ecdsa_sign_recoverable(both, &recsig, message, privkey, NULL, NULL) == 1); + CHECK(ecount == 2); + CHECK(secp256k1_ecdsa_sign_recoverable(both, NULL, message, privkey, NULL, NULL) == 0); + CHECK(ecount == 3); + CHECK(secp256k1_ecdsa_sign_recoverable(both, &recsig, NULL, privkey, NULL, NULL) == 0); + CHECK(ecount == 4); + CHECK(secp256k1_ecdsa_sign_recoverable(both, &recsig, message, NULL, NULL, NULL) == 0); + CHECK(ecount == 5); + /* This will fail or succeed randomly, and in either case will not ARG_CHECK failure */ + secp256k1_ecdsa_sign_recoverable(both, &recsig, message, privkey, recovery_test_nonce_function, NULL); + CHECK(ecount == 5); + /* These will all fail, but not in ARG_CHECK way */ + CHECK(secp256k1_ecdsa_sign_recoverable(both, &recsig, message, zero_privkey, NULL, NULL) == 0); + CHECK(secp256k1_ecdsa_sign_recoverable(both, &recsig, message, over_privkey, NULL, NULL) == 0); + /* This one will succeed. */ + CHECK(secp256k1_ecdsa_sign_recoverable(both, &recsig, message, privkey, NULL, NULL) == 1); + CHECK(ecount == 5); + + /* Check signing with a goofy nonce function */ + + /* Check bad contexts and NULLs for recovery */ + ecount = 0; + CHECK(secp256k1_ecdsa_recover(none, &recpubkey, &recsig, message) == 0); + CHECK(ecount == 1); + CHECK(secp256k1_ecdsa_recover(sign, &recpubkey, &recsig, message) == 0); + CHECK(ecount == 2); + CHECK(secp256k1_ecdsa_recover(vrfy, &recpubkey, &recsig, message) == 1); + CHECK(ecount == 2); + CHECK(secp256k1_ecdsa_recover(both, &recpubkey, &recsig, message) == 1); + CHECK(ecount == 2); + CHECK(secp256k1_ecdsa_recover(both, NULL, &recsig, message) == 0); + CHECK(ecount == 3); + CHECK(secp256k1_ecdsa_recover(both, &recpubkey, NULL, message) == 0); + CHECK(ecount == 4); + CHECK(secp256k1_ecdsa_recover(both, &recpubkey, &recsig, NULL) == 0); + CHECK(ecount == 5); + + /* Check NULLs for conversion */ + CHECK(secp256k1_ecdsa_sign(both, &normal_sig, message, privkey, NULL, NULL) == 1); + ecount = 0; + CHECK(secp256k1_ecdsa_recoverable_signature_convert(both, NULL, &recsig) == 0); + CHECK(ecount == 1); + CHECK(secp256k1_ecdsa_recoverable_signature_convert(both, &normal_sig, NULL) == 0); + CHECK(ecount == 2); + CHECK(secp256k1_ecdsa_recoverable_signature_convert(both, &normal_sig, &recsig) == 1); + + /* Check NULLs for de/serialization */ + CHECK(secp256k1_ecdsa_sign_recoverable(both, &recsig, message, privkey, NULL, NULL) == 1); + ecount = 0; + CHECK(secp256k1_ecdsa_recoverable_signature_serialize_compact(both, NULL, &recid, &recsig) == 0); + CHECK(ecount == 1); + CHECK(secp256k1_ecdsa_recoverable_signature_serialize_compact(both, sig, NULL, &recsig) == 0); + CHECK(ecount == 2); + CHECK(secp256k1_ecdsa_recoverable_signature_serialize_compact(both, sig, &recid, NULL) == 0); + CHECK(ecount == 3); + CHECK(secp256k1_ecdsa_recoverable_signature_serialize_compact(both, sig, &recid, &recsig) == 1); + + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(both, NULL, sig, recid) == 0); + CHECK(ecount == 4); + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(both, &recsig, NULL, recid) == 0); + CHECK(ecount == 5); + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(both, &recsig, sig, -1) == 0); + CHECK(ecount == 6); + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(both, &recsig, sig, 5) == 0); + CHECK(ecount == 7); + /* overflow in signature will fail but not affect ecount */ + memcpy(sig, over_privkey, 32); + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(both, &recsig, sig, recid) == 0); + CHECK(ecount == 7); + + /* cleanup */ + secp256k1_context_destroy(none); + secp256k1_context_destroy(sign); + secp256k1_context_destroy(vrfy); + secp256k1_context_destroy(both); +} + +void test_ecdsa_recovery_end_to_end(void) { + unsigned char extra[32] = {0x00}; + unsigned char privkey[32]; + unsigned char message[32]; + secp256k1_ecdsa_signature signature[5]; + secp256k1_ecdsa_recoverable_signature rsignature[5]; + unsigned char sig[74]; + secp256k1_pubkey pubkey; + secp256k1_pubkey recpubkey; + int recid = 0; + + /* Generate a random key and message. */ + { + secp256k1_scalar msg, key; + random_scalar_order_test(&msg); + random_scalar_order_test(&key); + secp256k1_scalar_get_b32(privkey, &key); + secp256k1_scalar_get_b32(message, &msg); + } + + /* Construct and verify corresponding public key. */ + CHECK(secp256k1_ec_seckey_verify(ctx, privkey) == 1); + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey, privkey) == 1); + + /* Serialize/parse compact and verify/recover. */ + extra[0] = 0; + CHECK(secp256k1_ecdsa_sign_recoverable(ctx, &rsignature[0], message, privkey, NULL, NULL) == 1); + CHECK(secp256k1_ecdsa_sign(ctx, &signature[0], message, privkey, NULL, NULL) == 1); + CHECK(secp256k1_ecdsa_sign_recoverable(ctx, &rsignature[4], message, privkey, NULL, NULL) == 1); + CHECK(secp256k1_ecdsa_sign_recoverable(ctx, &rsignature[1], message, privkey, NULL, extra) == 1); + extra[31] = 1; + CHECK(secp256k1_ecdsa_sign_recoverable(ctx, &rsignature[2], message, privkey, NULL, extra) == 1); + extra[31] = 0; + extra[0] = 1; + CHECK(secp256k1_ecdsa_sign_recoverable(ctx, &rsignature[3], message, privkey, NULL, extra) == 1); + CHECK(secp256k1_ecdsa_recoverable_signature_serialize_compact(ctx, sig, &recid, &rsignature[4]) == 1); + CHECK(secp256k1_ecdsa_recoverable_signature_convert(ctx, &signature[4], &rsignature[4]) == 1); + CHECK(memcmp(&signature[4], &signature[0], 64) == 0); + CHECK(secp256k1_ecdsa_verify(ctx, &signature[4], message, &pubkey) == 1); + memset(&rsignature[4], 0, sizeof(rsignature[4])); + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(ctx, &rsignature[4], sig, recid) == 1); + CHECK(secp256k1_ecdsa_recoverable_signature_convert(ctx, &signature[4], &rsignature[4]) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &signature[4], message, &pubkey) == 1); + /* Parse compact (with recovery id) and recover. */ + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(ctx, &rsignature[4], sig, recid) == 1); + CHECK(secp256k1_ecdsa_recover(ctx, &recpubkey, &rsignature[4], message) == 1); + CHECK(memcmp(&pubkey, &recpubkey, sizeof(pubkey)) == 0); + /* Serialize/destroy/parse signature and verify again. */ + CHECK(secp256k1_ecdsa_recoverable_signature_serialize_compact(ctx, sig, &recid, &rsignature[4]) == 1); + sig[secp256k1_rand_bits(6)] += 1 + secp256k1_rand_int(255); + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(ctx, &rsignature[4], sig, recid) == 1); + CHECK(secp256k1_ecdsa_recoverable_signature_convert(ctx, &signature[4], &rsignature[4]) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &signature[4], message, &pubkey) == 0); + /* Recover again */ + CHECK(secp256k1_ecdsa_recover(ctx, &recpubkey, &rsignature[4], message) == 0 || + memcmp(&pubkey, &recpubkey, sizeof(pubkey)) != 0); +} + +/* Tests several edge cases. */ +void test_ecdsa_recovery_edge_cases(void) { + const unsigned char msg32[32] = { + 'T', 'h', 'i', 's', ' ', 'i', 's', ' ', + 'a', ' ', 'v', 'e', 'r', 'y', ' ', 's', + 'e', 'c', 'r', 'e', 't', ' ', 'm', 'e', + 's', 's', 'a', 'g', 'e', '.', '.', '.' + }; + const unsigned char sig64[64] = { + /* Generated by signing the above message with nonce 'This is the nonce we will use...' + * and secret key 0 (which is not valid), resulting in recid 1. */ + 0x67, 0xCB, 0x28, 0x5F, 0x9C, 0xD1, 0x94, 0xE8, + 0x40, 0xD6, 0x29, 0x39, 0x7A, 0xF5, 0x56, 0x96, + 0x62, 0xFD, 0xE4, 0x46, 0x49, 0x99, 0x59, 0x63, + 0x17, 0x9A, 0x7D, 0xD1, 0x7B, 0xD2, 0x35, 0x32, + 0x4B, 0x1B, 0x7D, 0xF3, 0x4C, 0xE1, 0xF6, 0x8E, + 0x69, 0x4F, 0xF6, 0xF1, 0x1A, 0xC7, 0x51, 0xDD, + 0x7D, 0xD7, 0x3E, 0x38, 0x7E, 0xE4, 0xFC, 0x86, + 0x6E, 0x1B, 0xE8, 0xEC, 0xC7, 0xDD, 0x95, 0x57 + }; + secp256k1_pubkey pubkey; + /* signature (r,s) = (4,4), which can be recovered with all 4 recids. */ + const unsigned char sigb64[64] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x04, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x04, + }; + secp256k1_pubkey pubkeyb; + secp256k1_ecdsa_recoverable_signature rsig; + secp256k1_ecdsa_signature sig; + int recid; + + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(ctx, &rsig, sig64, 0)); + CHECK(!secp256k1_ecdsa_recover(ctx, &pubkey, &rsig, msg32)); + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(ctx, &rsig, sig64, 1)); + CHECK(secp256k1_ecdsa_recover(ctx, &pubkey, &rsig, msg32)); + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(ctx, &rsig, sig64, 2)); + CHECK(!secp256k1_ecdsa_recover(ctx, &pubkey, &rsig, msg32)); + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(ctx, &rsig, sig64, 3)); + CHECK(!secp256k1_ecdsa_recover(ctx, &pubkey, &rsig, msg32)); + + for (recid = 0; recid < 4; recid++) { + int i; + int recid2; + /* (4,4) encoded in DER. */ + unsigned char sigbder[8] = {0x30, 0x06, 0x02, 0x01, 0x04, 0x02, 0x01, 0x04}; + unsigned char sigcder_zr[7] = {0x30, 0x05, 0x02, 0x00, 0x02, 0x01, 0x01}; + unsigned char sigcder_zs[7] = {0x30, 0x05, 0x02, 0x01, 0x01, 0x02, 0x00}; + unsigned char sigbderalt1[39] = { + 0x30, 0x25, 0x02, 0x20, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x04, 0x02, 0x01, 0x04, + }; + unsigned char sigbderalt2[39] = { + 0x30, 0x25, 0x02, 0x01, 0x04, 0x02, 0x20, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x04, + }; + unsigned char sigbderalt3[40] = { + 0x30, 0x26, 0x02, 0x21, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x04, 0x02, 0x01, 0x04, + }; + unsigned char sigbderalt4[40] = { + 0x30, 0x26, 0x02, 0x01, 0x04, 0x02, 0x21, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x04, + }; + /* (order + r,4) encoded in DER. */ + unsigned char sigbderlong[40] = { + 0x30, 0x26, 0x02, 0x21, 0x00, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFE, 0xBA, 0xAE, 0xDC, + 0xE6, 0xAF, 0x48, 0xA0, 0x3B, 0xBF, 0xD2, 0x5E, + 0x8C, 0xD0, 0x36, 0x41, 0x45, 0x02, 0x01, 0x04 + }; + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(ctx, &rsig, sigb64, recid) == 1); + CHECK(secp256k1_ecdsa_recover(ctx, &pubkeyb, &rsig, msg32) == 1); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigbder, sizeof(sigbder)) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &sig, msg32, &pubkeyb) == 1); + for (recid2 = 0; recid2 < 4; recid2++) { + secp256k1_pubkey pubkey2b; + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(ctx, &rsig, sigb64, recid2) == 1); + CHECK(secp256k1_ecdsa_recover(ctx, &pubkey2b, &rsig, msg32) == 1); + /* Verifying with (order + r,4) should always fail. */ + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigbderlong, sizeof(sigbderlong)) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &sig, msg32, &pubkeyb) == 0); + } + /* DER parsing tests. */ + /* Zero length r/s. */ + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigcder_zr, sizeof(sigcder_zr)) == 0); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigcder_zs, sizeof(sigcder_zs)) == 0); + /* Leading zeros. */ + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigbderalt1, sizeof(sigbderalt1)) == 0); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigbderalt2, sizeof(sigbderalt2)) == 0); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigbderalt3, sizeof(sigbderalt3)) == 0); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigbderalt4, sizeof(sigbderalt4)) == 0); + sigbderalt3[4] = 1; + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigbderalt3, sizeof(sigbderalt3)) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &sig, msg32, &pubkeyb) == 0); + sigbderalt4[7] = 1; + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigbderalt4, sizeof(sigbderalt4)) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &sig, msg32, &pubkeyb) == 0); + /* Damage signature. */ + sigbder[7]++; + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigbder, sizeof(sigbder)) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &sig, msg32, &pubkeyb) == 0); + sigbder[7]--; + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigbder, 6) == 0); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigbder, sizeof(sigbder) - 1) == 0); + for(i = 0; i < 8; i++) { + int c; + unsigned char orig = sigbder[i]; + /*Try every single-byte change.*/ + for (c = 0; c < 256; c++) { + if (c == orig ) { + continue; + } + sigbder[i] = c; + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigbder, sizeof(sigbder)) == 0 || secp256k1_ecdsa_verify(ctx, &sig, msg32, &pubkeyb) == 0); + } + sigbder[i] = orig; + } + } + + /* Test r/s equal to zero */ + { + /* (1,1) encoded in DER. */ + unsigned char sigcder[8] = {0x30, 0x06, 0x02, 0x01, 0x01, 0x02, 0x01, 0x01}; + unsigned char sigc64[64] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + }; + secp256k1_pubkey pubkeyc; + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(ctx, &rsig, sigc64, 0) == 1); + CHECK(secp256k1_ecdsa_recover(ctx, &pubkeyc, &rsig, msg32) == 1); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigcder, sizeof(sigcder)) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &sig, msg32, &pubkeyc) == 1); + sigcder[4] = 0; + sigc64[31] = 0; + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(ctx, &rsig, sigc64, 0) == 1); + CHECK(secp256k1_ecdsa_recover(ctx, &pubkeyb, &rsig, msg32) == 0); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigcder, sizeof(sigcder)) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &sig, msg32, &pubkeyc) == 0); + sigcder[4] = 1; + sigcder[7] = 0; + sigc64[31] = 1; + sigc64[63] = 0; + CHECK(secp256k1_ecdsa_recoverable_signature_parse_compact(ctx, &rsig, sigc64, 0) == 1); + CHECK(secp256k1_ecdsa_recover(ctx, &pubkeyb, &rsig, msg32) == 0); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, sigcder, sizeof(sigcder)) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &sig, msg32, &pubkeyc) == 0); + } +} + +void run_recovery_tests(void) { + int i; + for (i = 0; i < count; i++) { + test_ecdsa_recovery_api(); + } + for (i = 0; i < 64*count; i++) { + test_ecdsa_recovery_end_to_end(); + } + test_ecdsa_recovery_edge_cases(); +} + +#endif /* SECP256K1_MODULE_RECOVERY_TESTS_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/num.h b/node_modules/secp256k1/src/secp256k1/src/num.h new file mode 100644 index 0000000..49f2dd7 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/num.h @@ -0,0 +1,74 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_NUM_H +#define SECP256K1_NUM_H + +#ifndef USE_NUM_NONE + +#if defined HAVE_CONFIG_H +#include "libsecp256k1-config.h" +#endif + +#if defined(USE_NUM_GMP) +#include "num_gmp.h" +#else +#error "Please select num implementation" +#endif + +/** Copy a number. */ +static void secp256k1_num_copy(secp256k1_num *r, const secp256k1_num *a); + +/** Convert a number's absolute value to a binary big-endian string. + * There must be enough place. */ +static void secp256k1_num_get_bin(unsigned char *r, unsigned int rlen, const secp256k1_num *a); + +/** Set a number to the value of a binary big-endian string. */ +static void secp256k1_num_set_bin(secp256k1_num *r, const unsigned char *a, unsigned int alen); + +/** Compute a modular inverse. The input must be less than the modulus. */ +static void secp256k1_num_mod_inverse(secp256k1_num *r, const secp256k1_num *a, const secp256k1_num *m); + +/** Compute the jacobi symbol (a|b). b must be positive and odd. */ +static int secp256k1_num_jacobi(const secp256k1_num *a, const secp256k1_num *b); + +/** Compare the absolute value of two numbers. */ +static int secp256k1_num_cmp(const secp256k1_num *a, const secp256k1_num *b); + +/** Test whether two number are equal (including sign). */ +static int secp256k1_num_eq(const secp256k1_num *a, const secp256k1_num *b); + +/** Add two (signed) numbers. */ +static void secp256k1_num_add(secp256k1_num *r, const secp256k1_num *a, const secp256k1_num *b); + +/** Subtract two (signed) numbers. */ +static void secp256k1_num_sub(secp256k1_num *r, const secp256k1_num *a, const secp256k1_num *b); + +/** Multiply two (signed) numbers. */ +static void secp256k1_num_mul(secp256k1_num *r, const secp256k1_num *a, const secp256k1_num *b); + +/** Replace a number by its remainder modulo m. M's sign is ignored. The result is a number between 0 and m-1, + even if r was negative. */ +static void secp256k1_num_mod(secp256k1_num *r, const secp256k1_num *m); + +/** Right-shift the passed number by bits bits. */ +static void secp256k1_num_shift(secp256k1_num *r, int bits); + +/** Check whether a number is zero. */ +static int secp256k1_num_is_zero(const secp256k1_num *a); + +/** Check whether a number is one. */ +static int secp256k1_num_is_one(const secp256k1_num *a); + +/** Check whether a number is strictly negative. */ +static int secp256k1_num_is_neg(const secp256k1_num *a); + +/** Change a number's sign. */ +static void secp256k1_num_negate(secp256k1_num *r); + +#endif + +#endif /* SECP256K1_NUM_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/num_gmp.h b/node_modules/secp256k1/src/secp256k1/src/num_gmp.h new file mode 100644 index 0000000..3619844 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/num_gmp.h @@ -0,0 +1,20 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_NUM_REPR_H +#define SECP256K1_NUM_REPR_H + +#include + +#define NUM_LIMBS ((256+GMP_NUMB_BITS-1)/GMP_NUMB_BITS) + +typedef struct { + mp_limb_t data[2*NUM_LIMBS]; + int neg; + int limbs; +} secp256k1_num; + +#endif /* SECP256K1_NUM_REPR_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/num_gmp_impl.h b/node_modules/secp256k1/src/secp256k1/src/num_gmp_impl.h new file mode 100644 index 0000000..0ae2a8b --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/num_gmp_impl.h @@ -0,0 +1,288 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_NUM_REPR_IMPL_H +#define SECP256K1_NUM_REPR_IMPL_H + +#include +#include +#include + +#include "util.h" +#include "num.h" + +#ifdef VERIFY +static void secp256k1_num_sanity(const secp256k1_num *a) { + VERIFY_CHECK(a->limbs == 1 || (a->limbs > 1 && a->data[a->limbs-1] != 0)); +} +#else +#define secp256k1_num_sanity(a) do { } while(0) +#endif + +static void secp256k1_num_copy(secp256k1_num *r, const secp256k1_num *a) { + *r = *a; +} + +static void secp256k1_num_get_bin(unsigned char *r, unsigned int rlen, const secp256k1_num *a) { + unsigned char tmp[65]; + int len = 0; + int shift = 0; + if (a->limbs>1 || a->data[0] != 0) { + len = mpn_get_str(tmp, 256, (mp_limb_t*)a->data, a->limbs); + } + while (shift < len && tmp[shift] == 0) shift++; + VERIFY_CHECK(len-shift <= (int)rlen); + memset(r, 0, rlen - len + shift); + if (len > shift) { + memcpy(r + rlen - len + shift, tmp + shift, len - shift); + } + memset(tmp, 0, sizeof(tmp)); +} + +static void secp256k1_num_set_bin(secp256k1_num *r, const unsigned char *a, unsigned int alen) { + int len; + VERIFY_CHECK(alen > 0); + VERIFY_CHECK(alen <= 64); + len = mpn_set_str(r->data, a, alen, 256); + if (len == 0) { + r->data[0] = 0; + len = 1; + } + VERIFY_CHECK(len <= NUM_LIMBS*2); + r->limbs = len; + r->neg = 0; + while (r->limbs > 1 && r->data[r->limbs-1]==0) { + r->limbs--; + } +} + +static void secp256k1_num_add_abs(secp256k1_num *r, const secp256k1_num *a, const secp256k1_num *b) { + mp_limb_t c = mpn_add(r->data, a->data, a->limbs, b->data, b->limbs); + r->limbs = a->limbs; + if (c != 0) { + VERIFY_CHECK(r->limbs < 2*NUM_LIMBS); + r->data[r->limbs++] = c; + } +} + +static void secp256k1_num_sub_abs(secp256k1_num *r, const secp256k1_num *a, const secp256k1_num *b) { + mp_limb_t c = mpn_sub(r->data, a->data, a->limbs, b->data, b->limbs); + (void)c; + VERIFY_CHECK(c == 0); + r->limbs = a->limbs; + while (r->limbs > 1 && r->data[r->limbs-1]==0) { + r->limbs--; + } +} + +static void secp256k1_num_mod(secp256k1_num *r, const secp256k1_num *m) { + secp256k1_num_sanity(r); + secp256k1_num_sanity(m); + + if (r->limbs >= m->limbs) { + mp_limb_t t[2*NUM_LIMBS]; + mpn_tdiv_qr(t, r->data, 0, r->data, r->limbs, m->data, m->limbs); + memset(t, 0, sizeof(t)); + r->limbs = m->limbs; + while (r->limbs > 1 && r->data[r->limbs-1]==0) { + r->limbs--; + } + } + + if (r->neg && (r->limbs > 1 || r->data[0] != 0)) { + secp256k1_num_sub_abs(r, m, r); + r->neg = 0; + } +} + +static void secp256k1_num_mod_inverse(secp256k1_num *r, const secp256k1_num *a, const secp256k1_num *m) { + int i; + mp_limb_t g[NUM_LIMBS+1]; + mp_limb_t u[NUM_LIMBS+1]; + mp_limb_t v[NUM_LIMBS+1]; + mp_size_t sn; + mp_size_t gn; + secp256k1_num_sanity(a); + secp256k1_num_sanity(m); + + /** mpn_gcdext computes: (G,S) = gcdext(U,V), where + * * G = gcd(U,V) + * * G = U*S + V*T + * * U has equal or more limbs than V, and V has no padding + * If we set U to be (a padded version of) a, and V = m: + * G = a*S + m*T + * G = a*S mod m + * Assuming G=1: + * S = 1/a mod m + */ + VERIFY_CHECK(m->limbs <= NUM_LIMBS); + VERIFY_CHECK(m->data[m->limbs-1] != 0); + for (i = 0; i < m->limbs; i++) { + u[i] = (i < a->limbs) ? a->data[i] : 0; + v[i] = m->data[i]; + } + sn = NUM_LIMBS+1; + gn = mpn_gcdext(g, r->data, &sn, u, m->limbs, v, m->limbs); + (void)gn; + VERIFY_CHECK(gn == 1); + VERIFY_CHECK(g[0] == 1); + r->neg = a->neg ^ m->neg; + if (sn < 0) { + mpn_sub(r->data, m->data, m->limbs, r->data, -sn); + r->limbs = m->limbs; + while (r->limbs > 1 && r->data[r->limbs-1]==0) { + r->limbs--; + } + } else { + r->limbs = sn; + } + memset(g, 0, sizeof(g)); + memset(u, 0, sizeof(u)); + memset(v, 0, sizeof(v)); +} + +static int secp256k1_num_jacobi(const secp256k1_num *a, const secp256k1_num *b) { + int ret; + mpz_t ga, gb; + secp256k1_num_sanity(a); + secp256k1_num_sanity(b); + VERIFY_CHECK(!b->neg && (b->limbs > 0) && (b->data[0] & 1)); + + mpz_inits(ga, gb, NULL); + + mpz_import(gb, b->limbs, -1, sizeof(mp_limb_t), 0, 0, b->data); + mpz_import(ga, a->limbs, -1, sizeof(mp_limb_t), 0, 0, a->data); + if (a->neg) { + mpz_neg(ga, ga); + } + + ret = mpz_jacobi(ga, gb); + + mpz_clears(ga, gb, NULL); + + return ret; +} + +static int secp256k1_num_is_one(const secp256k1_num *a) { + return (a->limbs == 1 && a->data[0] == 1); +} + +static int secp256k1_num_is_zero(const secp256k1_num *a) { + return (a->limbs == 1 && a->data[0] == 0); +} + +static int secp256k1_num_is_neg(const secp256k1_num *a) { + return (a->limbs > 1 || a->data[0] != 0) && a->neg; +} + +static int secp256k1_num_cmp(const secp256k1_num *a, const secp256k1_num *b) { + if (a->limbs > b->limbs) { + return 1; + } + if (a->limbs < b->limbs) { + return -1; + } + return mpn_cmp(a->data, b->data, a->limbs); +} + +static int secp256k1_num_eq(const secp256k1_num *a, const secp256k1_num *b) { + if (a->limbs > b->limbs) { + return 0; + } + if (a->limbs < b->limbs) { + return 0; + } + if ((a->neg && !secp256k1_num_is_zero(a)) != (b->neg && !secp256k1_num_is_zero(b))) { + return 0; + } + return mpn_cmp(a->data, b->data, a->limbs) == 0; +} + +static void secp256k1_num_subadd(secp256k1_num *r, const secp256k1_num *a, const secp256k1_num *b, int bneg) { + if (!(b->neg ^ bneg ^ a->neg)) { /* a and b have the same sign */ + r->neg = a->neg; + if (a->limbs >= b->limbs) { + secp256k1_num_add_abs(r, a, b); + } else { + secp256k1_num_add_abs(r, b, a); + } + } else { + if (secp256k1_num_cmp(a, b) > 0) { + r->neg = a->neg; + secp256k1_num_sub_abs(r, a, b); + } else { + r->neg = b->neg ^ bneg; + secp256k1_num_sub_abs(r, b, a); + } + } +} + +static void secp256k1_num_add(secp256k1_num *r, const secp256k1_num *a, const secp256k1_num *b) { + secp256k1_num_sanity(a); + secp256k1_num_sanity(b); + secp256k1_num_subadd(r, a, b, 0); +} + +static void secp256k1_num_sub(secp256k1_num *r, const secp256k1_num *a, const secp256k1_num *b) { + secp256k1_num_sanity(a); + secp256k1_num_sanity(b); + secp256k1_num_subadd(r, a, b, 1); +} + +static void secp256k1_num_mul(secp256k1_num *r, const secp256k1_num *a, const secp256k1_num *b) { + mp_limb_t tmp[2*NUM_LIMBS+1]; + secp256k1_num_sanity(a); + secp256k1_num_sanity(b); + + VERIFY_CHECK(a->limbs + b->limbs <= 2*NUM_LIMBS+1); + if ((a->limbs==1 && a->data[0]==0) || (b->limbs==1 && b->data[0]==0)) { + r->limbs = 1; + r->neg = 0; + r->data[0] = 0; + return; + } + if (a->limbs >= b->limbs) { + mpn_mul(tmp, a->data, a->limbs, b->data, b->limbs); + } else { + mpn_mul(tmp, b->data, b->limbs, a->data, a->limbs); + } + r->limbs = a->limbs + b->limbs; + if (r->limbs > 1 && tmp[r->limbs - 1]==0) { + r->limbs--; + } + VERIFY_CHECK(r->limbs <= 2*NUM_LIMBS); + mpn_copyi(r->data, tmp, r->limbs); + r->neg = a->neg ^ b->neg; + memset(tmp, 0, sizeof(tmp)); +} + +static void secp256k1_num_shift(secp256k1_num *r, int bits) { + if (bits % GMP_NUMB_BITS) { + /* Shift within limbs. */ + mpn_rshift(r->data, r->data, r->limbs, bits % GMP_NUMB_BITS); + } + if (bits >= GMP_NUMB_BITS) { + int i; + /* Shift full limbs. */ + for (i = 0; i < r->limbs; i++) { + int index = i + (bits / GMP_NUMB_BITS); + if (index < r->limbs && index < 2*NUM_LIMBS) { + r->data[i] = r->data[index]; + } else { + r->data[i] = 0; + } + } + } + while (r->limbs>1 && r->data[r->limbs-1]==0) { + r->limbs--; + } +} + +static void secp256k1_num_negate(secp256k1_num *r) { + r->neg ^= 1; +} + +#endif /* SECP256K1_NUM_REPR_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/num_impl.h b/node_modules/secp256k1/src/secp256k1/src/num_impl.h new file mode 100644 index 0000000..c45193b --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/num_impl.h @@ -0,0 +1,24 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_NUM_IMPL_H +#define SECP256K1_NUM_IMPL_H + +#if defined HAVE_CONFIG_H +#include "libsecp256k1-config.h" +#endif + +#include "num.h" + +#if defined(USE_NUM_GMP) +#include "num_gmp_impl.h" +#elif defined(USE_NUM_NONE) +/* Nothing. */ +#else +#error "Please select num implementation" +#endif + +#endif /* SECP256K1_NUM_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/scalar.h b/node_modules/secp256k1/src/secp256k1/src/scalar.h new file mode 100644 index 0000000..59304cb --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/scalar.h @@ -0,0 +1,106 @@ +/********************************************************************** + * Copyright (c) 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_SCALAR_H +#define SECP256K1_SCALAR_H + +#include "num.h" + +#if defined HAVE_CONFIG_H +#include "libsecp256k1-config.h" +#endif + +#if defined(EXHAUSTIVE_TEST_ORDER) +#include "scalar_low.h" +#elif defined(USE_SCALAR_4X64) +#include "scalar_4x64.h" +#elif defined(USE_SCALAR_8X32) +#include "scalar_8x32.h" +#else +#error "Please select scalar implementation" +#endif + +/** Clear a scalar to prevent the leak of sensitive data. */ +static void secp256k1_scalar_clear(secp256k1_scalar *r); + +/** Access bits from a scalar. All requested bits must belong to the same 32-bit limb. */ +static unsigned int secp256k1_scalar_get_bits(const secp256k1_scalar *a, unsigned int offset, unsigned int count); + +/** Access bits from a scalar. Not constant time. */ +static unsigned int secp256k1_scalar_get_bits_var(const secp256k1_scalar *a, unsigned int offset, unsigned int count); + +/** Set a scalar from a big endian byte array. */ +static void secp256k1_scalar_set_b32(secp256k1_scalar *r, const unsigned char *bin, int *overflow); + +/** Set a scalar to an unsigned integer. */ +static void secp256k1_scalar_set_int(secp256k1_scalar *r, unsigned int v); + +/** Convert a scalar to a byte array. */ +static void secp256k1_scalar_get_b32(unsigned char *bin, const secp256k1_scalar* a); + +/** Add two scalars together (modulo the group order). Returns whether it overflowed. */ +static int secp256k1_scalar_add(secp256k1_scalar *r, const secp256k1_scalar *a, const secp256k1_scalar *b); + +/** Conditionally add a power of two to a scalar. The result is not allowed to overflow. */ +static void secp256k1_scalar_cadd_bit(secp256k1_scalar *r, unsigned int bit, int flag); + +/** Multiply two scalars (modulo the group order). */ +static void secp256k1_scalar_mul(secp256k1_scalar *r, const secp256k1_scalar *a, const secp256k1_scalar *b); + +/** Shift a scalar right by some amount strictly between 0 and 16, returning + * the low bits that were shifted off */ +static int secp256k1_scalar_shr_int(secp256k1_scalar *r, int n); + +/** Compute the square of a scalar (modulo the group order). */ +static void secp256k1_scalar_sqr(secp256k1_scalar *r, const secp256k1_scalar *a); + +/** Compute the inverse of a scalar (modulo the group order). */ +static void secp256k1_scalar_inverse(secp256k1_scalar *r, const secp256k1_scalar *a); + +/** Compute the inverse of a scalar (modulo the group order), without constant-time guarantee. */ +static void secp256k1_scalar_inverse_var(secp256k1_scalar *r, const secp256k1_scalar *a); + +/** Compute the complement of a scalar (modulo the group order). */ +static void secp256k1_scalar_negate(secp256k1_scalar *r, const secp256k1_scalar *a); + +/** Check whether a scalar equals zero. */ +static int secp256k1_scalar_is_zero(const secp256k1_scalar *a); + +/** Check whether a scalar equals one. */ +static int secp256k1_scalar_is_one(const secp256k1_scalar *a); + +/** Check whether a scalar, considered as an nonnegative integer, is even. */ +static int secp256k1_scalar_is_even(const secp256k1_scalar *a); + +/** Check whether a scalar is higher than the group order divided by 2. */ +static int secp256k1_scalar_is_high(const secp256k1_scalar *a); + +/** Conditionally negate a number, in constant time. + * Returns -1 if the number was negated, 1 otherwise */ +static int secp256k1_scalar_cond_negate(secp256k1_scalar *a, int flag); + +#ifndef USE_NUM_NONE +/** Convert a scalar to a number. */ +static void secp256k1_scalar_get_num(secp256k1_num *r, const secp256k1_scalar *a); + +/** Get the order of the group as a number. */ +static void secp256k1_scalar_order_get_num(secp256k1_num *r); +#endif + +/** Compare two scalars. */ +static int secp256k1_scalar_eq(const secp256k1_scalar *a, const secp256k1_scalar *b); + +#ifdef USE_ENDOMORPHISM +/** Find r1 and r2 such that r1+r2*2^128 = a. */ +static void secp256k1_scalar_split_128(secp256k1_scalar *r1, secp256k1_scalar *r2, const secp256k1_scalar *a); +/** Find r1 and r2 such that r1+r2*lambda = a, and r1 and r2 are maximum 128 bits long (see secp256k1_gej_mul_lambda). */ +static void secp256k1_scalar_split_lambda(secp256k1_scalar *r1, secp256k1_scalar *r2, const secp256k1_scalar *a); +#endif + +/** Multiply a and b (without taking the modulus!), divide by 2**shift, and round to the nearest integer. Shift must be at least 256. */ +static void secp256k1_scalar_mul_shift_var(secp256k1_scalar *r, const secp256k1_scalar *a, const secp256k1_scalar *b, unsigned int shift); + +#endif /* SECP256K1_SCALAR_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/scalar_4x64.h b/node_modules/secp256k1/src/secp256k1/src/scalar_4x64.h new file mode 100644 index 0000000..19c7495 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/scalar_4x64.h @@ -0,0 +1,19 @@ +/********************************************************************** + * Copyright (c) 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_SCALAR_REPR_H +#define SECP256K1_SCALAR_REPR_H + +#include + +/** A scalar modulo the group order of the secp256k1 curve. */ +typedef struct { + uint64_t d[4]; +} secp256k1_scalar; + +#define SECP256K1_SCALAR_CONST(d7, d6, d5, d4, d3, d2, d1, d0) {{((uint64_t)(d1)) << 32 | (d0), ((uint64_t)(d3)) << 32 | (d2), ((uint64_t)(d5)) << 32 | (d4), ((uint64_t)(d7)) << 32 | (d6)}} + +#endif /* SECP256K1_SCALAR_REPR_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/scalar_4x64_impl.h b/node_modules/secp256k1/src/secp256k1/src/scalar_4x64_impl.h new file mode 100644 index 0000000..d378335 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/scalar_4x64_impl.h @@ -0,0 +1,949 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_SCALAR_REPR_IMPL_H +#define SECP256K1_SCALAR_REPR_IMPL_H + +/* Limbs of the secp256k1 order. */ +#define SECP256K1_N_0 ((uint64_t)0xBFD25E8CD0364141ULL) +#define SECP256K1_N_1 ((uint64_t)0xBAAEDCE6AF48A03BULL) +#define SECP256K1_N_2 ((uint64_t)0xFFFFFFFFFFFFFFFEULL) +#define SECP256K1_N_3 ((uint64_t)0xFFFFFFFFFFFFFFFFULL) + +/* Limbs of 2^256 minus the secp256k1 order. */ +#define SECP256K1_N_C_0 (~SECP256K1_N_0 + 1) +#define SECP256K1_N_C_1 (~SECP256K1_N_1) +#define SECP256K1_N_C_2 (1) + +/* Limbs of half the secp256k1 order. */ +#define SECP256K1_N_H_0 ((uint64_t)0xDFE92F46681B20A0ULL) +#define SECP256K1_N_H_1 ((uint64_t)0x5D576E7357A4501DULL) +#define SECP256K1_N_H_2 ((uint64_t)0xFFFFFFFFFFFFFFFFULL) +#define SECP256K1_N_H_3 ((uint64_t)0x7FFFFFFFFFFFFFFFULL) + +SECP256K1_INLINE static void secp256k1_scalar_clear(secp256k1_scalar *r) { + r->d[0] = 0; + r->d[1] = 0; + r->d[2] = 0; + r->d[3] = 0; +} + +SECP256K1_INLINE static void secp256k1_scalar_set_int(secp256k1_scalar *r, unsigned int v) { + r->d[0] = v; + r->d[1] = 0; + r->d[2] = 0; + r->d[3] = 0; +} + +SECP256K1_INLINE static unsigned int secp256k1_scalar_get_bits(const secp256k1_scalar *a, unsigned int offset, unsigned int count) { + VERIFY_CHECK((offset + count - 1) >> 6 == offset >> 6); + return (a->d[offset >> 6] >> (offset & 0x3F)) & ((((uint64_t)1) << count) - 1); +} + +SECP256K1_INLINE static unsigned int secp256k1_scalar_get_bits_var(const secp256k1_scalar *a, unsigned int offset, unsigned int count) { + VERIFY_CHECK(count < 32); + VERIFY_CHECK(offset + count <= 256); + if ((offset + count - 1) >> 6 == offset >> 6) { + return secp256k1_scalar_get_bits(a, offset, count); + } else { + VERIFY_CHECK((offset >> 6) + 1 < 4); + return ((a->d[offset >> 6] >> (offset & 0x3F)) | (a->d[(offset >> 6) + 1] << (64 - (offset & 0x3F)))) & ((((uint64_t)1) << count) - 1); + } +} + +SECP256K1_INLINE static int secp256k1_scalar_check_overflow(const secp256k1_scalar *a) { + int yes = 0; + int no = 0; + no |= (a->d[3] < SECP256K1_N_3); /* No need for a > check. */ + no |= (a->d[2] < SECP256K1_N_2); + yes |= (a->d[2] > SECP256K1_N_2) & ~no; + no |= (a->d[1] < SECP256K1_N_1); + yes |= (a->d[1] > SECP256K1_N_1) & ~no; + yes |= (a->d[0] >= SECP256K1_N_0) & ~no; + return yes; +} + +SECP256K1_INLINE static int secp256k1_scalar_reduce(secp256k1_scalar *r, unsigned int overflow) { + uint128_t t; + VERIFY_CHECK(overflow <= 1); + t = (uint128_t)r->d[0] + overflow * SECP256K1_N_C_0; + r->d[0] = t & 0xFFFFFFFFFFFFFFFFULL; t >>= 64; + t += (uint128_t)r->d[1] + overflow * SECP256K1_N_C_1; + r->d[1] = t & 0xFFFFFFFFFFFFFFFFULL; t >>= 64; + t += (uint128_t)r->d[2] + overflow * SECP256K1_N_C_2; + r->d[2] = t & 0xFFFFFFFFFFFFFFFFULL; t >>= 64; + t += (uint64_t)r->d[3]; + r->d[3] = t & 0xFFFFFFFFFFFFFFFFULL; + return overflow; +} + +static int secp256k1_scalar_add(secp256k1_scalar *r, const secp256k1_scalar *a, const secp256k1_scalar *b) { + int overflow; + uint128_t t = (uint128_t)a->d[0] + b->d[0]; + r->d[0] = t & 0xFFFFFFFFFFFFFFFFULL; t >>= 64; + t += (uint128_t)a->d[1] + b->d[1]; + r->d[1] = t & 0xFFFFFFFFFFFFFFFFULL; t >>= 64; + t += (uint128_t)a->d[2] + b->d[2]; + r->d[2] = t & 0xFFFFFFFFFFFFFFFFULL; t >>= 64; + t += (uint128_t)a->d[3] + b->d[3]; + r->d[3] = t & 0xFFFFFFFFFFFFFFFFULL; t >>= 64; + overflow = t + secp256k1_scalar_check_overflow(r); + VERIFY_CHECK(overflow == 0 || overflow == 1); + secp256k1_scalar_reduce(r, overflow); + return overflow; +} + +static void secp256k1_scalar_cadd_bit(secp256k1_scalar *r, unsigned int bit, int flag) { + uint128_t t; + VERIFY_CHECK(bit < 256); + bit += ((uint32_t) flag - 1) & 0x100; /* forcing (bit >> 6) > 3 makes this a noop */ + t = (uint128_t)r->d[0] + (((uint64_t)((bit >> 6) == 0)) << (bit & 0x3F)); + r->d[0] = t & 0xFFFFFFFFFFFFFFFFULL; t >>= 64; + t += (uint128_t)r->d[1] + (((uint64_t)((bit >> 6) == 1)) << (bit & 0x3F)); + r->d[1] = t & 0xFFFFFFFFFFFFFFFFULL; t >>= 64; + t += (uint128_t)r->d[2] + (((uint64_t)((bit >> 6) == 2)) << (bit & 0x3F)); + r->d[2] = t & 0xFFFFFFFFFFFFFFFFULL; t >>= 64; + t += (uint128_t)r->d[3] + (((uint64_t)((bit >> 6) == 3)) << (bit & 0x3F)); + r->d[3] = t & 0xFFFFFFFFFFFFFFFFULL; +#ifdef VERIFY + VERIFY_CHECK((t >> 64) == 0); + VERIFY_CHECK(secp256k1_scalar_check_overflow(r) == 0); +#endif +} + +static void secp256k1_scalar_set_b32(secp256k1_scalar *r, const unsigned char *b32, int *overflow) { + int over; + r->d[0] = (uint64_t)b32[31] | (uint64_t)b32[30] << 8 | (uint64_t)b32[29] << 16 | (uint64_t)b32[28] << 24 | (uint64_t)b32[27] << 32 | (uint64_t)b32[26] << 40 | (uint64_t)b32[25] << 48 | (uint64_t)b32[24] << 56; + r->d[1] = (uint64_t)b32[23] | (uint64_t)b32[22] << 8 | (uint64_t)b32[21] << 16 | (uint64_t)b32[20] << 24 | (uint64_t)b32[19] << 32 | (uint64_t)b32[18] << 40 | (uint64_t)b32[17] << 48 | (uint64_t)b32[16] << 56; + r->d[2] = (uint64_t)b32[15] | (uint64_t)b32[14] << 8 | (uint64_t)b32[13] << 16 | (uint64_t)b32[12] << 24 | (uint64_t)b32[11] << 32 | (uint64_t)b32[10] << 40 | (uint64_t)b32[9] << 48 | (uint64_t)b32[8] << 56; + r->d[3] = (uint64_t)b32[7] | (uint64_t)b32[6] << 8 | (uint64_t)b32[5] << 16 | (uint64_t)b32[4] << 24 | (uint64_t)b32[3] << 32 | (uint64_t)b32[2] << 40 | (uint64_t)b32[1] << 48 | (uint64_t)b32[0] << 56; + over = secp256k1_scalar_reduce(r, secp256k1_scalar_check_overflow(r)); + if (overflow) { + *overflow = over; + } +} + +static void secp256k1_scalar_get_b32(unsigned char *bin, const secp256k1_scalar* a) { + bin[0] = a->d[3] >> 56; bin[1] = a->d[3] >> 48; bin[2] = a->d[3] >> 40; bin[3] = a->d[3] >> 32; bin[4] = a->d[3] >> 24; bin[5] = a->d[3] >> 16; bin[6] = a->d[3] >> 8; bin[7] = a->d[3]; + bin[8] = a->d[2] >> 56; bin[9] = a->d[2] >> 48; bin[10] = a->d[2] >> 40; bin[11] = a->d[2] >> 32; bin[12] = a->d[2] >> 24; bin[13] = a->d[2] >> 16; bin[14] = a->d[2] >> 8; bin[15] = a->d[2]; + bin[16] = a->d[1] >> 56; bin[17] = a->d[1] >> 48; bin[18] = a->d[1] >> 40; bin[19] = a->d[1] >> 32; bin[20] = a->d[1] >> 24; bin[21] = a->d[1] >> 16; bin[22] = a->d[1] >> 8; bin[23] = a->d[1]; + bin[24] = a->d[0] >> 56; bin[25] = a->d[0] >> 48; bin[26] = a->d[0] >> 40; bin[27] = a->d[0] >> 32; bin[28] = a->d[0] >> 24; bin[29] = a->d[0] >> 16; bin[30] = a->d[0] >> 8; bin[31] = a->d[0]; +} + +SECP256K1_INLINE static int secp256k1_scalar_is_zero(const secp256k1_scalar *a) { + return (a->d[0] | a->d[1] | a->d[2] | a->d[3]) == 0; +} + +static void secp256k1_scalar_negate(secp256k1_scalar *r, const secp256k1_scalar *a) { + uint64_t nonzero = 0xFFFFFFFFFFFFFFFFULL * (secp256k1_scalar_is_zero(a) == 0); + uint128_t t = (uint128_t)(~a->d[0]) + SECP256K1_N_0 + 1; + r->d[0] = t & nonzero; t >>= 64; + t += (uint128_t)(~a->d[1]) + SECP256K1_N_1; + r->d[1] = t & nonzero; t >>= 64; + t += (uint128_t)(~a->d[2]) + SECP256K1_N_2; + r->d[2] = t & nonzero; t >>= 64; + t += (uint128_t)(~a->d[3]) + SECP256K1_N_3; + r->d[3] = t & nonzero; +} + +SECP256K1_INLINE static int secp256k1_scalar_is_one(const secp256k1_scalar *a) { + return ((a->d[0] ^ 1) | a->d[1] | a->d[2] | a->d[3]) == 0; +} + +static int secp256k1_scalar_is_high(const secp256k1_scalar *a) { + int yes = 0; + int no = 0; + no |= (a->d[3] < SECP256K1_N_H_3); + yes |= (a->d[3] > SECP256K1_N_H_3) & ~no; + no |= (a->d[2] < SECP256K1_N_H_2) & ~yes; /* No need for a > check. */ + no |= (a->d[1] < SECP256K1_N_H_1) & ~yes; + yes |= (a->d[1] > SECP256K1_N_H_1) & ~no; + yes |= (a->d[0] > SECP256K1_N_H_0) & ~no; + return yes; +} + +static int secp256k1_scalar_cond_negate(secp256k1_scalar *r, int flag) { + /* If we are flag = 0, mask = 00...00 and this is a no-op; + * if we are flag = 1, mask = 11...11 and this is identical to secp256k1_scalar_negate */ + uint64_t mask = !flag - 1; + uint64_t nonzero = (secp256k1_scalar_is_zero(r) != 0) - 1; + uint128_t t = (uint128_t)(r->d[0] ^ mask) + ((SECP256K1_N_0 + 1) & mask); + r->d[0] = t & nonzero; t >>= 64; + t += (uint128_t)(r->d[1] ^ mask) + (SECP256K1_N_1 & mask); + r->d[1] = t & nonzero; t >>= 64; + t += (uint128_t)(r->d[2] ^ mask) + (SECP256K1_N_2 & mask); + r->d[2] = t & nonzero; t >>= 64; + t += (uint128_t)(r->d[3] ^ mask) + (SECP256K1_N_3 & mask); + r->d[3] = t & nonzero; + return 2 * (mask == 0) - 1; +} + +/* Inspired by the macros in OpenSSL's crypto/bn/asm/x86_64-gcc.c. */ + +/** Add a*b to the number defined by (c0,c1,c2). c2 must never overflow. */ +#define muladd(a,b) { \ + uint64_t tl, th; \ + { \ + uint128_t t = (uint128_t)a * b; \ + th = t >> 64; /* at most 0xFFFFFFFFFFFFFFFE */ \ + tl = t; \ + } \ + c0 += tl; /* overflow is handled on the next line */ \ + th += (c0 < tl) ? 1 : 0; /* at most 0xFFFFFFFFFFFFFFFF */ \ + c1 += th; /* overflow is handled on the next line */ \ + c2 += (c1 < th) ? 1 : 0; /* never overflows by contract (verified in the next line) */ \ + VERIFY_CHECK((c1 >= th) || (c2 != 0)); \ +} + +/** Add a*b to the number defined by (c0,c1). c1 must never overflow. */ +#define muladd_fast(a,b) { \ + uint64_t tl, th; \ + { \ + uint128_t t = (uint128_t)a * b; \ + th = t >> 64; /* at most 0xFFFFFFFFFFFFFFFE */ \ + tl = t; \ + } \ + c0 += tl; /* overflow is handled on the next line */ \ + th += (c0 < tl) ? 1 : 0; /* at most 0xFFFFFFFFFFFFFFFF */ \ + c1 += th; /* never overflows by contract (verified in the next line) */ \ + VERIFY_CHECK(c1 >= th); \ +} + +/** Add 2*a*b to the number defined by (c0,c1,c2). c2 must never overflow. */ +#define muladd2(a,b) { \ + uint64_t tl, th, th2, tl2; \ + { \ + uint128_t t = (uint128_t)a * b; \ + th = t >> 64; /* at most 0xFFFFFFFFFFFFFFFE */ \ + tl = t; \ + } \ + th2 = th + th; /* at most 0xFFFFFFFFFFFFFFFE (in case th was 0x7FFFFFFFFFFFFFFF) */ \ + c2 += (th2 < th) ? 1 : 0; /* never overflows by contract (verified the next line) */ \ + VERIFY_CHECK((th2 >= th) || (c2 != 0)); \ + tl2 = tl + tl; /* at most 0xFFFFFFFFFFFFFFFE (in case the lowest 63 bits of tl were 0x7FFFFFFFFFFFFFFF) */ \ + th2 += (tl2 < tl) ? 1 : 0; /* at most 0xFFFFFFFFFFFFFFFF */ \ + c0 += tl2; /* overflow is handled on the next line */ \ + th2 += (c0 < tl2) ? 1 : 0; /* second overflow is handled on the next line */ \ + c2 += (c0 < tl2) & (th2 == 0); /* never overflows by contract (verified the next line) */ \ + VERIFY_CHECK((c0 >= tl2) || (th2 != 0) || (c2 != 0)); \ + c1 += th2; /* overflow is handled on the next line */ \ + c2 += (c1 < th2) ? 1 : 0; /* never overflows by contract (verified the next line) */ \ + VERIFY_CHECK((c1 >= th2) || (c2 != 0)); \ +} + +/** Add a to the number defined by (c0,c1,c2). c2 must never overflow. */ +#define sumadd(a) { \ + unsigned int over; \ + c0 += (a); /* overflow is handled on the next line */ \ + over = (c0 < (a)) ? 1 : 0; \ + c1 += over; /* overflow is handled on the next line */ \ + c2 += (c1 < over) ? 1 : 0; /* never overflows by contract */ \ +} + +/** Add a to the number defined by (c0,c1). c1 must never overflow, c2 must be zero. */ +#define sumadd_fast(a) { \ + c0 += (a); /* overflow is handled on the next line */ \ + c1 += (c0 < (a)) ? 1 : 0; /* never overflows by contract (verified the next line) */ \ + VERIFY_CHECK((c1 != 0) | (c0 >= (a))); \ + VERIFY_CHECK(c2 == 0); \ +} + +/** Extract the lowest 64 bits of (c0,c1,c2) into n, and left shift the number 64 bits. */ +#define extract(n) { \ + (n) = c0; \ + c0 = c1; \ + c1 = c2; \ + c2 = 0; \ +} + +/** Extract the lowest 64 bits of (c0,c1,c2) into n, and left shift the number 64 bits. c2 is required to be zero. */ +#define extract_fast(n) { \ + (n) = c0; \ + c0 = c1; \ + c1 = 0; \ + VERIFY_CHECK(c2 == 0); \ +} + +static void secp256k1_scalar_reduce_512(secp256k1_scalar *r, const uint64_t *l) { +#ifdef USE_ASM_X86_64 + /* Reduce 512 bits into 385. */ + uint64_t m0, m1, m2, m3, m4, m5, m6; + uint64_t p0, p1, p2, p3, p4; + uint64_t c; + + __asm__ __volatile__( + /* Preload. */ + "movq 32(%%rsi), %%r11\n" + "movq 40(%%rsi), %%r12\n" + "movq 48(%%rsi), %%r13\n" + "movq 56(%%rsi), %%r14\n" + /* Initialize r8,r9,r10 */ + "movq 0(%%rsi), %%r8\n" + "xorq %%r9, %%r9\n" + "xorq %%r10, %%r10\n" + /* (r8,r9) += n0 * c0 */ + "movq %8, %%rax\n" + "mulq %%r11\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + /* extract m0 */ + "movq %%r8, %q0\n" + "xorq %%r8, %%r8\n" + /* (r9,r10) += l1 */ + "addq 8(%%rsi), %%r9\n" + "adcq $0, %%r10\n" + /* (r9,r10,r8) += n1 * c0 */ + "movq %8, %%rax\n" + "mulq %%r12\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* (r9,r10,r8) += n0 * c1 */ + "movq %9, %%rax\n" + "mulq %%r11\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* extract m1 */ + "movq %%r9, %q1\n" + "xorq %%r9, %%r9\n" + /* (r10,r8,r9) += l2 */ + "addq 16(%%rsi), %%r10\n" + "adcq $0, %%r8\n" + "adcq $0, %%r9\n" + /* (r10,r8,r9) += n2 * c0 */ + "movq %8, %%rax\n" + "mulq %%r13\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + "adcq $0, %%r9\n" + /* (r10,r8,r9) += n1 * c1 */ + "movq %9, %%rax\n" + "mulq %%r12\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + "adcq $0, %%r9\n" + /* (r10,r8,r9) += n0 */ + "addq %%r11, %%r10\n" + "adcq $0, %%r8\n" + "adcq $0, %%r9\n" + /* extract m2 */ + "movq %%r10, %q2\n" + "xorq %%r10, %%r10\n" + /* (r8,r9,r10) += l3 */ + "addq 24(%%rsi), %%r8\n" + "adcq $0, %%r9\n" + "adcq $0, %%r10\n" + /* (r8,r9,r10) += n3 * c0 */ + "movq %8, %%rax\n" + "mulq %%r14\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + "adcq $0, %%r10\n" + /* (r8,r9,r10) += n2 * c1 */ + "movq %9, %%rax\n" + "mulq %%r13\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + "adcq $0, %%r10\n" + /* (r8,r9,r10) += n1 */ + "addq %%r12, %%r8\n" + "adcq $0, %%r9\n" + "adcq $0, %%r10\n" + /* extract m3 */ + "movq %%r8, %q3\n" + "xorq %%r8, %%r8\n" + /* (r9,r10,r8) += n3 * c1 */ + "movq %9, %%rax\n" + "mulq %%r14\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* (r9,r10,r8) += n2 */ + "addq %%r13, %%r9\n" + "adcq $0, %%r10\n" + "adcq $0, %%r8\n" + /* extract m4 */ + "movq %%r9, %q4\n" + /* (r10,r8) += n3 */ + "addq %%r14, %%r10\n" + "adcq $0, %%r8\n" + /* extract m5 */ + "movq %%r10, %q5\n" + /* extract m6 */ + "movq %%r8, %q6\n" + : "=g"(m0), "=g"(m1), "=g"(m2), "=g"(m3), "=g"(m4), "=g"(m5), "=g"(m6) + : "S"(l), "i"(SECP256K1_N_C_0), "i"(SECP256K1_N_C_1) + : "rax", "rdx", "r8", "r9", "r10", "r11", "r12", "r13", "r14", "cc"); + + /* Reduce 385 bits into 258. */ + __asm__ __volatile__( + /* Preload */ + "movq %q9, %%r11\n" + "movq %q10, %%r12\n" + "movq %q11, %%r13\n" + /* Initialize (r8,r9,r10) */ + "movq %q5, %%r8\n" + "xorq %%r9, %%r9\n" + "xorq %%r10, %%r10\n" + /* (r8,r9) += m4 * c0 */ + "movq %12, %%rax\n" + "mulq %%r11\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + /* extract p0 */ + "movq %%r8, %q0\n" + "xorq %%r8, %%r8\n" + /* (r9,r10) += m1 */ + "addq %q6, %%r9\n" + "adcq $0, %%r10\n" + /* (r9,r10,r8) += m5 * c0 */ + "movq %12, %%rax\n" + "mulq %%r12\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* (r9,r10,r8) += m4 * c1 */ + "movq %13, %%rax\n" + "mulq %%r11\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* extract p1 */ + "movq %%r9, %q1\n" + "xorq %%r9, %%r9\n" + /* (r10,r8,r9) += m2 */ + "addq %q7, %%r10\n" + "adcq $0, %%r8\n" + "adcq $0, %%r9\n" + /* (r10,r8,r9) += m6 * c0 */ + "movq %12, %%rax\n" + "mulq %%r13\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + "adcq $0, %%r9\n" + /* (r10,r8,r9) += m5 * c1 */ + "movq %13, %%rax\n" + "mulq %%r12\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + "adcq $0, %%r9\n" + /* (r10,r8,r9) += m4 */ + "addq %%r11, %%r10\n" + "adcq $0, %%r8\n" + "adcq $0, %%r9\n" + /* extract p2 */ + "movq %%r10, %q2\n" + /* (r8,r9) += m3 */ + "addq %q8, %%r8\n" + "adcq $0, %%r9\n" + /* (r8,r9) += m6 * c1 */ + "movq %13, %%rax\n" + "mulq %%r13\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + /* (r8,r9) += m5 */ + "addq %%r12, %%r8\n" + "adcq $0, %%r9\n" + /* extract p3 */ + "movq %%r8, %q3\n" + /* (r9) += m6 */ + "addq %%r13, %%r9\n" + /* extract p4 */ + "movq %%r9, %q4\n" + : "=&g"(p0), "=&g"(p1), "=&g"(p2), "=g"(p3), "=g"(p4) + : "g"(m0), "g"(m1), "g"(m2), "g"(m3), "g"(m4), "g"(m5), "g"(m6), "i"(SECP256K1_N_C_0), "i"(SECP256K1_N_C_1) + : "rax", "rdx", "r8", "r9", "r10", "r11", "r12", "r13", "cc"); + + /* Reduce 258 bits into 256. */ + __asm__ __volatile__( + /* Preload */ + "movq %q5, %%r10\n" + /* (rax,rdx) = p4 * c0 */ + "movq %7, %%rax\n" + "mulq %%r10\n" + /* (rax,rdx) += p0 */ + "addq %q1, %%rax\n" + "adcq $0, %%rdx\n" + /* extract r0 */ + "movq %%rax, 0(%q6)\n" + /* Move to (r8,r9) */ + "movq %%rdx, %%r8\n" + "xorq %%r9, %%r9\n" + /* (r8,r9) += p1 */ + "addq %q2, %%r8\n" + "adcq $0, %%r9\n" + /* (r8,r9) += p4 * c1 */ + "movq %8, %%rax\n" + "mulq %%r10\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + /* Extract r1 */ + "movq %%r8, 8(%q6)\n" + "xorq %%r8, %%r8\n" + /* (r9,r8) += p4 */ + "addq %%r10, %%r9\n" + "adcq $0, %%r8\n" + /* (r9,r8) += p2 */ + "addq %q3, %%r9\n" + "adcq $0, %%r8\n" + /* Extract r2 */ + "movq %%r9, 16(%q6)\n" + "xorq %%r9, %%r9\n" + /* (r8,r9) += p3 */ + "addq %q4, %%r8\n" + "adcq $0, %%r9\n" + /* Extract r3 */ + "movq %%r8, 24(%q6)\n" + /* Extract c */ + "movq %%r9, %q0\n" + : "=g"(c) + : "g"(p0), "g"(p1), "g"(p2), "g"(p3), "g"(p4), "D"(r), "i"(SECP256K1_N_C_0), "i"(SECP256K1_N_C_1) + : "rax", "rdx", "r8", "r9", "r10", "cc", "memory"); +#else + uint128_t c; + uint64_t c0, c1, c2; + uint64_t n0 = l[4], n1 = l[5], n2 = l[6], n3 = l[7]; + uint64_t m0, m1, m2, m3, m4, m5; + uint32_t m6; + uint64_t p0, p1, p2, p3; + uint32_t p4; + + /* Reduce 512 bits into 385. */ + /* m[0..6] = l[0..3] + n[0..3] * SECP256K1_N_C. */ + c0 = l[0]; c1 = 0; c2 = 0; + muladd_fast(n0, SECP256K1_N_C_0); + extract_fast(m0); + sumadd_fast(l[1]); + muladd(n1, SECP256K1_N_C_0); + muladd(n0, SECP256K1_N_C_1); + extract(m1); + sumadd(l[2]); + muladd(n2, SECP256K1_N_C_0); + muladd(n1, SECP256K1_N_C_1); + sumadd(n0); + extract(m2); + sumadd(l[3]); + muladd(n3, SECP256K1_N_C_0); + muladd(n2, SECP256K1_N_C_1); + sumadd(n1); + extract(m3); + muladd(n3, SECP256K1_N_C_1); + sumadd(n2); + extract(m4); + sumadd_fast(n3); + extract_fast(m5); + VERIFY_CHECK(c0 <= 1); + m6 = c0; + + /* Reduce 385 bits into 258. */ + /* p[0..4] = m[0..3] + m[4..6] * SECP256K1_N_C. */ + c0 = m0; c1 = 0; c2 = 0; + muladd_fast(m4, SECP256K1_N_C_0); + extract_fast(p0); + sumadd_fast(m1); + muladd(m5, SECP256K1_N_C_0); + muladd(m4, SECP256K1_N_C_1); + extract(p1); + sumadd(m2); + muladd(m6, SECP256K1_N_C_0); + muladd(m5, SECP256K1_N_C_1); + sumadd(m4); + extract(p2); + sumadd_fast(m3); + muladd_fast(m6, SECP256K1_N_C_1); + sumadd_fast(m5); + extract_fast(p3); + p4 = c0 + m6; + VERIFY_CHECK(p4 <= 2); + + /* Reduce 258 bits into 256. */ + /* r[0..3] = p[0..3] + p[4] * SECP256K1_N_C. */ + c = p0 + (uint128_t)SECP256K1_N_C_0 * p4; + r->d[0] = c & 0xFFFFFFFFFFFFFFFFULL; c >>= 64; + c += p1 + (uint128_t)SECP256K1_N_C_1 * p4; + r->d[1] = c & 0xFFFFFFFFFFFFFFFFULL; c >>= 64; + c += p2 + (uint128_t)p4; + r->d[2] = c & 0xFFFFFFFFFFFFFFFFULL; c >>= 64; + c += p3; + r->d[3] = c & 0xFFFFFFFFFFFFFFFFULL; c >>= 64; +#endif + + /* Final reduction of r. */ + secp256k1_scalar_reduce(r, c + secp256k1_scalar_check_overflow(r)); +} + +static void secp256k1_scalar_mul_512(uint64_t l[8], const secp256k1_scalar *a, const secp256k1_scalar *b) { +#ifdef USE_ASM_X86_64 + const uint64_t *pb = b->d; + __asm__ __volatile__( + /* Preload */ + "movq 0(%%rdi), %%r15\n" + "movq 8(%%rdi), %%rbx\n" + "movq 16(%%rdi), %%rcx\n" + "movq 0(%%rdx), %%r11\n" + "movq 8(%%rdx), %%r12\n" + "movq 16(%%rdx), %%r13\n" + "movq 24(%%rdx), %%r14\n" + /* (rax,rdx) = a0 * b0 */ + "movq %%r15, %%rax\n" + "mulq %%r11\n" + /* Extract l0 */ + "movq %%rax, 0(%%rsi)\n" + /* (r8,r9,r10) = (rdx) */ + "movq %%rdx, %%r8\n" + "xorq %%r9, %%r9\n" + "xorq %%r10, %%r10\n" + /* (r8,r9,r10) += a0 * b1 */ + "movq %%r15, %%rax\n" + "mulq %%r12\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + "adcq $0, %%r10\n" + /* (r8,r9,r10) += a1 * b0 */ + "movq %%rbx, %%rax\n" + "mulq %%r11\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + "adcq $0, %%r10\n" + /* Extract l1 */ + "movq %%r8, 8(%%rsi)\n" + "xorq %%r8, %%r8\n" + /* (r9,r10,r8) += a0 * b2 */ + "movq %%r15, %%rax\n" + "mulq %%r13\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* (r9,r10,r8) += a1 * b1 */ + "movq %%rbx, %%rax\n" + "mulq %%r12\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* (r9,r10,r8) += a2 * b0 */ + "movq %%rcx, %%rax\n" + "mulq %%r11\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* Extract l2 */ + "movq %%r9, 16(%%rsi)\n" + "xorq %%r9, %%r9\n" + /* (r10,r8,r9) += a0 * b3 */ + "movq %%r15, %%rax\n" + "mulq %%r14\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + "adcq $0, %%r9\n" + /* Preload a3 */ + "movq 24(%%rdi), %%r15\n" + /* (r10,r8,r9) += a1 * b2 */ + "movq %%rbx, %%rax\n" + "mulq %%r13\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + "adcq $0, %%r9\n" + /* (r10,r8,r9) += a2 * b1 */ + "movq %%rcx, %%rax\n" + "mulq %%r12\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + "adcq $0, %%r9\n" + /* (r10,r8,r9) += a3 * b0 */ + "movq %%r15, %%rax\n" + "mulq %%r11\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + "adcq $0, %%r9\n" + /* Extract l3 */ + "movq %%r10, 24(%%rsi)\n" + "xorq %%r10, %%r10\n" + /* (r8,r9,r10) += a1 * b3 */ + "movq %%rbx, %%rax\n" + "mulq %%r14\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + "adcq $0, %%r10\n" + /* (r8,r9,r10) += a2 * b2 */ + "movq %%rcx, %%rax\n" + "mulq %%r13\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + "adcq $0, %%r10\n" + /* (r8,r9,r10) += a3 * b1 */ + "movq %%r15, %%rax\n" + "mulq %%r12\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + "adcq $0, %%r10\n" + /* Extract l4 */ + "movq %%r8, 32(%%rsi)\n" + "xorq %%r8, %%r8\n" + /* (r9,r10,r8) += a2 * b3 */ + "movq %%rcx, %%rax\n" + "mulq %%r14\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* (r9,r10,r8) += a3 * b2 */ + "movq %%r15, %%rax\n" + "mulq %%r13\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* Extract l5 */ + "movq %%r9, 40(%%rsi)\n" + /* (r10,r8) += a3 * b3 */ + "movq %%r15, %%rax\n" + "mulq %%r14\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + /* Extract l6 */ + "movq %%r10, 48(%%rsi)\n" + /* Extract l7 */ + "movq %%r8, 56(%%rsi)\n" + : "+d"(pb) + : "S"(l), "D"(a->d) + : "rax", "rbx", "rcx", "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15", "cc", "memory"); +#else + /* 160 bit accumulator. */ + uint64_t c0 = 0, c1 = 0; + uint32_t c2 = 0; + + /* l[0..7] = a[0..3] * b[0..3]. */ + muladd_fast(a->d[0], b->d[0]); + extract_fast(l[0]); + muladd(a->d[0], b->d[1]); + muladd(a->d[1], b->d[0]); + extract(l[1]); + muladd(a->d[0], b->d[2]); + muladd(a->d[1], b->d[1]); + muladd(a->d[2], b->d[0]); + extract(l[2]); + muladd(a->d[0], b->d[3]); + muladd(a->d[1], b->d[2]); + muladd(a->d[2], b->d[1]); + muladd(a->d[3], b->d[0]); + extract(l[3]); + muladd(a->d[1], b->d[3]); + muladd(a->d[2], b->d[2]); + muladd(a->d[3], b->d[1]); + extract(l[4]); + muladd(a->d[2], b->d[3]); + muladd(a->d[3], b->d[2]); + extract(l[5]); + muladd_fast(a->d[3], b->d[3]); + extract_fast(l[6]); + VERIFY_CHECK(c1 == 0); + l[7] = c0; +#endif +} + +static void secp256k1_scalar_sqr_512(uint64_t l[8], const secp256k1_scalar *a) { +#ifdef USE_ASM_X86_64 + __asm__ __volatile__( + /* Preload */ + "movq 0(%%rdi), %%r11\n" + "movq 8(%%rdi), %%r12\n" + "movq 16(%%rdi), %%r13\n" + "movq 24(%%rdi), %%r14\n" + /* (rax,rdx) = a0 * a0 */ + "movq %%r11, %%rax\n" + "mulq %%r11\n" + /* Extract l0 */ + "movq %%rax, 0(%%rsi)\n" + /* (r8,r9,r10) = (rdx,0) */ + "movq %%rdx, %%r8\n" + "xorq %%r9, %%r9\n" + "xorq %%r10, %%r10\n" + /* (r8,r9,r10) += 2 * a0 * a1 */ + "movq %%r11, %%rax\n" + "mulq %%r12\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + "adcq $0, %%r10\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + "adcq $0, %%r10\n" + /* Extract l1 */ + "movq %%r8, 8(%%rsi)\n" + "xorq %%r8, %%r8\n" + /* (r9,r10,r8) += 2 * a0 * a2 */ + "movq %%r11, %%rax\n" + "mulq %%r13\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* (r9,r10,r8) += a1 * a1 */ + "movq %%r12, %%rax\n" + "mulq %%r12\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* Extract l2 */ + "movq %%r9, 16(%%rsi)\n" + "xorq %%r9, %%r9\n" + /* (r10,r8,r9) += 2 * a0 * a3 */ + "movq %%r11, %%rax\n" + "mulq %%r14\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + "adcq $0, %%r9\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + "adcq $0, %%r9\n" + /* (r10,r8,r9) += 2 * a1 * a2 */ + "movq %%r12, %%rax\n" + "mulq %%r13\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + "adcq $0, %%r9\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + "adcq $0, %%r9\n" + /* Extract l3 */ + "movq %%r10, 24(%%rsi)\n" + "xorq %%r10, %%r10\n" + /* (r8,r9,r10) += 2 * a1 * a3 */ + "movq %%r12, %%rax\n" + "mulq %%r14\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + "adcq $0, %%r10\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + "adcq $0, %%r10\n" + /* (r8,r9,r10) += a2 * a2 */ + "movq %%r13, %%rax\n" + "mulq %%r13\n" + "addq %%rax, %%r8\n" + "adcq %%rdx, %%r9\n" + "adcq $0, %%r10\n" + /* Extract l4 */ + "movq %%r8, 32(%%rsi)\n" + "xorq %%r8, %%r8\n" + /* (r9,r10,r8) += 2 * a2 * a3 */ + "movq %%r13, %%rax\n" + "mulq %%r14\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + "addq %%rax, %%r9\n" + "adcq %%rdx, %%r10\n" + "adcq $0, %%r8\n" + /* Extract l5 */ + "movq %%r9, 40(%%rsi)\n" + /* (r10,r8) += a3 * a3 */ + "movq %%r14, %%rax\n" + "mulq %%r14\n" + "addq %%rax, %%r10\n" + "adcq %%rdx, %%r8\n" + /* Extract l6 */ + "movq %%r10, 48(%%rsi)\n" + /* Extract l7 */ + "movq %%r8, 56(%%rsi)\n" + : + : "S"(l), "D"(a->d) + : "rax", "rdx", "r8", "r9", "r10", "r11", "r12", "r13", "r14", "cc", "memory"); +#else + /* 160 bit accumulator. */ + uint64_t c0 = 0, c1 = 0; + uint32_t c2 = 0; + + /* l[0..7] = a[0..3] * b[0..3]. */ + muladd_fast(a->d[0], a->d[0]); + extract_fast(l[0]); + muladd2(a->d[0], a->d[1]); + extract(l[1]); + muladd2(a->d[0], a->d[2]); + muladd(a->d[1], a->d[1]); + extract(l[2]); + muladd2(a->d[0], a->d[3]); + muladd2(a->d[1], a->d[2]); + extract(l[3]); + muladd2(a->d[1], a->d[3]); + muladd(a->d[2], a->d[2]); + extract(l[4]); + muladd2(a->d[2], a->d[3]); + extract(l[5]); + muladd_fast(a->d[3], a->d[3]); + extract_fast(l[6]); + VERIFY_CHECK(c1 == 0); + l[7] = c0; +#endif +} + +#undef sumadd +#undef sumadd_fast +#undef muladd +#undef muladd_fast +#undef muladd2 +#undef extract +#undef extract_fast + +static void secp256k1_scalar_mul(secp256k1_scalar *r, const secp256k1_scalar *a, const secp256k1_scalar *b) { + uint64_t l[8]; + secp256k1_scalar_mul_512(l, a, b); + secp256k1_scalar_reduce_512(r, l); +} + +static int secp256k1_scalar_shr_int(secp256k1_scalar *r, int n) { + int ret; + VERIFY_CHECK(n > 0); + VERIFY_CHECK(n < 16); + ret = r->d[0] & ((1 << n) - 1); + r->d[0] = (r->d[0] >> n) + (r->d[1] << (64 - n)); + r->d[1] = (r->d[1] >> n) + (r->d[2] << (64 - n)); + r->d[2] = (r->d[2] >> n) + (r->d[3] << (64 - n)); + r->d[3] = (r->d[3] >> n); + return ret; +} + +static void secp256k1_scalar_sqr(secp256k1_scalar *r, const secp256k1_scalar *a) { + uint64_t l[8]; + secp256k1_scalar_sqr_512(l, a); + secp256k1_scalar_reduce_512(r, l); +} + +#ifdef USE_ENDOMORPHISM +static void secp256k1_scalar_split_128(secp256k1_scalar *r1, secp256k1_scalar *r2, const secp256k1_scalar *a) { + r1->d[0] = a->d[0]; + r1->d[1] = a->d[1]; + r1->d[2] = 0; + r1->d[3] = 0; + r2->d[0] = a->d[2]; + r2->d[1] = a->d[3]; + r2->d[2] = 0; + r2->d[3] = 0; +} +#endif + +SECP256K1_INLINE static int secp256k1_scalar_eq(const secp256k1_scalar *a, const secp256k1_scalar *b) { + return ((a->d[0] ^ b->d[0]) | (a->d[1] ^ b->d[1]) | (a->d[2] ^ b->d[2]) | (a->d[3] ^ b->d[3])) == 0; +} + +SECP256K1_INLINE static void secp256k1_scalar_mul_shift_var(secp256k1_scalar *r, const secp256k1_scalar *a, const secp256k1_scalar *b, unsigned int shift) { + uint64_t l[8]; + unsigned int shiftlimbs; + unsigned int shiftlow; + unsigned int shifthigh; + VERIFY_CHECK(shift >= 256); + secp256k1_scalar_mul_512(l, a, b); + shiftlimbs = shift >> 6; + shiftlow = shift & 0x3F; + shifthigh = 64 - shiftlow; + r->d[0] = shift < 512 ? (l[0 + shiftlimbs] >> shiftlow | (shift < 448 && shiftlow ? (l[1 + shiftlimbs] << shifthigh) : 0)) : 0; + r->d[1] = shift < 448 ? (l[1 + shiftlimbs] >> shiftlow | (shift < 384 && shiftlow ? (l[2 + shiftlimbs] << shifthigh) : 0)) : 0; + r->d[2] = shift < 384 ? (l[2 + shiftlimbs] >> shiftlow | (shift < 320 && shiftlow ? (l[3 + shiftlimbs] << shifthigh) : 0)) : 0; + r->d[3] = shift < 320 ? (l[3 + shiftlimbs] >> shiftlow) : 0; + secp256k1_scalar_cadd_bit(r, 0, (l[(shift - 1) >> 6] >> ((shift - 1) & 0x3f)) & 1); +} + +#endif /* SECP256K1_SCALAR_REPR_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/scalar_8x32.h b/node_modules/secp256k1/src/secp256k1/src/scalar_8x32.h new file mode 100644 index 0000000..2c9a348 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/scalar_8x32.h @@ -0,0 +1,19 @@ +/********************************************************************** + * Copyright (c) 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_SCALAR_REPR_H +#define SECP256K1_SCALAR_REPR_H + +#include + +/** A scalar modulo the group order of the secp256k1 curve. */ +typedef struct { + uint32_t d[8]; +} secp256k1_scalar; + +#define SECP256K1_SCALAR_CONST(d7, d6, d5, d4, d3, d2, d1, d0) {{(d0), (d1), (d2), (d3), (d4), (d5), (d6), (d7)}} + +#endif /* SECP256K1_SCALAR_REPR_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/scalar_8x32_impl.h b/node_modules/secp256k1/src/secp256k1/src/scalar_8x32_impl.h new file mode 100644 index 0000000..4f9ed61 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/scalar_8x32_impl.h @@ -0,0 +1,721 @@ +/********************************************************************** + * Copyright (c) 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_SCALAR_REPR_IMPL_H +#define SECP256K1_SCALAR_REPR_IMPL_H + +/* Limbs of the secp256k1 order. */ +#define SECP256K1_N_0 ((uint32_t)0xD0364141UL) +#define SECP256K1_N_1 ((uint32_t)0xBFD25E8CUL) +#define SECP256K1_N_2 ((uint32_t)0xAF48A03BUL) +#define SECP256K1_N_3 ((uint32_t)0xBAAEDCE6UL) +#define SECP256K1_N_4 ((uint32_t)0xFFFFFFFEUL) +#define SECP256K1_N_5 ((uint32_t)0xFFFFFFFFUL) +#define SECP256K1_N_6 ((uint32_t)0xFFFFFFFFUL) +#define SECP256K1_N_7 ((uint32_t)0xFFFFFFFFUL) + +/* Limbs of 2^256 minus the secp256k1 order. */ +#define SECP256K1_N_C_0 (~SECP256K1_N_0 + 1) +#define SECP256K1_N_C_1 (~SECP256K1_N_1) +#define SECP256K1_N_C_2 (~SECP256K1_N_2) +#define SECP256K1_N_C_3 (~SECP256K1_N_3) +#define SECP256K1_N_C_4 (1) + +/* Limbs of half the secp256k1 order. */ +#define SECP256K1_N_H_0 ((uint32_t)0x681B20A0UL) +#define SECP256K1_N_H_1 ((uint32_t)0xDFE92F46UL) +#define SECP256K1_N_H_2 ((uint32_t)0x57A4501DUL) +#define SECP256K1_N_H_3 ((uint32_t)0x5D576E73UL) +#define SECP256K1_N_H_4 ((uint32_t)0xFFFFFFFFUL) +#define SECP256K1_N_H_5 ((uint32_t)0xFFFFFFFFUL) +#define SECP256K1_N_H_6 ((uint32_t)0xFFFFFFFFUL) +#define SECP256K1_N_H_7 ((uint32_t)0x7FFFFFFFUL) + +SECP256K1_INLINE static void secp256k1_scalar_clear(secp256k1_scalar *r) { + r->d[0] = 0; + r->d[1] = 0; + r->d[2] = 0; + r->d[3] = 0; + r->d[4] = 0; + r->d[5] = 0; + r->d[6] = 0; + r->d[7] = 0; +} + +SECP256K1_INLINE static void secp256k1_scalar_set_int(secp256k1_scalar *r, unsigned int v) { + r->d[0] = v; + r->d[1] = 0; + r->d[2] = 0; + r->d[3] = 0; + r->d[4] = 0; + r->d[5] = 0; + r->d[6] = 0; + r->d[7] = 0; +} + +SECP256K1_INLINE static unsigned int secp256k1_scalar_get_bits(const secp256k1_scalar *a, unsigned int offset, unsigned int count) { + VERIFY_CHECK((offset + count - 1) >> 5 == offset >> 5); + return (a->d[offset >> 5] >> (offset & 0x1F)) & ((1 << count) - 1); +} + +SECP256K1_INLINE static unsigned int secp256k1_scalar_get_bits_var(const secp256k1_scalar *a, unsigned int offset, unsigned int count) { + VERIFY_CHECK(count < 32); + VERIFY_CHECK(offset + count <= 256); + if ((offset + count - 1) >> 5 == offset >> 5) { + return secp256k1_scalar_get_bits(a, offset, count); + } else { + VERIFY_CHECK((offset >> 5) + 1 < 8); + return ((a->d[offset >> 5] >> (offset & 0x1F)) | (a->d[(offset >> 5) + 1] << (32 - (offset & 0x1F)))) & ((((uint32_t)1) << count) - 1); + } +} + +SECP256K1_INLINE static int secp256k1_scalar_check_overflow(const secp256k1_scalar *a) { + int yes = 0; + int no = 0; + no |= (a->d[7] < SECP256K1_N_7); /* No need for a > check. */ + no |= (a->d[6] < SECP256K1_N_6); /* No need for a > check. */ + no |= (a->d[5] < SECP256K1_N_5); /* No need for a > check. */ + no |= (a->d[4] < SECP256K1_N_4); + yes |= (a->d[4] > SECP256K1_N_4) & ~no; + no |= (a->d[3] < SECP256K1_N_3) & ~yes; + yes |= (a->d[3] > SECP256K1_N_3) & ~no; + no |= (a->d[2] < SECP256K1_N_2) & ~yes; + yes |= (a->d[2] > SECP256K1_N_2) & ~no; + no |= (a->d[1] < SECP256K1_N_1) & ~yes; + yes |= (a->d[1] > SECP256K1_N_1) & ~no; + yes |= (a->d[0] >= SECP256K1_N_0) & ~no; + return yes; +} + +SECP256K1_INLINE static int secp256k1_scalar_reduce(secp256k1_scalar *r, uint32_t overflow) { + uint64_t t; + VERIFY_CHECK(overflow <= 1); + t = (uint64_t)r->d[0] + overflow * SECP256K1_N_C_0; + r->d[0] = t & 0xFFFFFFFFUL; t >>= 32; + t += (uint64_t)r->d[1] + overflow * SECP256K1_N_C_1; + r->d[1] = t & 0xFFFFFFFFUL; t >>= 32; + t += (uint64_t)r->d[2] + overflow * SECP256K1_N_C_2; + r->d[2] = t & 0xFFFFFFFFUL; t >>= 32; + t += (uint64_t)r->d[3] + overflow * SECP256K1_N_C_3; + r->d[3] = t & 0xFFFFFFFFUL; t >>= 32; + t += (uint64_t)r->d[4] + overflow * SECP256K1_N_C_4; + r->d[4] = t & 0xFFFFFFFFUL; t >>= 32; + t += (uint64_t)r->d[5]; + r->d[5] = t & 0xFFFFFFFFUL; t >>= 32; + t += (uint64_t)r->d[6]; + r->d[6] = t & 0xFFFFFFFFUL; t >>= 32; + t += (uint64_t)r->d[7]; + r->d[7] = t & 0xFFFFFFFFUL; + return overflow; +} + +static int secp256k1_scalar_add(secp256k1_scalar *r, const secp256k1_scalar *a, const secp256k1_scalar *b) { + int overflow; + uint64_t t = (uint64_t)a->d[0] + b->d[0]; + r->d[0] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)a->d[1] + b->d[1]; + r->d[1] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)a->d[2] + b->d[2]; + r->d[2] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)a->d[3] + b->d[3]; + r->d[3] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)a->d[4] + b->d[4]; + r->d[4] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)a->d[5] + b->d[5]; + r->d[5] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)a->d[6] + b->d[6]; + r->d[6] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)a->d[7] + b->d[7]; + r->d[7] = t & 0xFFFFFFFFULL; t >>= 32; + overflow = t + secp256k1_scalar_check_overflow(r); + VERIFY_CHECK(overflow == 0 || overflow == 1); + secp256k1_scalar_reduce(r, overflow); + return overflow; +} + +static void secp256k1_scalar_cadd_bit(secp256k1_scalar *r, unsigned int bit, int flag) { + uint64_t t; + VERIFY_CHECK(bit < 256); + bit += ((uint32_t) flag - 1) & 0x100; /* forcing (bit >> 5) > 7 makes this a noop */ + t = (uint64_t)r->d[0] + (((uint32_t)((bit >> 5) == 0)) << (bit & 0x1F)); + r->d[0] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)r->d[1] + (((uint32_t)((bit >> 5) == 1)) << (bit & 0x1F)); + r->d[1] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)r->d[2] + (((uint32_t)((bit >> 5) == 2)) << (bit & 0x1F)); + r->d[2] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)r->d[3] + (((uint32_t)((bit >> 5) == 3)) << (bit & 0x1F)); + r->d[3] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)r->d[4] + (((uint32_t)((bit >> 5) == 4)) << (bit & 0x1F)); + r->d[4] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)r->d[5] + (((uint32_t)((bit >> 5) == 5)) << (bit & 0x1F)); + r->d[5] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)r->d[6] + (((uint32_t)((bit >> 5) == 6)) << (bit & 0x1F)); + r->d[6] = t & 0xFFFFFFFFULL; t >>= 32; + t += (uint64_t)r->d[7] + (((uint32_t)((bit >> 5) == 7)) << (bit & 0x1F)); + r->d[7] = t & 0xFFFFFFFFULL; +#ifdef VERIFY + VERIFY_CHECK((t >> 32) == 0); + VERIFY_CHECK(secp256k1_scalar_check_overflow(r) == 0); +#endif +} + +static void secp256k1_scalar_set_b32(secp256k1_scalar *r, const unsigned char *b32, int *overflow) { + int over; + r->d[0] = (uint32_t)b32[31] | (uint32_t)b32[30] << 8 | (uint32_t)b32[29] << 16 | (uint32_t)b32[28] << 24; + r->d[1] = (uint32_t)b32[27] | (uint32_t)b32[26] << 8 | (uint32_t)b32[25] << 16 | (uint32_t)b32[24] << 24; + r->d[2] = (uint32_t)b32[23] | (uint32_t)b32[22] << 8 | (uint32_t)b32[21] << 16 | (uint32_t)b32[20] << 24; + r->d[3] = (uint32_t)b32[19] | (uint32_t)b32[18] << 8 | (uint32_t)b32[17] << 16 | (uint32_t)b32[16] << 24; + r->d[4] = (uint32_t)b32[15] | (uint32_t)b32[14] << 8 | (uint32_t)b32[13] << 16 | (uint32_t)b32[12] << 24; + r->d[5] = (uint32_t)b32[11] | (uint32_t)b32[10] << 8 | (uint32_t)b32[9] << 16 | (uint32_t)b32[8] << 24; + r->d[6] = (uint32_t)b32[7] | (uint32_t)b32[6] << 8 | (uint32_t)b32[5] << 16 | (uint32_t)b32[4] << 24; + r->d[7] = (uint32_t)b32[3] | (uint32_t)b32[2] << 8 | (uint32_t)b32[1] << 16 | (uint32_t)b32[0] << 24; + over = secp256k1_scalar_reduce(r, secp256k1_scalar_check_overflow(r)); + if (overflow) { + *overflow = over; + } +} + +static void secp256k1_scalar_get_b32(unsigned char *bin, const secp256k1_scalar* a) { + bin[0] = a->d[7] >> 24; bin[1] = a->d[7] >> 16; bin[2] = a->d[7] >> 8; bin[3] = a->d[7]; + bin[4] = a->d[6] >> 24; bin[5] = a->d[6] >> 16; bin[6] = a->d[6] >> 8; bin[7] = a->d[6]; + bin[8] = a->d[5] >> 24; bin[9] = a->d[5] >> 16; bin[10] = a->d[5] >> 8; bin[11] = a->d[5]; + bin[12] = a->d[4] >> 24; bin[13] = a->d[4] >> 16; bin[14] = a->d[4] >> 8; bin[15] = a->d[4]; + bin[16] = a->d[3] >> 24; bin[17] = a->d[3] >> 16; bin[18] = a->d[3] >> 8; bin[19] = a->d[3]; + bin[20] = a->d[2] >> 24; bin[21] = a->d[2] >> 16; bin[22] = a->d[2] >> 8; bin[23] = a->d[2]; + bin[24] = a->d[1] >> 24; bin[25] = a->d[1] >> 16; bin[26] = a->d[1] >> 8; bin[27] = a->d[1]; + bin[28] = a->d[0] >> 24; bin[29] = a->d[0] >> 16; bin[30] = a->d[0] >> 8; bin[31] = a->d[0]; +} + +SECP256K1_INLINE static int secp256k1_scalar_is_zero(const secp256k1_scalar *a) { + return (a->d[0] | a->d[1] | a->d[2] | a->d[3] | a->d[4] | a->d[5] | a->d[6] | a->d[7]) == 0; +} + +static void secp256k1_scalar_negate(secp256k1_scalar *r, const secp256k1_scalar *a) { + uint32_t nonzero = 0xFFFFFFFFUL * (secp256k1_scalar_is_zero(a) == 0); + uint64_t t = (uint64_t)(~a->d[0]) + SECP256K1_N_0 + 1; + r->d[0] = t & nonzero; t >>= 32; + t += (uint64_t)(~a->d[1]) + SECP256K1_N_1; + r->d[1] = t & nonzero; t >>= 32; + t += (uint64_t)(~a->d[2]) + SECP256K1_N_2; + r->d[2] = t & nonzero; t >>= 32; + t += (uint64_t)(~a->d[3]) + SECP256K1_N_3; + r->d[3] = t & nonzero; t >>= 32; + t += (uint64_t)(~a->d[4]) + SECP256K1_N_4; + r->d[4] = t & nonzero; t >>= 32; + t += (uint64_t)(~a->d[5]) + SECP256K1_N_5; + r->d[5] = t & nonzero; t >>= 32; + t += (uint64_t)(~a->d[6]) + SECP256K1_N_6; + r->d[6] = t & nonzero; t >>= 32; + t += (uint64_t)(~a->d[7]) + SECP256K1_N_7; + r->d[7] = t & nonzero; +} + +SECP256K1_INLINE static int secp256k1_scalar_is_one(const secp256k1_scalar *a) { + return ((a->d[0] ^ 1) | a->d[1] | a->d[2] | a->d[3] | a->d[4] | a->d[5] | a->d[6] | a->d[7]) == 0; +} + +static int secp256k1_scalar_is_high(const secp256k1_scalar *a) { + int yes = 0; + int no = 0; + no |= (a->d[7] < SECP256K1_N_H_7); + yes |= (a->d[7] > SECP256K1_N_H_7) & ~no; + no |= (a->d[6] < SECP256K1_N_H_6) & ~yes; /* No need for a > check. */ + no |= (a->d[5] < SECP256K1_N_H_5) & ~yes; /* No need for a > check. */ + no |= (a->d[4] < SECP256K1_N_H_4) & ~yes; /* No need for a > check. */ + no |= (a->d[3] < SECP256K1_N_H_3) & ~yes; + yes |= (a->d[3] > SECP256K1_N_H_3) & ~no; + no |= (a->d[2] < SECP256K1_N_H_2) & ~yes; + yes |= (a->d[2] > SECP256K1_N_H_2) & ~no; + no |= (a->d[1] < SECP256K1_N_H_1) & ~yes; + yes |= (a->d[1] > SECP256K1_N_H_1) & ~no; + yes |= (a->d[0] > SECP256K1_N_H_0) & ~no; + return yes; +} + +static int secp256k1_scalar_cond_negate(secp256k1_scalar *r, int flag) { + /* If we are flag = 0, mask = 00...00 and this is a no-op; + * if we are flag = 1, mask = 11...11 and this is identical to secp256k1_scalar_negate */ + uint32_t mask = !flag - 1; + uint32_t nonzero = 0xFFFFFFFFUL * (secp256k1_scalar_is_zero(r) == 0); + uint64_t t = (uint64_t)(r->d[0] ^ mask) + ((SECP256K1_N_0 + 1) & mask); + r->d[0] = t & nonzero; t >>= 32; + t += (uint64_t)(r->d[1] ^ mask) + (SECP256K1_N_1 & mask); + r->d[1] = t & nonzero; t >>= 32; + t += (uint64_t)(r->d[2] ^ mask) + (SECP256K1_N_2 & mask); + r->d[2] = t & nonzero; t >>= 32; + t += (uint64_t)(r->d[3] ^ mask) + (SECP256K1_N_3 & mask); + r->d[3] = t & nonzero; t >>= 32; + t += (uint64_t)(r->d[4] ^ mask) + (SECP256K1_N_4 & mask); + r->d[4] = t & nonzero; t >>= 32; + t += (uint64_t)(r->d[5] ^ mask) + (SECP256K1_N_5 & mask); + r->d[5] = t & nonzero; t >>= 32; + t += (uint64_t)(r->d[6] ^ mask) + (SECP256K1_N_6 & mask); + r->d[6] = t & nonzero; t >>= 32; + t += (uint64_t)(r->d[7] ^ mask) + (SECP256K1_N_7 & mask); + r->d[7] = t & nonzero; + return 2 * (mask == 0) - 1; +} + + +/* Inspired by the macros in OpenSSL's crypto/bn/asm/x86_64-gcc.c. */ + +/** Add a*b to the number defined by (c0,c1,c2). c2 must never overflow. */ +#define muladd(a,b) { \ + uint32_t tl, th; \ + { \ + uint64_t t = (uint64_t)a * b; \ + th = t >> 32; /* at most 0xFFFFFFFE */ \ + tl = t; \ + } \ + c0 += tl; /* overflow is handled on the next line */ \ + th += (c0 < tl) ? 1 : 0; /* at most 0xFFFFFFFF */ \ + c1 += th; /* overflow is handled on the next line */ \ + c2 += (c1 < th) ? 1 : 0; /* never overflows by contract (verified in the next line) */ \ + VERIFY_CHECK((c1 >= th) || (c2 != 0)); \ +} + +/** Add a*b to the number defined by (c0,c1). c1 must never overflow. */ +#define muladd_fast(a,b) { \ + uint32_t tl, th; \ + { \ + uint64_t t = (uint64_t)a * b; \ + th = t >> 32; /* at most 0xFFFFFFFE */ \ + tl = t; \ + } \ + c0 += tl; /* overflow is handled on the next line */ \ + th += (c0 < tl) ? 1 : 0; /* at most 0xFFFFFFFF */ \ + c1 += th; /* never overflows by contract (verified in the next line) */ \ + VERIFY_CHECK(c1 >= th); \ +} + +/** Add 2*a*b to the number defined by (c0,c1,c2). c2 must never overflow. */ +#define muladd2(a,b) { \ + uint32_t tl, th, th2, tl2; \ + { \ + uint64_t t = (uint64_t)a * b; \ + th = t >> 32; /* at most 0xFFFFFFFE */ \ + tl = t; \ + } \ + th2 = th + th; /* at most 0xFFFFFFFE (in case th was 0x7FFFFFFF) */ \ + c2 += (th2 < th) ? 1 : 0; /* never overflows by contract (verified the next line) */ \ + VERIFY_CHECK((th2 >= th) || (c2 != 0)); \ + tl2 = tl + tl; /* at most 0xFFFFFFFE (in case the lowest 63 bits of tl were 0x7FFFFFFF) */ \ + th2 += (tl2 < tl) ? 1 : 0; /* at most 0xFFFFFFFF */ \ + c0 += tl2; /* overflow is handled on the next line */ \ + th2 += (c0 < tl2) ? 1 : 0; /* second overflow is handled on the next line */ \ + c2 += (c0 < tl2) & (th2 == 0); /* never overflows by contract (verified the next line) */ \ + VERIFY_CHECK((c0 >= tl2) || (th2 != 0) || (c2 != 0)); \ + c1 += th2; /* overflow is handled on the next line */ \ + c2 += (c1 < th2) ? 1 : 0; /* never overflows by contract (verified the next line) */ \ + VERIFY_CHECK((c1 >= th2) || (c2 != 0)); \ +} + +/** Add a to the number defined by (c0,c1,c2). c2 must never overflow. */ +#define sumadd(a) { \ + unsigned int over; \ + c0 += (a); /* overflow is handled on the next line */ \ + over = (c0 < (a)) ? 1 : 0; \ + c1 += over; /* overflow is handled on the next line */ \ + c2 += (c1 < over) ? 1 : 0; /* never overflows by contract */ \ +} + +/** Add a to the number defined by (c0,c1). c1 must never overflow, c2 must be zero. */ +#define sumadd_fast(a) { \ + c0 += (a); /* overflow is handled on the next line */ \ + c1 += (c0 < (a)) ? 1 : 0; /* never overflows by contract (verified the next line) */ \ + VERIFY_CHECK((c1 != 0) | (c0 >= (a))); \ + VERIFY_CHECK(c2 == 0); \ +} + +/** Extract the lowest 32 bits of (c0,c1,c2) into n, and left shift the number 32 bits. */ +#define extract(n) { \ + (n) = c0; \ + c0 = c1; \ + c1 = c2; \ + c2 = 0; \ +} + +/** Extract the lowest 32 bits of (c0,c1,c2) into n, and left shift the number 32 bits. c2 is required to be zero. */ +#define extract_fast(n) { \ + (n) = c0; \ + c0 = c1; \ + c1 = 0; \ + VERIFY_CHECK(c2 == 0); \ +} + +static void secp256k1_scalar_reduce_512(secp256k1_scalar *r, const uint32_t *l) { + uint64_t c; + uint32_t n0 = l[8], n1 = l[9], n2 = l[10], n3 = l[11], n4 = l[12], n5 = l[13], n6 = l[14], n7 = l[15]; + uint32_t m0, m1, m2, m3, m4, m5, m6, m7, m8, m9, m10, m11, m12; + uint32_t p0, p1, p2, p3, p4, p5, p6, p7, p8; + + /* 96 bit accumulator. */ + uint32_t c0, c1, c2; + + /* Reduce 512 bits into 385. */ + /* m[0..12] = l[0..7] + n[0..7] * SECP256K1_N_C. */ + c0 = l[0]; c1 = 0; c2 = 0; + muladd_fast(n0, SECP256K1_N_C_0); + extract_fast(m0); + sumadd_fast(l[1]); + muladd(n1, SECP256K1_N_C_0); + muladd(n0, SECP256K1_N_C_1); + extract(m1); + sumadd(l[2]); + muladd(n2, SECP256K1_N_C_0); + muladd(n1, SECP256K1_N_C_1); + muladd(n0, SECP256K1_N_C_2); + extract(m2); + sumadd(l[3]); + muladd(n3, SECP256K1_N_C_0); + muladd(n2, SECP256K1_N_C_1); + muladd(n1, SECP256K1_N_C_2); + muladd(n0, SECP256K1_N_C_3); + extract(m3); + sumadd(l[4]); + muladd(n4, SECP256K1_N_C_0); + muladd(n3, SECP256K1_N_C_1); + muladd(n2, SECP256K1_N_C_2); + muladd(n1, SECP256K1_N_C_3); + sumadd(n0); + extract(m4); + sumadd(l[5]); + muladd(n5, SECP256K1_N_C_0); + muladd(n4, SECP256K1_N_C_1); + muladd(n3, SECP256K1_N_C_2); + muladd(n2, SECP256K1_N_C_3); + sumadd(n1); + extract(m5); + sumadd(l[6]); + muladd(n6, SECP256K1_N_C_0); + muladd(n5, SECP256K1_N_C_1); + muladd(n4, SECP256K1_N_C_2); + muladd(n3, SECP256K1_N_C_3); + sumadd(n2); + extract(m6); + sumadd(l[7]); + muladd(n7, SECP256K1_N_C_0); + muladd(n6, SECP256K1_N_C_1); + muladd(n5, SECP256K1_N_C_2); + muladd(n4, SECP256K1_N_C_3); + sumadd(n3); + extract(m7); + muladd(n7, SECP256K1_N_C_1); + muladd(n6, SECP256K1_N_C_2); + muladd(n5, SECP256K1_N_C_3); + sumadd(n4); + extract(m8); + muladd(n7, SECP256K1_N_C_2); + muladd(n6, SECP256K1_N_C_3); + sumadd(n5); + extract(m9); + muladd(n7, SECP256K1_N_C_3); + sumadd(n6); + extract(m10); + sumadd_fast(n7); + extract_fast(m11); + VERIFY_CHECK(c0 <= 1); + m12 = c0; + + /* Reduce 385 bits into 258. */ + /* p[0..8] = m[0..7] + m[8..12] * SECP256K1_N_C. */ + c0 = m0; c1 = 0; c2 = 0; + muladd_fast(m8, SECP256K1_N_C_0); + extract_fast(p0); + sumadd_fast(m1); + muladd(m9, SECP256K1_N_C_0); + muladd(m8, SECP256K1_N_C_1); + extract(p1); + sumadd(m2); + muladd(m10, SECP256K1_N_C_0); + muladd(m9, SECP256K1_N_C_1); + muladd(m8, SECP256K1_N_C_2); + extract(p2); + sumadd(m3); + muladd(m11, SECP256K1_N_C_0); + muladd(m10, SECP256K1_N_C_1); + muladd(m9, SECP256K1_N_C_2); + muladd(m8, SECP256K1_N_C_3); + extract(p3); + sumadd(m4); + muladd(m12, SECP256K1_N_C_0); + muladd(m11, SECP256K1_N_C_1); + muladd(m10, SECP256K1_N_C_2); + muladd(m9, SECP256K1_N_C_3); + sumadd(m8); + extract(p4); + sumadd(m5); + muladd(m12, SECP256K1_N_C_1); + muladd(m11, SECP256K1_N_C_2); + muladd(m10, SECP256K1_N_C_3); + sumadd(m9); + extract(p5); + sumadd(m6); + muladd(m12, SECP256K1_N_C_2); + muladd(m11, SECP256K1_N_C_3); + sumadd(m10); + extract(p6); + sumadd_fast(m7); + muladd_fast(m12, SECP256K1_N_C_3); + sumadd_fast(m11); + extract_fast(p7); + p8 = c0 + m12; + VERIFY_CHECK(p8 <= 2); + + /* Reduce 258 bits into 256. */ + /* r[0..7] = p[0..7] + p[8] * SECP256K1_N_C. */ + c = p0 + (uint64_t)SECP256K1_N_C_0 * p8; + r->d[0] = c & 0xFFFFFFFFUL; c >>= 32; + c += p1 + (uint64_t)SECP256K1_N_C_1 * p8; + r->d[1] = c & 0xFFFFFFFFUL; c >>= 32; + c += p2 + (uint64_t)SECP256K1_N_C_2 * p8; + r->d[2] = c & 0xFFFFFFFFUL; c >>= 32; + c += p3 + (uint64_t)SECP256K1_N_C_3 * p8; + r->d[3] = c & 0xFFFFFFFFUL; c >>= 32; + c += p4 + (uint64_t)p8; + r->d[4] = c & 0xFFFFFFFFUL; c >>= 32; + c += p5; + r->d[5] = c & 0xFFFFFFFFUL; c >>= 32; + c += p6; + r->d[6] = c & 0xFFFFFFFFUL; c >>= 32; + c += p7; + r->d[7] = c & 0xFFFFFFFFUL; c >>= 32; + + /* Final reduction of r. */ + secp256k1_scalar_reduce(r, c + secp256k1_scalar_check_overflow(r)); +} + +static void secp256k1_scalar_mul_512(uint32_t *l, const secp256k1_scalar *a, const secp256k1_scalar *b) { + /* 96 bit accumulator. */ + uint32_t c0 = 0, c1 = 0, c2 = 0; + + /* l[0..15] = a[0..7] * b[0..7]. */ + muladd_fast(a->d[0], b->d[0]); + extract_fast(l[0]); + muladd(a->d[0], b->d[1]); + muladd(a->d[1], b->d[0]); + extract(l[1]); + muladd(a->d[0], b->d[2]); + muladd(a->d[1], b->d[1]); + muladd(a->d[2], b->d[0]); + extract(l[2]); + muladd(a->d[0], b->d[3]); + muladd(a->d[1], b->d[2]); + muladd(a->d[2], b->d[1]); + muladd(a->d[3], b->d[0]); + extract(l[3]); + muladd(a->d[0], b->d[4]); + muladd(a->d[1], b->d[3]); + muladd(a->d[2], b->d[2]); + muladd(a->d[3], b->d[1]); + muladd(a->d[4], b->d[0]); + extract(l[4]); + muladd(a->d[0], b->d[5]); + muladd(a->d[1], b->d[4]); + muladd(a->d[2], b->d[3]); + muladd(a->d[3], b->d[2]); + muladd(a->d[4], b->d[1]); + muladd(a->d[5], b->d[0]); + extract(l[5]); + muladd(a->d[0], b->d[6]); + muladd(a->d[1], b->d[5]); + muladd(a->d[2], b->d[4]); + muladd(a->d[3], b->d[3]); + muladd(a->d[4], b->d[2]); + muladd(a->d[5], b->d[1]); + muladd(a->d[6], b->d[0]); + extract(l[6]); + muladd(a->d[0], b->d[7]); + muladd(a->d[1], b->d[6]); + muladd(a->d[2], b->d[5]); + muladd(a->d[3], b->d[4]); + muladd(a->d[4], b->d[3]); + muladd(a->d[5], b->d[2]); + muladd(a->d[6], b->d[1]); + muladd(a->d[7], b->d[0]); + extract(l[7]); + muladd(a->d[1], b->d[7]); + muladd(a->d[2], b->d[6]); + muladd(a->d[3], b->d[5]); + muladd(a->d[4], b->d[4]); + muladd(a->d[5], b->d[3]); + muladd(a->d[6], b->d[2]); + muladd(a->d[7], b->d[1]); + extract(l[8]); + muladd(a->d[2], b->d[7]); + muladd(a->d[3], b->d[6]); + muladd(a->d[4], b->d[5]); + muladd(a->d[5], b->d[4]); + muladd(a->d[6], b->d[3]); + muladd(a->d[7], b->d[2]); + extract(l[9]); + muladd(a->d[3], b->d[7]); + muladd(a->d[4], b->d[6]); + muladd(a->d[5], b->d[5]); + muladd(a->d[6], b->d[4]); + muladd(a->d[7], b->d[3]); + extract(l[10]); + muladd(a->d[4], b->d[7]); + muladd(a->d[5], b->d[6]); + muladd(a->d[6], b->d[5]); + muladd(a->d[7], b->d[4]); + extract(l[11]); + muladd(a->d[5], b->d[7]); + muladd(a->d[6], b->d[6]); + muladd(a->d[7], b->d[5]); + extract(l[12]); + muladd(a->d[6], b->d[7]); + muladd(a->d[7], b->d[6]); + extract(l[13]); + muladd_fast(a->d[7], b->d[7]); + extract_fast(l[14]); + VERIFY_CHECK(c1 == 0); + l[15] = c0; +} + +static void secp256k1_scalar_sqr_512(uint32_t *l, const secp256k1_scalar *a) { + /* 96 bit accumulator. */ + uint32_t c0 = 0, c1 = 0, c2 = 0; + + /* l[0..15] = a[0..7]^2. */ + muladd_fast(a->d[0], a->d[0]); + extract_fast(l[0]); + muladd2(a->d[0], a->d[1]); + extract(l[1]); + muladd2(a->d[0], a->d[2]); + muladd(a->d[1], a->d[1]); + extract(l[2]); + muladd2(a->d[0], a->d[3]); + muladd2(a->d[1], a->d[2]); + extract(l[3]); + muladd2(a->d[0], a->d[4]); + muladd2(a->d[1], a->d[3]); + muladd(a->d[2], a->d[2]); + extract(l[4]); + muladd2(a->d[0], a->d[5]); + muladd2(a->d[1], a->d[4]); + muladd2(a->d[2], a->d[3]); + extract(l[5]); + muladd2(a->d[0], a->d[6]); + muladd2(a->d[1], a->d[5]); + muladd2(a->d[2], a->d[4]); + muladd(a->d[3], a->d[3]); + extract(l[6]); + muladd2(a->d[0], a->d[7]); + muladd2(a->d[1], a->d[6]); + muladd2(a->d[2], a->d[5]); + muladd2(a->d[3], a->d[4]); + extract(l[7]); + muladd2(a->d[1], a->d[7]); + muladd2(a->d[2], a->d[6]); + muladd2(a->d[3], a->d[5]); + muladd(a->d[4], a->d[4]); + extract(l[8]); + muladd2(a->d[2], a->d[7]); + muladd2(a->d[3], a->d[6]); + muladd2(a->d[4], a->d[5]); + extract(l[9]); + muladd2(a->d[3], a->d[7]); + muladd2(a->d[4], a->d[6]); + muladd(a->d[5], a->d[5]); + extract(l[10]); + muladd2(a->d[4], a->d[7]); + muladd2(a->d[5], a->d[6]); + extract(l[11]); + muladd2(a->d[5], a->d[7]); + muladd(a->d[6], a->d[6]); + extract(l[12]); + muladd2(a->d[6], a->d[7]); + extract(l[13]); + muladd_fast(a->d[7], a->d[7]); + extract_fast(l[14]); + VERIFY_CHECK(c1 == 0); + l[15] = c0; +} + +#undef sumadd +#undef sumadd_fast +#undef muladd +#undef muladd_fast +#undef muladd2 +#undef extract +#undef extract_fast + +static void secp256k1_scalar_mul(secp256k1_scalar *r, const secp256k1_scalar *a, const secp256k1_scalar *b) { + uint32_t l[16]; + secp256k1_scalar_mul_512(l, a, b); + secp256k1_scalar_reduce_512(r, l); +} + +static int secp256k1_scalar_shr_int(secp256k1_scalar *r, int n) { + int ret; + VERIFY_CHECK(n > 0); + VERIFY_CHECK(n < 16); + ret = r->d[0] & ((1 << n) - 1); + r->d[0] = (r->d[0] >> n) + (r->d[1] << (32 - n)); + r->d[1] = (r->d[1] >> n) + (r->d[2] << (32 - n)); + r->d[2] = (r->d[2] >> n) + (r->d[3] << (32 - n)); + r->d[3] = (r->d[3] >> n) + (r->d[4] << (32 - n)); + r->d[4] = (r->d[4] >> n) + (r->d[5] << (32 - n)); + r->d[5] = (r->d[5] >> n) + (r->d[6] << (32 - n)); + r->d[6] = (r->d[6] >> n) + (r->d[7] << (32 - n)); + r->d[7] = (r->d[7] >> n); + return ret; +} + +static void secp256k1_scalar_sqr(secp256k1_scalar *r, const secp256k1_scalar *a) { + uint32_t l[16]; + secp256k1_scalar_sqr_512(l, a); + secp256k1_scalar_reduce_512(r, l); +} + +#ifdef USE_ENDOMORPHISM +static void secp256k1_scalar_split_128(secp256k1_scalar *r1, secp256k1_scalar *r2, const secp256k1_scalar *a) { + r1->d[0] = a->d[0]; + r1->d[1] = a->d[1]; + r1->d[2] = a->d[2]; + r1->d[3] = a->d[3]; + r1->d[4] = 0; + r1->d[5] = 0; + r1->d[6] = 0; + r1->d[7] = 0; + r2->d[0] = a->d[4]; + r2->d[1] = a->d[5]; + r2->d[2] = a->d[6]; + r2->d[3] = a->d[7]; + r2->d[4] = 0; + r2->d[5] = 0; + r2->d[6] = 0; + r2->d[7] = 0; +} +#endif + +SECP256K1_INLINE static int secp256k1_scalar_eq(const secp256k1_scalar *a, const secp256k1_scalar *b) { + return ((a->d[0] ^ b->d[0]) | (a->d[1] ^ b->d[1]) | (a->d[2] ^ b->d[2]) | (a->d[3] ^ b->d[3]) | (a->d[4] ^ b->d[4]) | (a->d[5] ^ b->d[5]) | (a->d[6] ^ b->d[6]) | (a->d[7] ^ b->d[7])) == 0; +} + +SECP256K1_INLINE static void secp256k1_scalar_mul_shift_var(secp256k1_scalar *r, const secp256k1_scalar *a, const secp256k1_scalar *b, unsigned int shift) { + uint32_t l[16]; + unsigned int shiftlimbs; + unsigned int shiftlow; + unsigned int shifthigh; + VERIFY_CHECK(shift >= 256); + secp256k1_scalar_mul_512(l, a, b); + shiftlimbs = shift >> 5; + shiftlow = shift & 0x1F; + shifthigh = 32 - shiftlow; + r->d[0] = shift < 512 ? (l[0 + shiftlimbs] >> shiftlow | (shift < 480 && shiftlow ? (l[1 + shiftlimbs] << shifthigh) : 0)) : 0; + r->d[1] = shift < 480 ? (l[1 + shiftlimbs] >> shiftlow | (shift < 448 && shiftlow ? (l[2 + shiftlimbs] << shifthigh) : 0)) : 0; + r->d[2] = shift < 448 ? (l[2 + shiftlimbs] >> shiftlow | (shift < 416 && shiftlow ? (l[3 + shiftlimbs] << shifthigh) : 0)) : 0; + r->d[3] = shift < 416 ? (l[3 + shiftlimbs] >> shiftlow | (shift < 384 && shiftlow ? (l[4 + shiftlimbs] << shifthigh) : 0)) : 0; + r->d[4] = shift < 384 ? (l[4 + shiftlimbs] >> shiftlow | (shift < 352 && shiftlow ? (l[5 + shiftlimbs] << shifthigh) : 0)) : 0; + r->d[5] = shift < 352 ? (l[5 + shiftlimbs] >> shiftlow | (shift < 320 && shiftlow ? (l[6 + shiftlimbs] << shifthigh) : 0)) : 0; + r->d[6] = shift < 320 ? (l[6 + shiftlimbs] >> shiftlow | (shift < 288 && shiftlow ? (l[7 + shiftlimbs] << shifthigh) : 0)) : 0; + r->d[7] = shift < 288 ? (l[7 + shiftlimbs] >> shiftlow) : 0; + secp256k1_scalar_cadd_bit(r, 0, (l[(shift - 1) >> 5] >> ((shift - 1) & 0x1f)) & 1); +} + +#endif /* SECP256K1_SCALAR_REPR_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/scalar_impl.h b/node_modules/secp256k1/src/secp256k1/src/scalar_impl.h new file mode 100644 index 0000000..6b336d9 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/scalar_impl.h @@ -0,0 +1,333 @@ +/********************************************************************** + * Copyright (c) 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_SCALAR_IMPL_H +#define SECP256K1_SCALAR_IMPL_H + +#include "scalar.h" +#include "util.h" + +#if defined HAVE_CONFIG_H +#include "libsecp256k1-config.h" +#endif + +#if defined(EXHAUSTIVE_TEST_ORDER) +#include "scalar_low_impl.h" +#elif defined(USE_SCALAR_4X64) +#include "scalar_4x64_impl.h" +#elif defined(USE_SCALAR_8X32) +#include "scalar_8x32_impl.h" +#else +#error "Please select scalar implementation" +#endif + +#ifndef USE_NUM_NONE +static void secp256k1_scalar_get_num(secp256k1_num *r, const secp256k1_scalar *a) { + unsigned char c[32]; + secp256k1_scalar_get_b32(c, a); + secp256k1_num_set_bin(r, c, 32); +} + +/** secp256k1 curve order, see secp256k1_ecdsa_const_order_as_fe in ecdsa_impl.h */ +static void secp256k1_scalar_order_get_num(secp256k1_num *r) { +#if defined(EXHAUSTIVE_TEST_ORDER) + static const unsigned char order[32] = { + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,EXHAUSTIVE_TEST_ORDER + }; +#else + static const unsigned char order[32] = { + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFE, + 0xBA,0xAE,0xDC,0xE6,0xAF,0x48,0xA0,0x3B, + 0xBF,0xD2,0x5E,0x8C,0xD0,0x36,0x41,0x41 + }; +#endif + secp256k1_num_set_bin(r, order, 32); +} +#endif + +static void secp256k1_scalar_inverse(secp256k1_scalar *r, const secp256k1_scalar *x) { +#if defined(EXHAUSTIVE_TEST_ORDER) + int i; + *r = 0; + for (i = 0; i < EXHAUSTIVE_TEST_ORDER; i++) + if ((i * *x) % EXHAUSTIVE_TEST_ORDER == 1) + *r = i; + /* If this VERIFY_CHECK triggers we were given a noninvertible scalar (and thus + * have a composite group order; fix it in exhaustive_tests.c). */ + VERIFY_CHECK(*r != 0); +} +#else + secp256k1_scalar *t; + int i; + /* First compute xN as x ^ (2^N - 1) for some values of N, + * and uM as x ^ M for some values of M. */ + secp256k1_scalar x2, x3, x6, x8, x14, x28, x56, x112, x126; + secp256k1_scalar u2, u5, u9, u11, u13; + + secp256k1_scalar_sqr(&u2, x); + secp256k1_scalar_mul(&x2, &u2, x); + secp256k1_scalar_mul(&u5, &u2, &x2); + secp256k1_scalar_mul(&x3, &u5, &u2); + secp256k1_scalar_mul(&u9, &x3, &u2); + secp256k1_scalar_mul(&u11, &u9, &u2); + secp256k1_scalar_mul(&u13, &u11, &u2); + + secp256k1_scalar_sqr(&x6, &u13); + secp256k1_scalar_sqr(&x6, &x6); + secp256k1_scalar_mul(&x6, &x6, &u11); + + secp256k1_scalar_sqr(&x8, &x6); + secp256k1_scalar_sqr(&x8, &x8); + secp256k1_scalar_mul(&x8, &x8, &x2); + + secp256k1_scalar_sqr(&x14, &x8); + for (i = 0; i < 5; i++) { + secp256k1_scalar_sqr(&x14, &x14); + } + secp256k1_scalar_mul(&x14, &x14, &x6); + + secp256k1_scalar_sqr(&x28, &x14); + for (i = 0; i < 13; i++) { + secp256k1_scalar_sqr(&x28, &x28); + } + secp256k1_scalar_mul(&x28, &x28, &x14); + + secp256k1_scalar_sqr(&x56, &x28); + for (i = 0; i < 27; i++) { + secp256k1_scalar_sqr(&x56, &x56); + } + secp256k1_scalar_mul(&x56, &x56, &x28); + + secp256k1_scalar_sqr(&x112, &x56); + for (i = 0; i < 55; i++) { + secp256k1_scalar_sqr(&x112, &x112); + } + secp256k1_scalar_mul(&x112, &x112, &x56); + + secp256k1_scalar_sqr(&x126, &x112); + for (i = 0; i < 13; i++) { + secp256k1_scalar_sqr(&x126, &x126); + } + secp256k1_scalar_mul(&x126, &x126, &x14); + + /* Then accumulate the final result (t starts at x126). */ + t = &x126; + for (i = 0; i < 3; i++) { + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u5); /* 101 */ + for (i = 0; i < 4; i++) { /* 0 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &x3); /* 111 */ + for (i = 0; i < 4; i++) { /* 0 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u5); /* 101 */ + for (i = 0; i < 5; i++) { /* 0 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u11); /* 1011 */ + for (i = 0; i < 4; i++) { + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u11); /* 1011 */ + for (i = 0; i < 4; i++) { /* 0 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &x3); /* 111 */ + for (i = 0; i < 5; i++) { /* 00 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &x3); /* 111 */ + for (i = 0; i < 6; i++) { /* 00 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u13); /* 1101 */ + for (i = 0; i < 4; i++) { /* 0 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u5); /* 101 */ + for (i = 0; i < 3; i++) { + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &x3); /* 111 */ + for (i = 0; i < 5; i++) { /* 0 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u9); /* 1001 */ + for (i = 0; i < 6; i++) { /* 000 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u5); /* 101 */ + for (i = 0; i < 10; i++) { /* 0000000 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &x3); /* 111 */ + for (i = 0; i < 4; i++) { /* 0 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &x3); /* 111 */ + for (i = 0; i < 9; i++) { /* 0 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &x8); /* 11111111 */ + for (i = 0; i < 5; i++) { /* 0 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u9); /* 1001 */ + for (i = 0; i < 6; i++) { /* 00 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u11); /* 1011 */ + for (i = 0; i < 4; i++) { + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u13); /* 1101 */ + for (i = 0; i < 5; i++) { + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &x2); /* 11 */ + for (i = 0; i < 6; i++) { /* 00 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u13); /* 1101 */ + for (i = 0; i < 10; i++) { /* 000000 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u13); /* 1101 */ + for (i = 0; i < 4; i++) { + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, &u9); /* 1001 */ + for (i = 0; i < 6; i++) { /* 00000 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(t, t, x); /* 1 */ + for (i = 0; i < 8; i++) { /* 00 */ + secp256k1_scalar_sqr(t, t); + } + secp256k1_scalar_mul(r, t, &x6); /* 111111 */ +} + +SECP256K1_INLINE static int secp256k1_scalar_is_even(const secp256k1_scalar *a) { + return !(a->d[0] & 1); +} +#endif + +static void secp256k1_scalar_inverse_var(secp256k1_scalar *r, const secp256k1_scalar *x) { +#if defined(USE_SCALAR_INV_BUILTIN) + secp256k1_scalar_inverse(r, x); +#elif defined(USE_SCALAR_INV_NUM) + unsigned char b[32]; + secp256k1_num n, m; + secp256k1_scalar t = *x; + secp256k1_scalar_get_b32(b, &t); + secp256k1_num_set_bin(&n, b, 32); + secp256k1_scalar_order_get_num(&m); + secp256k1_num_mod_inverse(&n, &n, &m); + secp256k1_num_get_bin(b, 32, &n); + secp256k1_scalar_set_b32(r, b, NULL); + /* Verify that the inverse was computed correctly, without GMP code. */ + secp256k1_scalar_mul(&t, &t, r); + CHECK(secp256k1_scalar_is_one(&t)); +#else +#error "Please select scalar inverse implementation" +#endif +} + +#ifdef USE_ENDOMORPHISM +#if defined(EXHAUSTIVE_TEST_ORDER) +/** + * Find k1 and k2 given k, such that k1 + k2 * lambda == k mod n; unlike in the + * full case we don't bother making k1 and k2 be small, we just want them to be + * nontrivial to get full test coverage for the exhaustive tests. We therefore + * (arbitrarily) set k2 = k + 5 and k1 = k - k2 * lambda. + */ +static void secp256k1_scalar_split_lambda(secp256k1_scalar *r1, secp256k1_scalar *r2, const secp256k1_scalar *a) { + *r2 = (*a + 5) % EXHAUSTIVE_TEST_ORDER; + *r1 = (*a + (EXHAUSTIVE_TEST_ORDER - *r2) * EXHAUSTIVE_TEST_LAMBDA) % EXHAUSTIVE_TEST_ORDER; +} +#else +/** + * The Secp256k1 curve has an endomorphism, where lambda * (x, y) = (beta * x, y), where + * lambda is {0x53,0x63,0xad,0x4c,0xc0,0x5c,0x30,0xe0,0xa5,0x26,0x1c,0x02,0x88,0x12,0x64,0x5a, + * 0x12,0x2e,0x22,0xea,0x20,0x81,0x66,0x78,0xdf,0x02,0x96,0x7c,0x1b,0x23,0xbd,0x72} + * + * "Guide to Elliptic Curve Cryptography" (Hankerson, Menezes, Vanstone) gives an algorithm + * (algorithm 3.74) to find k1 and k2 given k, such that k1 + k2 * lambda == k mod n, and k1 + * and k2 have a small size. + * It relies on constants a1, b1, a2, b2. These constants for the value of lambda above are: + * + * - a1 = {0x30,0x86,0xd2,0x21,0xa7,0xd4,0x6b,0xcd,0xe8,0x6c,0x90,0xe4,0x92,0x84,0xeb,0x15} + * - b1 = -{0xe4,0x43,0x7e,0xd6,0x01,0x0e,0x88,0x28,0x6f,0x54,0x7f,0xa9,0x0a,0xbf,0xe4,0xc3} + * - a2 = {0x01,0x14,0xca,0x50,0xf7,0xa8,0xe2,0xf3,0xf6,0x57,0xc1,0x10,0x8d,0x9d,0x44,0xcf,0xd8} + * - b2 = {0x30,0x86,0xd2,0x21,0xa7,0xd4,0x6b,0xcd,0xe8,0x6c,0x90,0xe4,0x92,0x84,0xeb,0x15} + * + * The algorithm then computes c1 = round(b1 * k / n) and c2 = round(b2 * k / n), and gives + * k1 = k - (c1*a1 + c2*a2) and k2 = -(c1*b1 + c2*b2). Instead, we use modular arithmetic, and + * compute k1 as k - k2 * lambda, avoiding the need for constants a1 and a2. + * + * g1, g2 are precomputed constants used to replace division with a rounded multiplication + * when decomposing the scalar for an endomorphism-based point multiplication. + * + * The possibility of using precomputed estimates is mentioned in "Guide to Elliptic Curve + * Cryptography" (Hankerson, Menezes, Vanstone) in section 3.5. + * + * The derivation is described in the paper "Efficient Software Implementation of Public-Key + * Cryptography on Sensor Networks Using the MSP430X Microcontroller" (Gouvea, Oliveira, Lopez), + * Section 4.3 (here we use a somewhat higher-precision estimate): + * d = a1*b2 - b1*a2 + * g1 = round((2^272)*b2/d) + * g2 = round((2^272)*b1/d) + * + * (Note that 'd' is also equal to the curve order here because [a1,b1] and [a2,b2] are found + * as outputs of the Extended Euclidean Algorithm on inputs 'order' and 'lambda'). + * + * The function below splits a in r1 and r2, such that r1 + lambda * r2 == a (mod order). + */ + +static void secp256k1_scalar_split_lambda(secp256k1_scalar *r1, secp256k1_scalar *r2, const secp256k1_scalar *a) { + secp256k1_scalar c1, c2; + static const secp256k1_scalar minus_lambda = SECP256K1_SCALAR_CONST( + 0xAC9C52B3UL, 0x3FA3CF1FUL, 0x5AD9E3FDUL, 0x77ED9BA4UL, + 0xA880B9FCUL, 0x8EC739C2UL, 0xE0CFC810UL, 0xB51283CFUL + ); + static const secp256k1_scalar minus_b1 = SECP256K1_SCALAR_CONST( + 0x00000000UL, 0x00000000UL, 0x00000000UL, 0x00000000UL, + 0xE4437ED6UL, 0x010E8828UL, 0x6F547FA9UL, 0x0ABFE4C3UL + ); + static const secp256k1_scalar minus_b2 = SECP256K1_SCALAR_CONST( + 0xFFFFFFFFUL, 0xFFFFFFFFUL, 0xFFFFFFFFUL, 0xFFFFFFFEUL, + 0x8A280AC5UL, 0x0774346DUL, 0xD765CDA8UL, 0x3DB1562CUL + ); + static const secp256k1_scalar g1 = SECP256K1_SCALAR_CONST( + 0x00000000UL, 0x00000000UL, 0x00000000UL, 0x00003086UL, + 0xD221A7D4UL, 0x6BCDE86CUL, 0x90E49284UL, 0xEB153DABUL + ); + static const secp256k1_scalar g2 = SECP256K1_SCALAR_CONST( + 0x00000000UL, 0x00000000UL, 0x00000000UL, 0x0000E443UL, + 0x7ED6010EUL, 0x88286F54UL, 0x7FA90ABFUL, 0xE4C42212UL + ); + VERIFY_CHECK(r1 != a); + VERIFY_CHECK(r2 != a); + /* these _var calls are constant time since the shift amount is constant */ + secp256k1_scalar_mul_shift_var(&c1, a, &g1, 272); + secp256k1_scalar_mul_shift_var(&c2, a, &g2, 272); + secp256k1_scalar_mul(&c1, &c1, &minus_b1); + secp256k1_scalar_mul(&c2, &c2, &minus_b2); + secp256k1_scalar_add(r2, &c1, &c2); + secp256k1_scalar_mul(r1, r2, &minus_lambda); + secp256k1_scalar_add(r1, r1, a); +} +#endif +#endif + +#endif /* SECP256K1_SCALAR_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/scalar_low.h b/node_modules/secp256k1/src/secp256k1/src/scalar_low.h new file mode 100644 index 0000000..5836feb --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/scalar_low.h @@ -0,0 +1,15 @@ +/********************************************************************** + * Copyright (c) 2015 Andrew Poelstra * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_SCALAR_REPR_H +#define SECP256K1_SCALAR_REPR_H + +#include + +/** A scalar modulo the group order of the secp256k1 curve. */ +typedef uint32_t secp256k1_scalar; + +#endif /* SECP256K1_SCALAR_REPR_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/scalar_low_impl.h b/node_modules/secp256k1/src/secp256k1/src/scalar_low_impl.h new file mode 100644 index 0000000..910ce3f --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/scalar_low_impl.h @@ -0,0 +1,117 @@ +/********************************************************************** + * Copyright (c) 2015 Andrew Poelstra * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_SCALAR_REPR_IMPL_H +#define SECP256K1_SCALAR_REPR_IMPL_H + +#include "scalar.h" + +#include + +SECP256K1_INLINE static int secp256k1_scalar_is_even(const secp256k1_scalar *a) { + return !(*a & 1); +} + +SECP256K1_INLINE static void secp256k1_scalar_clear(secp256k1_scalar *r) { *r = 0; } +SECP256K1_INLINE static void secp256k1_scalar_set_int(secp256k1_scalar *r, unsigned int v) { *r = v; } + +SECP256K1_INLINE static unsigned int secp256k1_scalar_get_bits(const secp256k1_scalar *a, unsigned int offset, unsigned int count) { + if (offset < 32) + return ((*a >> offset) & ((((uint32_t)1) << count) - 1)); + else + return 0; +} + +SECP256K1_INLINE static unsigned int secp256k1_scalar_get_bits_var(const secp256k1_scalar *a, unsigned int offset, unsigned int count) { + return secp256k1_scalar_get_bits(a, offset, count); +} + +SECP256K1_INLINE static int secp256k1_scalar_check_overflow(const secp256k1_scalar *a) { return *a >= EXHAUSTIVE_TEST_ORDER; } + +static int secp256k1_scalar_add(secp256k1_scalar *r, const secp256k1_scalar *a, const secp256k1_scalar *b) { + *r = (*a + *b) % EXHAUSTIVE_TEST_ORDER; + return *r < *b; +} + +static void secp256k1_scalar_cadd_bit(secp256k1_scalar *r, unsigned int bit, int flag) { + if (flag && bit < 32) + *r += ((uint32_t)1 << bit); +#ifdef VERIFY + VERIFY_CHECK(bit < 32); + /* Verify that adding (1 << bit) will not overflow any in-range scalar *r by overflowing the underlying uint32_t. */ + VERIFY_CHECK(((uint32_t)1 << bit) - 1 <= UINT32_MAX - EXHAUSTIVE_TEST_ORDER); + VERIFY_CHECK(secp256k1_scalar_check_overflow(r) == 0); +#endif +} + +static void secp256k1_scalar_set_b32(secp256k1_scalar *r, const unsigned char *b32, int *overflow) { + const int base = 0x100 % EXHAUSTIVE_TEST_ORDER; + int i; + *r = 0; + for (i = 0; i < 32; i++) { + *r = ((*r * base) + b32[i]) % EXHAUSTIVE_TEST_ORDER; + } + /* just deny overflow, it basically always happens */ + if (overflow) *overflow = 0; +} + +static void secp256k1_scalar_get_b32(unsigned char *bin, const secp256k1_scalar* a) { + memset(bin, 0, 32); + bin[28] = *a >> 24; bin[29] = *a >> 16; bin[30] = *a >> 8; bin[31] = *a; +} + +SECP256K1_INLINE static int secp256k1_scalar_is_zero(const secp256k1_scalar *a) { + return *a == 0; +} + +static void secp256k1_scalar_negate(secp256k1_scalar *r, const secp256k1_scalar *a) { + if (*a == 0) { + *r = 0; + } else { + *r = EXHAUSTIVE_TEST_ORDER - *a; + } +} + +SECP256K1_INLINE static int secp256k1_scalar_is_one(const secp256k1_scalar *a) { + return *a == 1; +} + +static int secp256k1_scalar_is_high(const secp256k1_scalar *a) { + return *a > EXHAUSTIVE_TEST_ORDER / 2; +} + +static int secp256k1_scalar_cond_negate(secp256k1_scalar *r, int flag) { + if (flag) secp256k1_scalar_negate(r, r); + return flag ? -1 : 1; +} + +static void secp256k1_scalar_mul(secp256k1_scalar *r, const secp256k1_scalar *a, const secp256k1_scalar *b) { + *r = (*a * *b) % EXHAUSTIVE_TEST_ORDER; +} + +static int secp256k1_scalar_shr_int(secp256k1_scalar *r, int n) { + int ret; + VERIFY_CHECK(n > 0); + VERIFY_CHECK(n < 16); + ret = *r & ((1 << n) - 1); + *r >>= n; + return ret; +} + +static void secp256k1_scalar_sqr(secp256k1_scalar *r, const secp256k1_scalar *a) { + *r = (*a * *a) % EXHAUSTIVE_TEST_ORDER; +} + +static void secp256k1_scalar_split_128(secp256k1_scalar *r1, secp256k1_scalar *r2, const secp256k1_scalar *a) { + *r1 = *a; + *r2 = 0; +} + +SECP256K1_INLINE static int secp256k1_scalar_eq(const secp256k1_scalar *a, const secp256k1_scalar *b) { + return *a == *b; +} + +#endif /* SECP256K1_SCALAR_REPR_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/scratch.h b/node_modules/secp256k1/src/secp256k1/src/scratch.h new file mode 100644 index 0000000..77b35d1 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/scratch.h @@ -0,0 +1,42 @@ +/********************************************************************** + * Copyright (c) 2017 Andrew Poelstra * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef _SECP256K1_SCRATCH_ +#define _SECP256K1_SCRATCH_ + +/* The typedef is used internally; the struct name is used in the public API + * (where it is exposed as a different typedef) */ +typedef struct secp256k1_scratch_space_struct { + /** guard against interpreting this object as other types */ + unsigned char magic[8]; + /** actual allocated data */ + void *data; + /** amount that has been allocated (i.e. `data + offset` is the next + * available pointer) */ + size_t alloc_size; + /** maximum size available to allocate */ + size_t max_size; +} secp256k1_scratch; + +static secp256k1_scratch* secp256k1_scratch_create(const secp256k1_callback* error_callback, size_t max_size); + +static void secp256k1_scratch_destroy(const secp256k1_callback* error_callback, secp256k1_scratch* scratch); + +/** Returns an opaque object used to "checkpoint" a scratch space. Used + * with `secp256k1_scratch_apply_checkpoint` to undo allocations. */ +static size_t secp256k1_scratch_checkpoint(const secp256k1_callback* error_callback, const secp256k1_scratch* scratch); + +/** Applies a check point received from `secp256k1_scratch_checkpoint`, + * undoing all allocations since that point. */ +static void secp256k1_scratch_apply_checkpoint(const secp256k1_callback* error_callback, secp256k1_scratch* scratch, size_t checkpoint); + +/** Returns the maximum allocation the scratch space will allow */ +static size_t secp256k1_scratch_max_allocation(const secp256k1_callback* error_callback, const secp256k1_scratch* scratch, size_t n_objects); + +/** Returns a pointer into the most recently allocated frame, or NULL if there is insufficient available space */ +static void *secp256k1_scratch_alloc(const secp256k1_callback* error_callback, secp256k1_scratch* scratch, size_t n); + +#endif diff --git a/node_modules/secp256k1/src/secp256k1/src/scratch_impl.h b/node_modules/secp256k1/src/secp256k1/src/scratch_impl.h new file mode 100644 index 0000000..4cee700 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/scratch_impl.h @@ -0,0 +1,88 @@ +/********************************************************************** + * Copyright (c) 2017 Andrew Poelstra * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef _SECP256K1_SCRATCH_IMPL_H_ +#define _SECP256K1_SCRATCH_IMPL_H_ + +#include "util.h" +#include "scratch.h" + +static secp256k1_scratch* secp256k1_scratch_create(const secp256k1_callback* error_callback, size_t size) { + const size_t base_alloc = ((sizeof(secp256k1_scratch) + ALIGNMENT - 1) / ALIGNMENT) * ALIGNMENT; + void *alloc = checked_malloc(error_callback, base_alloc + size); + secp256k1_scratch* ret = (secp256k1_scratch *)alloc; + if (ret != NULL) { + memset(ret, 0, sizeof(*ret)); + memcpy(ret->magic, "scratch", 8); + ret->data = (void *) ((char *) alloc + base_alloc); + ret->max_size = size; + } + return ret; +} + +static void secp256k1_scratch_destroy(const secp256k1_callback* error_callback, secp256k1_scratch* scratch) { + if (scratch != NULL) { + VERIFY_CHECK(scratch->alloc_size == 0); /* all checkpoints should be applied */ + if (memcmp(scratch->magic, "scratch", 8) != 0) { + secp256k1_callback_call(error_callback, "invalid scratch space"); + return; + } + memset(scratch->magic, 0, sizeof(scratch->magic)); + free(scratch); + } +} + +static size_t secp256k1_scratch_checkpoint(const secp256k1_callback* error_callback, const secp256k1_scratch* scratch) { + if (memcmp(scratch->magic, "scratch", 8) != 0) { + secp256k1_callback_call(error_callback, "invalid scratch space"); + return 0; + } + return scratch->alloc_size; +} + +static void secp256k1_scratch_apply_checkpoint(const secp256k1_callback* error_callback, secp256k1_scratch* scratch, size_t checkpoint) { + if (memcmp(scratch->magic, "scratch", 8) != 0) { + secp256k1_callback_call(error_callback, "invalid scratch space"); + return; + } + if (checkpoint > scratch->alloc_size) { + secp256k1_callback_call(error_callback, "invalid checkpoint"); + return; + } + scratch->alloc_size = checkpoint; +} + +static size_t secp256k1_scratch_max_allocation(const secp256k1_callback* error_callback, const secp256k1_scratch* scratch, size_t objects) { + if (memcmp(scratch->magic, "scratch", 8) != 0) { + secp256k1_callback_call(error_callback, "invalid scratch space"); + return 0; + } + if (scratch->max_size - scratch->alloc_size <= objects * (ALIGNMENT - 1)) { + return 0; + } + return scratch->max_size - scratch->alloc_size - objects * (ALIGNMENT - 1); +} + +static void *secp256k1_scratch_alloc(const secp256k1_callback* error_callback, secp256k1_scratch* scratch, size_t size) { + void *ret; + size = ROUND_TO_ALIGN(size); + + if (memcmp(scratch->magic, "scratch", 8) != 0) { + secp256k1_callback_call(error_callback, "invalid scratch space"); + return NULL; + } + + if (size > scratch->max_size - scratch->alloc_size) { + return NULL; + } + ret = (void *) ((char *) scratch->data + scratch->alloc_size); + memset(ret, 0, size); + scratch->alloc_size += size; + + return ret; +} + +#endif diff --git a/node_modules/secp256k1/src/secp256k1/src/secp256k1.c b/node_modules/secp256k1/src/secp256k1/src/secp256k1.c new file mode 100644 index 0000000..a3f446e --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/secp256k1.c @@ -0,0 +1,690 @@ +/********************************************************************** + * Copyright (c) 2013-2015 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#include "include/secp256k1.h" +#include "include/secp256k1_preallocated.h" + +#include "util.h" +#include "num_impl.h" +#include "field_impl.h" +#include "scalar_impl.h" +#include "group_impl.h" +#include "ecmult_impl.h" +#include "ecmult_const_impl.h" +#include "ecmult_gen_impl.h" +#include "ecdsa_impl.h" +#include "eckey_impl.h" +#include "hash_impl.h" +#include "scratch_impl.h" + +#define ARG_CHECK(cond) do { \ + if (EXPECT(!(cond), 0)) { \ + secp256k1_callback_call(&ctx->illegal_callback, #cond); \ + return 0; \ + } \ +} while(0) + +#define ARG_CHECK_NO_RETURN(cond) do { \ + if (EXPECT(!(cond), 0)) { \ + secp256k1_callback_call(&ctx->illegal_callback, #cond); \ + } \ +} while(0) + +#ifndef USE_EXTERNAL_DEFAULT_CALLBACKS +#include +#include +static void secp256k1_default_illegal_callback_fn(const char* str, void* data) { + (void)data; + fprintf(stderr, "[libsecp256k1] illegal argument: %s\n", str); + abort(); +} +static void secp256k1_default_error_callback_fn(const char* str, void* data) { + (void)data; + fprintf(stderr, "[libsecp256k1] internal consistency check failed: %s\n", str); + abort(); +} +#else +void secp256k1_default_illegal_callback_fn(const char* str, void* data); +void secp256k1_default_error_callback_fn(const char* str, void* data); +#endif + +static const secp256k1_callback default_illegal_callback = { + secp256k1_default_illegal_callback_fn, + NULL +}; + +static const secp256k1_callback default_error_callback = { + secp256k1_default_error_callback_fn, + NULL +}; + +struct secp256k1_context_struct { + secp256k1_ecmult_context ecmult_ctx; + secp256k1_ecmult_gen_context ecmult_gen_ctx; + secp256k1_callback illegal_callback; + secp256k1_callback error_callback; +}; + +static const secp256k1_context secp256k1_context_no_precomp_ = { + { 0 }, + { 0 }, + { secp256k1_default_illegal_callback_fn, 0 }, + { secp256k1_default_error_callback_fn, 0 } +}; +const secp256k1_context *secp256k1_context_no_precomp = &secp256k1_context_no_precomp_; + +size_t secp256k1_context_preallocated_size(unsigned int flags) { + size_t ret = ROUND_TO_ALIGN(sizeof(secp256k1_context)); + + if (EXPECT((flags & SECP256K1_FLAGS_TYPE_MASK) != SECP256K1_FLAGS_TYPE_CONTEXT, 0)) { + secp256k1_callback_call(&default_illegal_callback, + "Invalid flags"); + return 0; + } + + if (flags & SECP256K1_FLAGS_BIT_CONTEXT_SIGN) { + ret += SECP256K1_ECMULT_GEN_CONTEXT_PREALLOCATED_SIZE; + } + if (flags & SECP256K1_FLAGS_BIT_CONTEXT_VERIFY) { + ret += SECP256K1_ECMULT_CONTEXT_PREALLOCATED_SIZE; + } + return ret; +} + +size_t secp256k1_context_preallocated_clone_size(const secp256k1_context* ctx) { + size_t ret = ROUND_TO_ALIGN(sizeof(secp256k1_context)); + VERIFY_CHECK(ctx != NULL); + if (secp256k1_ecmult_gen_context_is_built(&ctx->ecmult_gen_ctx)) { + ret += SECP256K1_ECMULT_GEN_CONTEXT_PREALLOCATED_SIZE; + } + if (secp256k1_ecmult_context_is_built(&ctx->ecmult_ctx)) { + ret += SECP256K1_ECMULT_CONTEXT_PREALLOCATED_SIZE; + } + return ret; +} + +secp256k1_context* secp256k1_context_preallocated_create(void* prealloc, unsigned int flags) { + void* const base = prealloc; + size_t prealloc_size; + secp256k1_context* ret; + + VERIFY_CHECK(prealloc != NULL); + prealloc_size = secp256k1_context_preallocated_size(flags); + ret = (secp256k1_context*)manual_alloc(&prealloc, sizeof(secp256k1_context), base, prealloc_size); + ret->illegal_callback = default_illegal_callback; + ret->error_callback = default_error_callback; + + if (EXPECT((flags & SECP256K1_FLAGS_TYPE_MASK) != SECP256K1_FLAGS_TYPE_CONTEXT, 0)) { + secp256k1_callback_call(&ret->illegal_callback, + "Invalid flags"); + return NULL; + } + + secp256k1_ecmult_context_init(&ret->ecmult_ctx); + secp256k1_ecmult_gen_context_init(&ret->ecmult_gen_ctx); + + if (flags & SECP256K1_FLAGS_BIT_CONTEXT_SIGN) { + secp256k1_ecmult_gen_context_build(&ret->ecmult_gen_ctx, &prealloc); + } + if (flags & SECP256K1_FLAGS_BIT_CONTEXT_VERIFY) { + secp256k1_ecmult_context_build(&ret->ecmult_ctx, &prealloc); + } + + return (secp256k1_context*) ret; +} + +secp256k1_context* secp256k1_context_create(unsigned int flags) { + size_t const prealloc_size = secp256k1_context_preallocated_size(flags); + secp256k1_context* ctx = (secp256k1_context*)checked_malloc(&default_error_callback, prealloc_size); + if (EXPECT(secp256k1_context_preallocated_create(ctx, flags) == NULL, 0)) { + free(ctx); + return NULL; + } + + return ctx; +} + +secp256k1_context* secp256k1_context_preallocated_clone(const secp256k1_context* ctx, void* prealloc) { + size_t prealloc_size; + secp256k1_context* ret; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(prealloc != NULL); + + prealloc_size = secp256k1_context_preallocated_clone_size(ctx); + ret = (secp256k1_context*)prealloc; + memcpy(ret, ctx, prealloc_size); + secp256k1_ecmult_gen_context_finalize_memcpy(&ret->ecmult_gen_ctx, &ctx->ecmult_gen_ctx); + secp256k1_ecmult_context_finalize_memcpy(&ret->ecmult_ctx, &ctx->ecmult_ctx); + return ret; +} + +secp256k1_context* secp256k1_context_clone(const secp256k1_context* ctx) { + secp256k1_context* ret; + size_t prealloc_size; + + VERIFY_CHECK(ctx != NULL); + prealloc_size = secp256k1_context_preallocated_clone_size(ctx); + ret = (secp256k1_context*)checked_malloc(&ctx->error_callback, prealloc_size); + ret = secp256k1_context_preallocated_clone(ctx, ret); + return ret; +} + +void secp256k1_context_preallocated_destroy(secp256k1_context* ctx) { + ARG_CHECK_NO_RETURN(ctx != secp256k1_context_no_precomp); + if (ctx != NULL) { + secp256k1_ecmult_context_clear(&ctx->ecmult_ctx); + secp256k1_ecmult_gen_context_clear(&ctx->ecmult_gen_ctx); + } +} + +void secp256k1_context_destroy(secp256k1_context* ctx) { + if (ctx != NULL) { + secp256k1_context_preallocated_destroy(ctx); + free(ctx); + } +} + +void secp256k1_context_set_illegal_callback(secp256k1_context* ctx, void (*fun)(const char* message, void* data), const void* data) { + ARG_CHECK_NO_RETURN(ctx != secp256k1_context_no_precomp); + if (fun == NULL) { + fun = secp256k1_default_illegal_callback_fn; + } + ctx->illegal_callback.fn = fun; + ctx->illegal_callback.data = data; +} + +void secp256k1_context_set_error_callback(secp256k1_context* ctx, void (*fun)(const char* message, void* data), const void* data) { + ARG_CHECK_NO_RETURN(ctx != secp256k1_context_no_precomp); + if (fun == NULL) { + fun = secp256k1_default_error_callback_fn; + } + ctx->error_callback.fn = fun; + ctx->error_callback.data = data; +} + +secp256k1_scratch_space* secp256k1_scratch_space_create(const secp256k1_context* ctx, size_t max_size) { + VERIFY_CHECK(ctx != NULL); + return secp256k1_scratch_create(&ctx->error_callback, max_size); +} + +void secp256k1_scratch_space_destroy(const secp256k1_context *ctx, secp256k1_scratch_space* scratch) { + VERIFY_CHECK(ctx != NULL); + secp256k1_scratch_destroy(&ctx->error_callback, scratch); +} + +static int secp256k1_pubkey_load(const secp256k1_context* ctx, secp256k1_ge* ge, const secp256k1_pubkey* pubkey) { + if (sizeof(secp256k1_ge_storage) == 64) { + /* When the secp256k1_ge_storage type is exactly 64 byte, use its + * representation inside secp256k1_pubkey, as conversion is very fast. + * Note that secp256k1_pubkey_save must use the same representation. */ + secp256k1_ge_storage s; + memcpy(&s, &pubkey->data[0], sizeof(s)); + secp256k1_ge_from_storage(ge, &s); + } else { + /* Otherwise, fall back to 32-byte big endian for X and Y. */ + secp256k1_fe x, y; + secp256k1_fe_set_b32(&x, pubkey->data); + secp256k1_fe_set_b32(&y, pubkey->data + 32); + secp256k1_ge_set_xy(ge, &x, &y); + } + ARG_CHECK(!secp256k1_fe_is_zero(&ge->x)); + return 1; +} + +static void secp256k1_pubkey_save(secp256k1_pubkey* pubkey, secp256k1_ge* ge) { + if (sizeof(secp256k1_ge_storage) == 64) { + secp256k1_ge_storage s; + secp256k1_ge_to_storage(&s, ge); + memcpy(&pubkey->data[0], &s, sizeof(s)); + } else { + VERIFY_CHECK(!secp256k1_ge_is_infinity(ge)); + secp256k1_fe_normalize_var(&ge->x); + secp256k1_fe_normalize_var(&ge->y); + secp256k1_fe_get_b32(pubkey->data, &ge->x); + secp256k1_fe_get_b32(pubkey->data + 32, &ge->y); + } +} + +int secp256k1_ec_pubkey_parse(const secp256k1_context* ctx, secp256k1_pubkey* pubkey, const unsigned char *input, size_t inputlen) { + secp256k1_ge Q; + + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(pubkey != NULL); + memset(pubkey, 0, sizeof(*pubkey)); + ARG_CHECK(input != NULL); + if (!secp256k1_eckey_pubkey_parse(&Q, input, inputlen)) { + return 0; + } + secp256k1_pubkey_save(pubkey, &Q); + secp256k1_ge_clear(&Q); + return 1; +} + +int secp256k1_ec_pubkey_serialize(const secp256k1_context* ctx, unsigned char *output, size_t *outputlen, const secp256k1_pubkey* pubkey, unsigned int flags) { + secp256k1_ge Q; + size_t len; + int ret = 0; + + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(outputlen != NULL); + ARG_CHECK(*outputlen >= ((flags & SECP256K1_FLAGS_BIT_COMPRESSION) ? 33 : 65)); + len = *outputlen; + *outputlen = 0; + ARG_CHECK(output != NULL); + memset(output, 0, len); + ARG_CHECK(pubkey != NULL); + ARG_CHECK((flags & SECP256K1_FLAGS_TYPE_MASK) == SECP256K1_FLAGS_TYPE_COMPRESSION); + if (secp256k1_pubkey_load(ctx, &Q, pubkey)) { + ret = secp256k1_eckey_pubkey_serialize(&Q, output, &len, flags & SECP256K1_FLAGS_BIT_COMPRESSION); + if (ret) { + *outputlen = len; + } + } + return ret; +} + +static void secp256k1_ecdsa_signature_load(const secp256k1_context* ctx, secp256k1_scalar* r, secp256k1_scalar* s, const secp256k1_ecdsa_signature* sig) { + (void)ctx; + if (sizeof(secp256k1_scalar) == 32) { + /* When the secp256k1_scalar type is exactly 32 byte, use its + * representation inside secp256k1_ecdsa_signature, as conversion is very fast. + * Note that secp256k1_ecdsa_signature_save must use the same representation. */ + memcpy(r, &sig->data[0], 32); + memcpy(s, &sig->data[32], 32); + } else { + secp256k1_scalar_set_b32(r, &sig->data[0], NULL); + secp256k1_scalar_set_b32(s, &sig->data[32], NULL); + } +} + +static void secp256k1_ecdsa_signature_save(secp256k1_ecdsa_signature* sig, const secp256k1_scalar* r, const secp256k1_scalar* s) { + if (sizeof(secp256k1_scalar) == 32) { + memcpy(&sig->data[0], r, 32); + memcpy(&sig->data[32], s, 32); + } else { + secp256k1_scalar_get_b32(&sig->data[0], r); + secp256k1_scalar_get_b32(&sig->data[32], s); + } +} + +int secp256k1_ecdsa_signature_parse_der(const secp256k1_context* ctx, secp256k1_ecdsa_signature* sig, const unsigned char *input, size_t inputlen) { + secp256k1_scalar r, s; + + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(sig != NULL); + ARG_CHECK(input != NULL); + + if (secp256k1_ecdsa_sig_parse(&r, &s, input, inputlen)) { + secp256k1_ecdsa_signature_save(sig, &r, &s); + return 1; + } else { + memset(sig, 0, sizeof(*sig)); + return 0; + } +} + +int secp256k1_ecdsa_signature_parse_compact(const secp256k1_context* ctx, secp256k1_ecdsa_signature* sig, const unsigned char *input64) { + secp256k1_scalar r, s; + int ret = 1; + int overflow = 0; + + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(sig != NULL); + ARG_CHECK(input64 != NULL); + + secp256k1_scalar_set_b32(&r, &input64[0], &overflow); + ret &= !overflow; + secp256k1_scalar_set_b32(&s, &input64[32], &overflow); + ret &= !overflow; + if (ret) { + secp256k1_ecdsa_signature_save(sig, &r, &s); + } else { + memset(sig, 0, sizeof(*sig)); + } + return ret; +} + +int secp256k1_ecdsa_signature_serialize_der(const secp256k1_context* ctx, unsigned char *output, size_t *outputlen, const secp256k1_ecdsa_signature* sig) { + secp256k1_scalar r, s; + + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(output != NULL); + ARG_CHECK(outputlen != NULL); + ARG_CHECK(sig != NULL); + + secp256k1_ecdsa_signature_load(ctx, &r, &s, sig); + return secp256k1_ecdsa_sig_serialize(output, outputlen, &r, &s); +} + +int secp256k1_ecdsa_signature_serialize_compact(const secp256k1_context* ctx, unsigned char *output64, const secp256k1_ecdsa_signature* sig) { + secp256k1_scalar r, s; + + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(output64 != NULL); + ARG_CHECK(sig != NULL); + + secp256k1_ecdsa_signature_load(ctx, &r, &s, sig); + secp256k1_scalar_get_b32(&output64[0], &r); + secp256k1_scalar_get_b32(&output64[32], &s); + return 1; +} + +int secp256k1_ecdsa_signature_normalize(const secp256k1_context* ctx, secp256k1_ecdsa_signature *sigout, const secp256k1_ecdsa_signature *sigin) { + secp256k1_scalar r, s; + int ret = 0; + + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(sigin != NULL); + + secp256k1_ecdsa_signature_load(ctx, &r, &s, sigin); + ret = secp256k1_scalar_is_high(&s); + if (sigout != NULL) { + if (ret) { + secp256k1_scalar_negate(&s, &s); + } + secp256k1_ecdsa_signature_save(sigout, &r, &s); + } + + return ret; +} + +int secp256k1_ecdsa_verify(const secp256k1_context* ctx, const secp256k1_ecdsa_signature *sig, const unsigned char *msg32, const secp256k1_pubkey *pubkey) { + secp256k1_ge q; + secp256k1_scalar r, s; + secp256k1_scalar m; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(secp256k1_ecmult_context_is_built(&ctx->ecmult_ctx)); + ARG_CHECK(msg32 != NULL); + ARG_CHECK(sig != NULL); + ARG_CHECK(pubkey != NULL); + + secp256k1_scalar_set_b32(&m, msg32, NULL); + secp256k1_ecdsa_signature_load(ctx, &r, &s, sig); + return (!secp256k1_scalar_is_high(&s) && + secp256k1_pubkey_load(ctx, &q, pubkey) && + secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &r, &s, &q, &m)); +} + +static SECP256K1_INLINE void buffer_append(unsigned char *buf, unsigned int *offset, const void *data, unsigned int len) { + memcpy(buf + *offset, data, len); + *offset += len; +} + +static int nonce_function_rfc6979(unsigned char *nonce32, const unsigned char *msg32, const unsigned char *key32, const unsigned char *algo16, void *data, unsigned int counter) { + unsigned char keydata[112]; + unsigned int offset = 0; + secp256k1_rfc6979_hmac_sha256 rng; + unsigned int i; + /* We feed a byte array to the PRNG as input, consisting of: + * - the private key (32 bytes) and message (32 bytes), see RFC 6979 3.2d. + * - optionally 32 extra bytes of data, see RFC 6979 3.6 Additional Data. + * - optionally 16 extra bytes with the algorithm name. + * Because the arguments have distinct fixed lengths it is not possible for + * different argument mixtures to emulate each other and result in the same + * nonces. + */ + buffer_append(keydata, &offset, key32, 32); + buffer_append(keydata, &offset, msg32, 32); + if (data != NULL) { + buffer_append(keydata, &offset, data, 32); + } + if (algo16 != NULL) { + buffer_append(keydata, &offset, algo16, 16); + } + secp256k1_rfc6979_hmac_sha256_initialize(&rng, keydata, offset); + memset(keydata, 0, sizeof(keydata)); + for (i = 0; i <= counter; i++) { + secp256k1_rfc6979_hmac_sha256_generate(&rng, nonce32, 32); + } + secp256k1_rfc6979_hmac_sha256_finalize(&rng); + return 1; +} + +const secp256k1_nonce_function secp256k1_nonce_function_rfc6979 = nonce_function_rfc6979; +const secp256k1_nonce_function secp256k1_nonce_function_default = nonce_function_rfc6979; + +int secp256k1_ecdsa_sign(const secp256k1_context* ctx, secp256k1_ecdsa_signature *signature, const unsigned char *msg32, const unsigned char *seckey, secp256k1_nonce_function noncefp, const void* noncedata) { + secp256k1_scalar r, s; + secp256k1_scalar sec, non, msg; + int ret = 0; + int overflow = 0; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(secp256k1_ecmult_gen_context_is_built(&ctx->ecmult_gen_ctx)); + ARG_CHECK(msg32 != NULL); + ARG_CHECK(signature != NULL); + ARG_CHECK(seckey != NULL); + if (noncefp == NULL) { + noncefp = secp256k1_nonce_function_default; + } + + secp256k1_scalar_set_b32(&sec, seckey, &overflow); + /* Fail if the secret key is invalid. */ + if (!overflow && !secp256k1_scalar_is_zero(&sec)) { + unsigned char nonce32[32]; + unsigned int count = 0; + secp256k1_scalar_set_b32(&msg, msg32, NULL); + while (1) { + ret = noncefp(nonce32, msg32, seckey, NULL, (void*)noncedata, count); + if (!ret) { + break; + } + secp256k1_scalar_set_b32(&non, nonce32, &overflow); + if (!overflow && !secp256k1_scalar_is_zero(&non)) { + if (secp256k1_ecdsa_sig_sign(&ctx->ecmult_gen_ctx, &r, &s, &sec, &msg, &non, NULL)) { + break; + } + } + count++; + } + memset(nonce32, 0, 32); + secp256k1_scalar_clear(&msg); + secp256k1_scalar_clear(&non); + secp256k1_scalar_clear(&sec); + } + if (ret) { + secp256k1_ecdsa_signature_save(signature, &r, &s); + } else { + memset(signature, 0, sizeof(*signature)); + } + return ret; +} + +int secp256k1_ec_seckey_verify(const secp256k1_context* ctx, const unsigned char *seckey) { + secp256k1_scalar sec; + int ret; + int overflow; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(seckey != NULL); + + secp256k1_scalar_set_b32(&sec, seckey, &overflow); + ret = !overflow && !secp256k1_scalar_is_zero(&sec); + secp256k1_scalar_clear(&sec); + return ret; +} + +int secp256k1_ec_pubkey_create(const secp256k1_context* ctx, secp256k1_pubkey *pubkey, const unsigned char *seckey) { + secp256k1_gej pj; + secp256k1_ge p; + secp256k1_scalar sec; + int overflow; + int ret = 0; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(pubkey != NULL); + memset(pubkey, 0, sizeof(*pubkey)); + ARG_CHECK(secp256k1_ecmult_gen_context_is_built(&ctx->ecmult_gen_ctx)); + ARG_CHECK(seckey != NULL); + + secp256k1_scalar_set_b32(&sec, seckey, &overflow); + ret = !overflow && !secp256k1_scalar_is_zero(&sec); + if (ret) { + secp256k1_ecmult_gen(&ctx->ecmult_gen_ctx, &pj, &sec); + secp256k1_ge_set_gej(&p, &pj); + secp256k1_pubkey_save(pubkey, &p); + } + secp256k1_scalar_clear(&sec); + return ret; +} + +int secp256k1_ec_privkey_negate(const secp256k1_context* ctx, unsigned char *seckey) { + secp256k1_scalar sec; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(seckey != NULL); + + secp256k1_scalar_set_b32(&sec, seckey, NULL); + secp256k1_scalar_negate(&sec, &sec); + secp256k1_scalar_get_b32(seckey, &sec); + + secp256k1_scalar_clear(&sec); + return 1; +} + +int secp256k1_ec_pubkey_negate(const secp256k1_context* ctx, secp256k1_pubkey *pubkey) { + int ret = 0; + secp256k1_ge p; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(pubkey != NULL); + + ret = secp256k1_pubkey_load(ctx, &p, pubkey); + memset(pubkey, 0, sizeof(*pubkey)); + if (ret) { + secp256k1_ge_neg(&p, &p); + secp256k1_pubkey_save(pubkey, &p); + } + return ret; +} + +int secp256k1_ec_privkey_tweak_add(const secp256k1_context* ctx, unsigned char *seckey, const unsigned char *tweak) { + secp256k1_scalar term; + secp256k1_scalar sec; + int ret = 0; + int overflow = 0; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(seckey != NULL); + ARG_CHECK(tweak != NULL); + + secp256k1_scalar_set_b32(&term, tweak, &overflow); + secp256k1_scalar_set_b32(&sec, seckey, NULL); + + ret = !overflow && secp256k1_eckey_privkey_tweak_add(&sec, &term); + memset(seckey, 0, 32); + if (ret) { + secp256k1_scalar_get_b32(seckey, &sec); + } + + secp256k1_scalar_clear(&sec); + secp256k1_scalar_clear(&term); + return ret; +} + +int secp256k1_ec_pubkey_tweak_add(const secp256k1_context* ctx, secp256k1_pubkey *pubkey, const unsigned char *tweak) { + secp256k1_ge p; + secp256k1_scalar term; + int ret = 0; + int overflow = 0; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(secp256k1_ecmult_context_is_built(&ctx->ecmult_ctx)); + ARG_CHECK(pubkey != NULL); + ARG_CHECK(tweak != NULL); + + secp256k1_scalar_set_b32(&term, tweak, &overflow); + ret = !overflow && secp256k1_pubkey_load(ctx, &p, pubkey); + memset(pubkey, 0, sizeof(*pubkey)); + if (ret) { + if (secp256k1_eckey_pubkey_tweak_add(&ctx->ecmult_ctx, &p, &term)) { + secp256k1_pubkey_save(pubkey, &p); + } else { + ret = 0; + } + } + + return ret; +} + +int secp256k1_ec_privkey_tweak_mul(const secp256k1_context* ctx, unsigned char *seckey, const unsigned char *tweak) { + secp256k1_scalar factor; + secp256k1_scalar sec; + int ret = 0; + int overflow = 0; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(seckey != NULL); + ARG_CHECK(tweak != NULL); + + secp256k1_scalar_set_b32(&factor, tweak, &overflow); + secp256k1_scalar_set_b32(&sec, seckey, NULL); + ret = !overflow && secp256k1_eckey_privkey_tweak_mul(&sec, &factor); + memset(seckey, 0, 32); + if (ret) { + secp256k1_scalar_get_b32(seckey, &sec); + } + + secp256k1_scalar_clear(&sec); + secp256k1_scalar_clear(&factor); + return ret; +} + +int secp256k1_ec_pubkey_tweak_mul(const secp256k1_context* ctx, secp256k1_pubkey *pubkey, const unsigned char *tweak) { + secp256k1_ge p; + secp256k1_scalar factor; + int ret = 0; + int overflow = 0; + VERIFY_CHECK(ctx != NULL); + ARG_CHECK(secp256k1_ecmult_context_is_built(&ctx->ecmult_ctx)); + ARG_CHECK(pubkey != NULL); + ARG_CHECK(tweak != NULL); + + secp256k1_scalar_set_b32(&factor, tweak, &overflow); + ret = !overflow && secp256k1_pubkey_load(ctx, &p, pubkey); + memset(pubkey, 0, sizeof(*pubkey)); + if (ret) { + if (secp256k1_eckey_pubkey_tweak_mul(&ctx->ecmult_ctx, &p, &factor)) { + secp256k1_pubkey_save(pubkey, &p); + } else { + ret = 0; + } + } + + return ret; +} + +int secp256k1_context_randomize(secp256k1_context* ctx, const unsigned char *seed32) { + VERIFY_CHECK(ctx != NULL); + if (secp256k1_ecmult_gen_context_is_built(&ctx->ecmult_gen_ctx)) { + secp256k1_ecmult_gen_blind(&ctx->ecmult_gen_ctx, seed32); + } + return 1; +} + +int secp256k1_ec_pubkey_combine(const secp256k1_context* ctx, secp256k1_pubkey *pubnonce, const secp256k1_pubkey * const *pubnonces, size_t n) { + size_t i; + secp256k1_gej Qj; + secp256k1_ge Q; + + ARG_CHECK(pubnonce != NULL); + memset(pubnonce, 0, sizeof(*pubnonce)); + ARG_CHECK(n >= 1); + ARG_CHECK(pubnonces != NULL); + + secp256k1_gej_set_infinity(&Qj); + + for (i = 0; i < n; i++) { + secp256k1_pubkey_load(ctx, &Q, pubnonces[i]); + secp256k1_gej_add_ge(&Qj, &Qj, &Q); + } + if (secp256k1_gej_is_infinity(&Qj)) { + return 0; + } + secp256k1_ge_set_gej(&Q, &Qj); + secp256k1_pubkey_save(pubnonce, &Q); + return 1; +} + +#ifdef ENABLE_MODULE_ECDH +# include "modules/ecdh/main_impl.h" +#endif + +#ifdef ENABLE_MODULE_RECOVERY +# include "modules/recovery/main_impl.h" +#endif diff --git a/node_modules/secp256k1/src/secp256k1/src/testrand.h b/node_modules/secp256k1/src/secp256k1/src/testrand.h new file mode 100644 index 0000000..f1f9be0 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/testrand.h @@ -0,0 +1,38 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_TESTRAND_H +#define SECP256K1_TESTRAND_H + +#if defined HAVE_CONFIG_H +#include "libsecp256k1-config.h" +#endif + +/* A non-cryptographic RNG used only for test infrastructure. */ + +/** Seed the pseudorandom number generator for testing. */ +SECP256K1_INLINE static void secp256k1_rand_seed(const unsigned char *seed16); + +/** Generate a pseudorandom number in the range [0..2**32-1]. */ +static uint32_t secp256k1_rand32(void); + +/** Generate a pseudorandom number in the range [0..2**bits-1]. Bits must be 1 or + * more. */ +static uint32_t secp256k1_rand_bits(int bits); + +/** Generate a pseudorandom number in the range [0..range-1]. */ +static uint32_t secp256k1_rand_int(uint32_t range); + +/** Generate a pseudorandom 32-byte array. */ +static void secp256k1_rand256(unsigned char *b32); + +/** Generate a pseudorandom 32-byte array with long sequences of zero and one bits. */ +static void secp256k1_rand256_test(unsigned char *b32); + +/** Generate pseudorandom bytes with long sequences of zero and one bits. */ +static void secp256k1_rand_bytes_test(unsigned char *bytes, size_t len); + +#endif /* SECP256K1_TESTRAND_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/testrand_impl.h b/node_modules/secp256k1/src/secp256k1/src/testrand_impl.h new file mode 100644 index 0000000..30a91e5 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/testrand_impl.h @@ -0,0 +1,110 @@ +/********************************************************************** + * Copyright (c) 2013-2015 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_TESTRAND_IMPL_H +#define SECP256K1_TESTRAND_IMPL_H + +#include +#include + +#include "testrand.h" +#include "hash.h" + +static secp256k1_rfc6979_hmac_sha256 secp256k1_test_rng; +static uint32_t secp256k1_test_rng_precomputed[8]; +static int secp256k1_test_rng_precomputed_used = 8; +static uint64_t secp256k1_test_rng_integer; +static int secp256k1_test_rng_integer_bits_left = 0; + +SECP256K1_INLINE static void secp256k1_rand_seed(const unsigned char *seed16) { + secp256k1_rfc6979_hmac_sha256_initialize(&secp256k1_test_rng, seed16, 16); +} + +SECP256K1_INLINE static uint32_t secp256k1_rand32(void) { + if (secp256k1_test_rng_precomputed_used == 8) { + secp256k1_rfc6979_hmac_sha256_generate(&secp256k1_test_rng, (unsigned char*)(&secp256k1_test_rng_precomputed[0]), sizeof(secp256k1_test_rng_precomputed)); + secp256k1_test_rng_precomputed_used = 0; + } + return secp256k1_test_rng_precomputed[secp256k1_test_rng_precomputed_used++]; +} + +static uint32_t secp256k1_rand_bits(int bits) { + uint32_t ret; + if (secp256k1_test_rng_integer_bits_left < bits) { + secp256k1_test_rng_integer |= (((uint64_t)secp256k1_rand32()) << secp256k1_test_rng_integer_bits_left); + secp256k1_test_rng_integer_bits_left += 32; + } + ret = secp256k1_test_rng_integer; + secp256k1_test_rng_integer >>= bits; + secp256k1_test_rng_integer_bits_left -= bits; + ret &= ((~((uint32_t)0)) >> (32 - bits)); + return ret; +} + +static uint32_t secp256k1_rand_int(uint32_t range) { + /* We want a uniform integer between 0 and range-1, inclusive. + * B is the smallest number such that range <= 2**B. + * two mechanisms implemented here: + * - generate B bits numbers until one below range is found, and return it + * - find the largest multiple M of range that is <= 2**(B+A), generate B+A + * bits numbers until one below M is found, and return it modulo range + * The second mechanism consumes A more bits of entropy in every iteration, + * but may need fewer iterations due to M being closer to 2**(B+A) then + * range is to 2**B. The array below (indexed by B) contains a 0 when the + * first mechanism is to be used, and the number A otherwise. + */ + static const int addbits[] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 2, 1, 0}; + uint32_t trange, mult; + int bits = 0; + if (range <= 1) { + return 0; + } + trange = range - 1; + while (trange > 0) { + trange >>= 1; + bits++; + } + if (addbits[bits]) { + bits = bits + addbits[bits]; + mult = ((~((uint32_t)0)) >> (32 - bits)) / range; + trange = range * mult; + } else { + trange = range; + mult = 1; + } + while(1) { + uint32_t x = secp256k1_rand_bits(bits); + if (x < trange) { + return (mult == 1) ? x : (x % range); + } + } +} + +static void secp256k1_rand256(unsigned char *b32) { + secp256k1_rfc6979_hmac_sha256_generate(&secp256k1_test_rng, b32, 32); +} + +static void secp256k1_rand_bytes_test(unsigned char *bytes, size_t len) { + size_t bits = 0; + memset(bytes, 0, len); + while (bits < len * 8) { + int now; + uint32_t val; + now = 1 + (secp256k1_rand_bits(6) * secp256k1_rand_bits(5) + 16) / 31; + val = secp256k1_rand_bits(1); + while (now > 0 && bits < len * 8) { + bytes[bits / 8] |= val << (bits % 8); + now--; + bits++; + } + } +} + +static void secp256k1_rand256_test(unsigned char *b32) { + secp256k1_rand_bytes_test(b32, 32); +} + +#endif /* SECP256K1_TESTRAND_IMPL_H */ diff --git a/node_modules/secp256k1/src/secp256k1/src/tests.c b/node_modules/secp256k1/src/secp256k1/src/tests.c new file mode 100644 index 0000000..d1cfcb5 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/tests.c @@ -0,0 +1,5310 @@ +/********************************************************************** + * Copyright (c) 2013, 2014, 2015 Pieter Wuille, Gregory Maxwell * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#if defined HAVE_CONFIG_H +#include "libsecp256k1-config.h" +#endif + +#include +#include +#include + +#include + +#include "secp256k1.c" +#include "include/secp256k1.h" +#include "include/secp256k1_preallocated.h" +#include "testrand_impl.h" + +#ifdef ENABLE_OPENSSL_TESTS +#include "openssl/bn.h" +#include "openssl/ec.h" +#include "openssl/ecdsa.h" +#include "openssl/obj_mac.h" +# if OPENSSL_VERSION_NUMBER < 0x10100000L +void ECDSA_SIG_get0(const ECDSA_SIG *sig, const BIGNUM **pr, const BIGNUM **ps) {*pr = sig->r; *ps = sig->s;} +# endif +#endif + +#include "contrib/lax_der_parsing.c" +#include "contrib/lax_der_privatekey_parsing.c" + +#if !defined(VG_CHECK) +# if defined(VALGRIND) +# include +# define VG_UNDEF(x,y) VALGRIND_MAKE_MEM_UNDEFINED((x),(y)) +# define VG_CHECK(x,y) VALGRIND_CHECK_MEM_IS_DEFINED((x),(y)) +# else +# define VG_UNDEF(x,y) +# define VG_CHECK(x,y) +# endif +#endif + +static int count = 64; +static secp256k1_context *ctx = NULL; + +static void counting_illegal_callback_fn(const char* str, void* data) { + /* Dummy callback function that just counts. */ + int32_t *p; + (void)str; + p = data; + (*p)++; +} + +static void uncounting_illegal_callback_fn(const char* str, void* data) { + /* Dummy callback function that just counts (backwards). */ + int32_t *p; + (void)str; + p = data; + (*p)--; +} + +void random_field_element_test(secp256k1_fe *fe) { + do { + unsigned char b32[32]; + secp256k1_rand256_test(b32); + if (secp256k1_fe_set_b32(fe, b32)) { + break; + } + } while(1); +} + +void random_field_element_magnitude(secp256k1_fe *fe) { + secp256k1_fe zero; + int n = secp256k1_rand_int(9); + secp256k1_fe_normalize(fe); + if (n == 0) { + return; + } + secp256k1_fe_clear(&zero); + secp256k1_fe_negate(&zero, &zero, 0); + secp256k1_fe_mul_int(&zero, n - 1); + secp256k1_fe_add(fe, &zero); +#ifdef VERIFY + CHECK(fe->magnitude == n); +#endif +} + +void random_group_element_test(secp256k1_ge *ge) { + secp256k1_fe fe; + do { + random_field_element_test(&fe); + if (secp256k1_ge_set_xo_var(ge, &fe, secp256k1_rand_bits(1))) { + secp256k1_fe_normalize(&ge->y); + break; + } + } while(1); +} + +void random_group_element_jacobian_test(secp256k1_gej *gej, const secp256k1_ge *ge) { + secp256k1_fe z2, z3; + do { + random_field_element_test(&gej->z); + if (!secp256k1_fe_is_zero(&gej->z)) { + break; + } + } while(1); + secp256k1_fe_sqr(&z2, &gej->z); + secp256k1_fe_mul(&z3, &z2, &gej->z); + secp256k1_fe_mul(&gej->x, &ge->x, &z2); + secp256k1_fe_mul(&gej->y, &ge->y, &z3); + gej->infinity = ge->infinity; +} + +void random_scalar_order_test(secp256k1_scalar *num) { + do { + unsigned char b32[32]; + int overflow = 0; + secp256k1_rand256_test(b32); + secp256k1_scalar_set_b32(num, b32, &overflow); + if (overflow || secp256k1_scalar_is_zero(num)) { + continue; + } + break; + } while(1); +} + +void random_scalar_order(secp256k1_scalar *num) { + do { + unsigned char b32[32]; + int overflow = 0; + secp256k1_rand256(b32); + secp256k1_scalar_set_b32(num, b32, &overflow); + if (overflow || secp256k1_scalar_is_zero(num)) { + continue; + } + break; + } while(1); +} + +void run_context_tests(int use_prealloc) { + secp256k1_pubkey pubkey; + secp256k1_pubkey zero_pubkey; + secp256k1_ecdsa_signature sig; + unsigned char ctmp[32]; + int32_t ecount; + int32_t ecount2; + secp256k1_context *none; + secp256k1_context *sign; + secp256k1_context *vrfy; + secp256k1_context *both; + void *none_prealloc = NULL; + void *sign_prealloc = NULL; + void *vrfy_prealloc = NULL; + void *both_prealloc = NULL; + + secp256k1_gej pubj; + secp256k1_ge pub; + secp256k1_scalar msg, key, nonce; + secp256k1_scalar sigr, sigs; + + if (use_prealloc) { + none_prealloc = malloc(secp256k1_context_preallocated_size(SECP256K1_CONTEXT_NONE)); + sign_prealloc = malloc(secp256k1_context_preallocated_size(SECP256K1_CONTEXT_SIGN)); + vrfy_prealloc = malloc(secp256k1_context_preallocated_size(SECP256K1_CONTEXT_VERIFY)); + both_prealloc = malloc(secp256k1_context_preallocated_size(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY)); + CHECK(none_prealloc != NULL); + CHECK(sign_prealloc != NULL); + CHECK(vrfy_prealloc != NULL); + CHECK(both_prealloc != NULL); + none = secp256k1_context_preallocated_create(none_prealloc, SECP256K1_CONTEXT_NONE); + sign = secp256k1_context_preallocated_create(sign_prealloc, SECP256K1_CONTEXT_SIGN); + vrfy = secp256k1_context_preallocated_create(vrfy_prealloc, SECP256K1_CONTEXT_VERIFY); + both = secp256k1_context_preallocated_create(both_prealloc, SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY); + } else { + none = secp256k1_context_create(SECP256K1_CONTEXT_NONE); + sign = secp256k1_context_create(SECP256K1_CONTEXT_SIGN); + vrfy = secp256k1_context_create(SECP256K1_CONTEXT_VERIFY); + both = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY); + } + + memset(&zero_pubkey, 0, sizeof(zero_pubkey)); + + ecount = 0; + ecount2 = 10; + secp256k1_context_set_illegal_callback(vrfy, counting_illegal_callback_fn, &ecount); + secp256k1_context_set_illegal_callback(sign, counting_illegal_callback_fn, &ecount2); + secp256k1_context_set_error_callback(sign, counting_illegal_callback_fn, NULL); + CHECK(vrfy->error_callback.fn != sign->error_callback.fn); + + /* check if sizes for cloning are consistent */ + CHECK(secp256k1_context_preallocated_clone_size(none) == secp256k1_context_preallocated_size(SECP256K1_CONTEXT_NONE)); + CHECK(secp256k1_context_preallocated_clone_size(sign) == secp256k1_context_preallocated_size(SECP256K1_CONTEXT_SIGN)); + CHECK(secp256k1_context_preallocated_clone_size(vrfy) == secp256k1_context_preallocated_size(SECP256K1_CONTEXT_VERIFY)); + CHECK(secp256k1_context_preallocated_clone_size(both) == secp256k1_context_preallocated_size(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY)); + + /*** clone and destroy all of them to make sure cloning was complete ***/ + { + secp256k1_context *ctx_tmp; + + if (use_prealloc) { + /* clone into a non-preallocated context and then again into a new preallocated one. */ + ctx_tmp = none; none = secp256k1_context_clone(none); secp256k1_context_preallocated_destroy(ctx_tmp); + free(none_prealloc); none_prealloc = malloc(secp256k1_context_preallocated_size(SECP256K1_CONTEXT_NONE)); CHECK(none_prealloc != NULL); + ctx_tmp = none; none = secp256k1_context_preallocated_clone(none, none_prealloc); secp256k1_context_destroy(ctx_tmp); + + ctx_tmp = sign; sign = secp256k1_context_clone(sign); secp256k1_context_preallocated_destroy(ctx_tmp); + free(sign_prealloc); sign_prealloc = malloc(secp256k1_context_preallocated_size(SECP256K1_CONTEXT_SIGN)); CHECK(sign_prealloc != NULL); + ctx_tmp = sign; sign = secp256k1_context_preallocated_clone(sign, sign_prealloc); secp256k1_context_destroy(ctx_tmp); + + ctx_tmp = vrfy; vrfy = secp256k1_context_clone(vrfy); secp256k1_context_preallocated_destroy(ctx_tmp); + free(vrfy_prealloc); vrfy_prealloc = malloc(secp256k1_context_preallocated_size(SECP256K1_CONTEXT_VERIFY)); CHECK(vrfy_prealloc != NULL); + ctx_tmp = vrfy; vrfy = secp256k1_context_preallocated_clone(vrfy, vrfy_prealloc); secp256k1_context_destroy(ctx_tmp); + + ctx_tmp = both; both = secp256k1_context_clone(both); secp256k1_context_preallocated_destroy(ctx_tmp); + free(both_prealloc); both_prealloc = malloc(secp256k1_context_preallocated_size(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY)); CHECK(both_prealloc != NULL); + ctx_tmp = both; both = secp256k1_context_preallocated_clone(both, both_prealloc); secp256k1_context_destroy(ctx_tmp); + } else { + /* clone into a preallocated context and then again into a new non-preallocated one. */ + void *prealloc_tmp; + + prealloc_tmp = malloc(secp256k1_context_preallocated_size(SECP256K1_CONTEXT_NONE)); CHECK(prealloc_tmp != NULL); + ctx_tmp = none; none = secp256k1_context_preallocated_clone(none, prealloc_tmp); secp256k1_context_destroy(ctx_tmp); + ctx_tmp = none; none = secp256k1_context_clone(none); secp256k1_context_preallocated_destroy(ctx_tmp); + free(prealloc_tmp); + + prealloc_tmp = malloc(secp256k1_context_preallocated_size(SECP256K1_CONTEXT_SIGN)); CHECK(prealloc_tmp != NULL); + ctx_tmp = sign; sign = secp256k1_context_preallocated_clone(sign, prealloc_tmp); secp256k1_context_destroy(ctx_tmp); + ctx_tmp = sign; sign = secp256k1_context_clone(sign); secp256k1_context_preallocated_destroy(ctx_tmp); + free(prealloc_tmp); + + prealloc_tmp = malloc(secp256k1_context_preallocated_size(SECP256K1_CONTEXT_VERIFY)); CHECK(prealloc_tmp != NULL); + ctx_tmp = vrfy; vrfy = secp256k1_context_preallocated_clone(vrfy, prealloc_tmp); secp256k1_context_destroy(ctx_tmp); + ctx_tmp = vrfy; vrfy = secp256k1_context_clone(vrfy); secp256k1_context_preallocated_destroy(ctx_tmp); + free(prealloc_tmp); + + prealloc_tmp = malloc(secp256k1_context_preallocated_size(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY)); CHECK(prealloc_tmp != NULL); + ctx_tmp = both; both = secp256k1_context_preallocated_clone(both, prealloc_tmp); secp256k1_context_destroy(ctx_tmp); + ctx_tmp = both; both = secp256k1_context_clone(both); secp256k1_context_preallocated_destroy(ctx_tmp); + free(prealloc_tmp); + } + } + + /* Verify that the error callback makes it across the clone. */ + CHECK(vrfy->error_callback.fn != sign->error_callback.fn); + /* And that it resets back to default. */ + secp256k1_context_set_error_callback(sign, NULL, NULL); + CHECK(vrfy->error_callback.fn == sign->error_callback.fn); + + /*** attempt to use them ***/ + random_scalar_order_test(&msg); + random_scalar_order_test(&key); + secp256k1_ecmult_gen(&both->ecmult_gen_ctx, &pubj, &key); + secp256k1_ge_set_gej(&pub, &pubj); + + /* Verify context-type checking illegal-argument errors. */ + memset(ctmp, 1, 32); + CHECK(secp256k1_ec_pubkey_create(vrfy, &pubkey, ctmp) == 0); + CHECK(ecount == 1); + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_create(sign, &pubkey, ctmp) == 1); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ecdsa_sign(vrfy, &sig, ctmp, ctmp, NULL, NULL) == 0); + CHECK(ecount == 2); + VG_UNDEF(&sig, sizeof(sig)); + CHECK(secp256k1_ecdsa_sign(sign, &sig, ctmp, ctmp, NULL, NULL) == 1); + VG_CHECK(&sig, sizeof(sig)); + CHECK(ecount2 == 10); + CHECK(secp256k1_ecdsa_verify(sign, &sig, ctmp, &pubkey) == 0); + CHECK(ecount2 == 11); + CHECK(secp256k1_ecdsa_verify(vrfy, &sig, ctmp, &pubkey) == 1); + CHECK(ecount == 2); + CHECK(secp256k1_ec_pubkey_tweak_add(sign, &pubkey, ctmp) == 0); + CHECK(ecount2 == 12); + CHECK(secp256k1_ec_pubkey_tweak_add(vrfy, &pubkey, ctmp) == 1); + CHECK(ecount == 2); + CHECK(secp256k1_ec_pubkey_tweak_mul(sign, &pubkey, ctmp) == 0); + CHECK(ecount2 == 13); + CHECK(secp256k1_ec_pubkey_negate(vrfy, &pubkey) == 1); + CHECK(ecount == 2); + CHECK(secp256k1_ec_pubkey_negate(sign, &pubkey) == 1); + CHECK(ecount == 2); + CHECK(secp256k1_ec_pubkey_negate(sign, NULL) == 0); + CHECK(ecount2 == 14); + CHECK(secp256k1_ec_pubkey_negate(vrfy, &zero_pubkey) == 0); + CHECK(ecount == 3); + CHECK(secp256k1_ec_pubkey_tweak_mul(vrfy, &pubkey, ctmp) == 1); + CHECK(ecount == 3); + CHECK(secp256k1_context_randomize(vrfy, ctmp) == 1); + CHECK(ecount == 3); + CHECK(secp256k1_context_randomize(vrfy, NULL) == 1); + CHECK(ecount == 3); + CHECK(secp256k1_context_randomize(sign, ctmp) == 1); + CHECK(ecount2 == 14); + CHECK(secp256k1_context_randomize(sign, NULL) == 1); + CHECK(ecount2 == 14); + secp256k1_context_set_illegal_callback(vrfy, NULL, NULL); + secp256k1_context_set_illegal_callback(sign, NULL, NULL); + + /* obtain a working nonce */ + do { + random_scalar_order_test(&nonce); + } while(!secp256k1_ecdsa_sig_sign(&both->ecmult_gen_ctx, &sigr, &sigs, &key, &msg, &nonce, NULL)); + + /* try signing */ + CHECK(secp256k1_ecdsa_sig_sign(&sign->ecmult_gen_ctx, &sigr, &sigs, &key, &msg, &nonce, NULL)); + CHECK(secp256k1_ecdsa_sig_sign(&both->ecmult_gen_ctx, &sigr, &sigs, &key, &msg, &nonce, NULL)); + + /* try verifying */ + CHECK(secp256k1_ecdsa_sig_verify(&vrfy->ecmult_ctx, &sigr, &sigs, &pub, &msg)); + CHECK(secp256k1_ecdsa_sig_verify(&both->ecmult_ctx, &sigr, &sigs, &pub, &msg)); + + /* cleanup */ + if (use_prealloc) { + secp256k1_context_preallocated_destroy(none); + secp256k1_context_preallocated_destroy(sign); + secp256k1_context_preallocated_destroy(vrfy); + secp256k1_context_preallocated_destroy(both); + free(none_prealloc); + free(sign_prealloc); + free(vrfy_prealloc); + free(both_prealloc); + } else { + secp256k1_context_destroy(none); + secp256k1_context_destroy(sign); + secp256k1_context_destroy(vrfy); + secp256k1_context_destroy(both); + } + /* Defined as no-op. */ + secp256k1_context_destroy(NULL); + secp256k1_context_preallocated_destroy(NULL); + +} + +void run_scratch_tests(void) { + const size_t adj_alloc = ((500 + ALIGNMENT - 1) / ALIGNMENT) * ALIGNMENT; + + int32_t ecount = 0; + size_t checkpoint; + size_t checkpoint_2; + secp256k1_context *none = secp256k1_context_create(SECP256K1_CONTEXT_NONE); + secp256k1_scratch_space *scratch; + secp256k1_scratch_space local_scratch; + + /* Test public API */ + secp256k1_context_set_illegal_callback(none, counting_illegal_callback_fn, &ecount); + secp256k1_context_set_error_callback(none, counting_illegal_callback_fn, &ecount); + + scratch = secp256k1_scratch_space_create(none, 1000); + CHECK(scratch != NULL); + CHECK(ecount == 0); + + /* Test internal API */ + CHECK(secp256k1_scratch_max_allocation(&none->error_callback, scratch, 0) == 1000); + CHECK(secp256k1_scratch_max_allocation(&none->error_callback, scratch, 1) == 1000 - (ALIGNMENT - 1)); + CHECK(scratch->alloc_size == 0); + CHECK(scratch->alloc_size % ALIGNMENT == 0); + + /* Allocating 500 bytes succeeds */ + checkpoint = secp256k1_scratch_checkpoint(&none->error_callback, scratch); + CHECK(secp256k1_scratch_alloc(&none->error_callback, scratch, 500) != NULL); + CHECK(secp256k1_scratch_max_allocation(&none->error_callback, scratch, 0) == 1000 - adj_alloc); + CHECK(secp256k1_scratch_max_allocation(&none->error_callback, scratch, 1) == 1000 - adj_alloc - (ALIGNMENT - 1)); + CHECK(scratch->alloc_size != 0); + CHECK(scratch->alloc_size % ALIGNMENT == 0); + + /* Allocating another 500 bytes fails */ + CHECK(secp256k1_scratch_alloc(&none->error_callback, scratch, 500) == NULL); + CHECK(secp256k1_scratch_max_allocation(&none->error_callback, scratch, 0) == 1000 - adj_alloc); + CHECK(secp256k1_scratch_max_allocation(&none->error_callback, scratch, 1) == 1000 - adj_alloc - (ALIGNMENT - 1)); + CHECK(scratch->alloc_size != 0); + CHECK(scratch->alloc_size % ALIGNMENT == 0); + + /* ...but it succeeds once we apply the checkpoint to undo it */ + secp256k1_scratch_apply_checkpoint(&none->error_callback, scratch, checkpoint); + CHECK(scratch->alloc_size == 0); + CHECK(secp256k1_scratch_max_allocation(&none->error_callback, scratch, 0) == 1000); + CHECK(secp256k1_scratch_alloc(&none->error_callback, scratch, 500) != NULL); + CHECK(scratch->alloc_size != 0); + + /* try to apply a bad checkpoint */ + checkpoint_2 = secp256k1_scratch_checkpoint(&none->error_callback, scratch); + secp256k1_scratch_apply_checkpoint(&none->error_callback, scratch, checkpoint); + CHECK(ecount == 0); + secp256k1_scratch_apply_checkpoint(&none->error_callback, scratch, checkpoint_2); /* checkpoint_2 is after checkpoint */ + CHECK(ecount == 1); + secp256k1_scratch_apply_checkpoint(&none->error_callback, scratch, (size_t) -1); /* this is just wildly invalid */ + CHECK(ecount == 2); + + /* try to use badly initialized scratch space */ + secp256k1_scratch_space_destroy(none, scratch); + memset(&local_scratch, 0, sizeof(local_scratch)); + scratch = &local_scratch; + CHECK(!secp256k1_scratch_max_allocation(&none->error_callback, scratch, 0)); + CHECK(ecount == 3); + CHECK(secp256k1_scratch_alloc(&none->error_callback, scratch, 500) == NULL); + CHECK(ecount == 4); + secp256k1_scratch_space_destroy(none, scratch); + CHECK(ecount == 5); + + /* cleanup */ + secp256k1_scratch_space_destroy(none, NULL); /* no-op */ + secp256k1_context_destroy(none); +} + +/***** HASH TESTS *****/ + +void run_sha256_tests(void) { + static const char *inputs[8] = { + "", "abc", "message digest", "secure hash algorithm", "SHA256 is considered to be safe", + "abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq", + "For this sample, this 63-byte string will be used as input data", + "This is exactly 64 bytes long, not counting the terminating byte" + }; + static const unsigned char outputs[8][32] = { + {0xe3, 0xb0, 0xc4, 0x42, 0x98, 0xfc, 0x1c, 0x14, 0x9a, 0xfb, 0xf4, 0xc8, 0x99, 0x6f, 0xb9, 0x24, 0x27, 0xae, 0x41, 0xe4, 0x64, 0x9b, 0x93, 0x4c, 0xa4, 0x95, 0x99, 0x1b, 0x78, 0x52, 0xb8, 0x55}, + {0xba, 0x78, 0x16, 0xbf, 0x8f, 0x01, 0xcf, 0xea, 0x41, 0x41, 0x40, 0xde, 0x5d, 0xae, 0x22, 0x23, 0xb0, 0x03, 0x61, 0xa3, 0x96, 0x17, 0x7a, 0x9c, 0xb4, 0x10, 0xff, 0x61, 0xf2, 0x00, 0x15, 0xad}, + {0xf7, 0x84, 0x6f, 0x55, 0xcf, 0x23, 0xe1, 0x4e, 0xeb, 0xea, 0xb5, 0xb4, 0xe1, 0x55, 0x0c, 0xad, 0x5b, 0x50, 0x9e, 0x33, 0x48, 0xfb, 0xc4, 0xef, 0xa3, 0xa1, 0x41, 0x3d, 0x39, 0x3c, 0xb6, 0x50}, + {0xf3, 0x0c, 0xeb, 0x2b, 0xb2, 0x82, 0x9e, 0x79, 0xe4, 0xca, 0x97, 0x53, 0xd3, 0x5a, 0x8e, 0xcc, 0x00, 0x26, 0x2d, 0x16, 0x4c, 0xc0, 0x77, 0x08, 0x02, 0x95, 0x38, 0x1c, 0xbd, 0x64, 0x3f, 0x0d}, + {0x68, 0x19, 0xd9, 0x15, 0xc7, 0x3f, 0x4d, 0x1e, 0x77, 0xe4, 0xe1, 0xb5, 0x2d, 0x1f, 0xa0, 0xf9, 0xcf, 0x9b, 0xea, 0xea, 0xd3, 0x93, 0x9f, 0x15, 0x87, 0x4b, 0xd9, 0x88, 0xe2, 0xa2, 0x36, 0x30}, + {0x24, 0x8d, 0x6a, 0x61, 0xd2, 0x06, 0x38, 0xb8, 0xe5, 0xc0, 0x26, 0x93, 0x0c, 0x3e, 0x60, 0x39, 0xa3, 0x3c, 0xe4, 0x59, 0x64, 0xff, 0x21, 0x67, 0xf6, 0xec, 0xed, 0xd4, 0x19, 0xdb, 0x06, 0xc1}, + {0xf0, 0x8a, 0x78, 0xcb, 0xba, 0xee, 0x08, 0x2b, 0x05, 0x2a, 0xe0, 0x70, 0x8f, 0x32, 0xfa, 0x1e, 0x50, 0xc5, 0xc4, 0x21, 0xaa, 0x77, 0x2b, 0xa5, 0xdb, 0xb4, 0x06, 0xa2, 0xea, 0x6b, 0xe3, 0x42}, + {0xab, 0x64, 0xef, 0xf7, 0xe8, 0x8e, 0x2e, 0x46, 0x16, 0x5e, 0x29, 0xf2, 0xbc, 0xe4, 0x18, 0x26, 0xbd, 0x4c, 0x7b, 0x35, 0x52, 0xf6, 0xb3, 0x82, 0xa9, 0xe7, 0xd3, 0xaf, 0x47, 0xc2, 0x45, 0xf8} + }; + int i; + for (i = 0; i < 8; i++) { + unsigned char out[32]; + secp256k1_sha256 hasher; + secp256k1_sha256_initialize(&hasher); + secp256k1_sha256_write(&hasher, (const unsigned char*)(inputs[i]), strlen(inputs[i])); + secp256k1_sha256_finalize(&hasher, out); + CHECK(memcmp(out, outputs[i], 32) == 0); + if (strlen(inputs[i]) > 0) { + int split = secp256k1_rand_int(strlen(inputs[i])); + secp256k1_sha256_initialize(&hasher); + secp256k1_sha256_write(&hasher, (const unsigned char*)(inputs[i]), split); + secp256k1_sha256_write(&hasher, (const unsigned char*)(inputs[i] + split), strlen(inputs[i]) - split); + secp256k1_sha256_finalize(&hasher, out); + CHECK(memcmp(out, outputs[i], 32) == 0); + } + } +} + +void run_hmac_sha256_tests(void) { + static const char *keys[6] = { + "\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b\x0b", + "\x4a\x65\x66\x65", + "\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa", + "\x01\x02\x03\x04\x05\x06\x07\x08\x09\x0a\x0b\x0c\x0d\x0e\x0f\x10\x11\x12\x13\x14\x15\x16\x17\x18\x19", + "\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa", + "\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa\xaa" + }; + static const char *inputs[6] = { + "\x48\x69\x20\x54\x68\x65\x72\x65", + "\x77\x68\x61\x74\x20\x64\x6f\x20\x79\x61\x20\x77\x61\x6e\x74\x20\x66\x6f\x72\x20\x6e\x6f\x74\x68\x69\x6e\x67\x3f", + "\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd\xdd", + "\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd\xcd", + "\x54\x65\x73\x74\x20\x55\x73\x69\x6e\x67\x20\x4c\x61\x72\x67\x65\x72\x20\x54\x68\x61\x6e\x20\x42\x6c\x6f\x63\x6b\x2d\x53\x69\x7a\x65\x20\x4b\x65\x79\x20\x2d\x20\x48\x61\x73\x68\x20\x4b\x65\x79\x20\x46\x69\x72\x73\x74", + "\x54\x68\x69\x73\x20\x69\x73\x20\x61\x20\x74\x65\x73\x74\x20\x75\x73\x69\x6e\x67\x20\x61\x20\x6c\x61\x72\x67\x65\x72\x20\x74\x68\x61\x6e\x20\x62\x6c\x6f\x63\x6b\x2d\x73\x69\x7a\x65\x20\x6b\x65\x79\x20\x61\x6e\x64\x20\x61\x20\x6c\x61\x72\x67\x65\x72\x20\x74\x68\x61\x6e\x20\x62\x6c\x6f\x63\x6b\x2d\x73\x69\x7a\x65\x20\x64\x61\x74\x61\x2e\x20\x54\x68\x65\x20\x6b\x65\x79\x20\x6e\x65\x65\x64\x73\x20\x74\x6f\x20\x62\x65\x20\x68\x61\x73\x68\x65\x64\x20\x62\x65\x66\x6f\x72\x65\x20\x62\x65\x69\x6e\x67\x20\x75\x73\x65\x64\x20\x62\x79\x20\x74\x68\x65\x20\x48\x4d\x41\x43\x20\x61\x6c\x67\x6f\x72\x69\x74\x68\x6d\x2e" + }; + static const unsigned char outputs[6][32] = { + {0xb0, 0x34, 0x4c, 0x61, 0xd8, 0xdb, 0x38, 0x53, 0x5c, 0xa8, 0xaf, 0xce, 0xaf, 0x0b, 0xf1, 0x2b, 0x88, 0x1d, 0xc2, 0x00, 0xc9, 0x83, 0x3d, 0xa7, 0x26, 0xe9, 0x37, 0x6c, 0x2e, 0x32, 0xcf, 0xf7}, + {0x5b, 0xdc, 0xc1, 0x46, 0xbf, 0x60, 0x75, 0x4e, 0x6a, 0x04, 0x24, 0x26, 0x08, 0x95, 0x75, 0xc7, 0x5a, 0x00, 0x3f, 0x08, 0x9d, 0x27, 0x39, 0x83, 0x9d, 0xec, 0x58, 0xb9, 0x64, 0xec, 0x38, 0x43}, + {0x77, 0x3e, 0xa9, 0x1e, 0x36, 0x80, 0x0e, 0x46, 0x85, 0x4d, 0xb8, 0xeb, 0xd0, 0x91, 0x81, 0xa7, 0x29, 0x59, 0x09, 0x8b, 0x3e, 0xf8, 0xc1, 0x22, 0xd9, 0x63, 0x55, 0x14, 0xce, 0xd5, 0x65, 0xfe}, + {0x82, 0x55, 0x8a, 0x38, 0x9a, 0x44, 0x3c, 0x0e, 0xa4, 0xcc, 0x81, 0x98, 0x99, 0xf2, 0x08, 0x3a, 0x85, 0xf0, 0xfa, 0xa3, 0xe5, 0x78, 0xf8, 0x07, 0x7a, 0x2e, 0x3f, 0xf4, 0x67, 0x29, 0x66, 0x5b}, + {0x60, 0xe4, 0x31, 0x59, 0x1e, 0xe0, 0xb6, 0x7f, 0x0d, 0x8a, 0x26, 0xaa, 0xcb, 0xf5, 0xb7, 0x7f, 0x8e, 0x0b, 0xc6, 0x21, 0x37, 0x28, 0xc5, 0x14, 0x05, 0x46, 0x04, 0x0f, 0x0e, 0xe3, 0x7f, 0x54}, + {0x9b, 0x09, 0xff, 0xa7, 0x1b, 0x94, 0x2f, 0xcb, 0x27, 0x63, 0x5f, 0xbc, 0xd5, 0xb0, 0xe9, 0x44, 0xbf, 0xdc, 0x63, 0x64, 0x4f, 0x07, 0x13, 0x93, 0x8a, 0x7f, 0x51, 0x53, 0x5c, 0x3a, 0x35, 0xe2} + }; + int i; + for (i = 0; i < 6; i++) { + secp256k1_hmac_sha256 hasher; + unsigned char out[32]; + secp256k1_hmac_sha256_initialize(&hasher, (const unsigned char*)(keys[i]), strlen(keys[i])); + secp256k1_hmac_sha256_write(&hasher, (const unsigned char*)(inputs[i]), strlen(inputs[i])); + secp256k1_hmac_sha256_finalize(&hasher, out); + CHECK(memcmp(out, outputs[i], 32) == 0); + if (strlen(inputs[i]) > 0) { + int split = secp256k1_rand_int(strlen(inputs[i])); + secp256k1_hmac_sha256_initialize(&hasher, (const unsigned char*)(keys[i]), strlen(keys[i])); + secp256k1_hmac_sha256_write(&hasher, (const unsigned char*)(inputs[i]), split); + secp256k1_hmac_sha256_write(&hasher, (const unsigned char*)(inputs[i] + split), strlen(inputs[i]) - split); + secp256k1_hmac_sha256_finalize(&hasher, out); + CHECK(memcmp(out, outputs[i], 32) == 0); + } + } +} + +void run_rfc6979_hmac_sha256_tests(void) { + static const unsigned char key1[65] = {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f, 0x00, 0x4b, 0xf5, 0x12, 0x2f, 0x34, 0x45, 0x54, 0xc5, 0x3b, 0xde, 0x2e, 0xbb, 0x8c, 0xd2, 0xb7, 0xe3, 0xd1, 0x60, 0x0a, 0xd6, 0x31, 0xc3, 0x85, 0xa5, 0xd7, 0xcc, 0xe2, 0x3c, 0x77, 0x85, 0x45, 0x9a, 0}; + static const unsigned char out1[3][32] = { + {0x4f, 0xe2, 0x95, 0x25, 0xb2, 0x08, 0x68, 0x09, 0x15, 0x9a, 0xcd, 0xf0, 0x50, 0x6e, 0xfb, 0x86, 0xb0, 0xec, 0x93, 0x2c, 0x7b, 0xa4, 0x42, 0x56, 0xab, 0x32, 0x1e, 0x42, 0x1e, 0x67, 0xe9, 0xfb}, + {0x2b, 0xf0, 0xff, 0xf1, 0xd3, 0xc3, 0x78, 0xa2, 0x2d, 0xc5, 0xde, 0x1d, 0x85, 0x65, 0x22, 0x32, 0x5c, 0x65, 0xb5, 0x04, 0x49, 0x1a, 0x0c, 0xbd, 0x01, 0xcb, 0x8f, 0x3a, 0xa6, 0x7f, 0xfd, 0x4a}, + {0xf5, 0x28, 0xb4, 0x10, 0xcb, 0x54, 0x1f, 0x77, 0x00, 0x0d, 0x7a, 0xfb, 0x6c, 0x5b, 0x53, 0xc5, 0xc4, 0x71, 0xea, 0xb4, 0x3e, 0x46, 0x6d, 0x9a, 0xc5, 0x19, 0x0c, 0x39, 0xc8, 0x2f, 0xd8, 0x2e} + }; + + static const unsigned char key2[64] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xe3, 0xb0, 0xc4, 0x42, 0x98, 0xfc, 0x1c, 0x14, 0x9a, 0xfb, 0xf4, 0xc8, 0x99, 0x6f, 0xb9, 0x24, 0x27, 0xae, 0x41, 0xe4, 0x64, 0x9b, 0x93, 0x4c, 0xa4, 0x95, 0x99, 0x1b, 0x78, 0x52, 0xb8, 0x55}; + static const unsigned char out2[3][32] = { + {0x9c, 0x23, 0x6c, 0x16, 0x5b, 0x82, 0xae, 0x0c, 0xd5, 0x90, 0x65, 0x9e, 0x10, 0x0b, 0x6b, 0xab, 0x30, 0x36, 0xe7, 0xba, 0x8b, 0x06, 0x74, 0x9b, 0xaf, 0x69, 0x81, 0xe1, 0x6f, 0x1a, 0x2b, 0x95}, + {0xdf, 0x47, 0x10, 0x61, 0x62, 0x5b, 0xc0, 0xea, 0x14, 0xb6, 0x82, 0xfe, 0xee, 0x2c, 0x9c, 0x02, 0xf2, 0x35, 0xda, 0x04, 0x20, 0x4c, 0x1d, 0x62, 0xa1, 0x53, 0x6c, 0x6e, 0x17, 0xae, 0xd7, 0xa9}, + {0x75, 0x97, 0x88, 0x7c, 0xbd, 0x76, 0x32, 0x1f, 0x32, 0xe3, 0x04, 0x40, 0x67, 0x9a, 0x22, 0xcf, 0x7f, 0x8d, 0x9d, 0x2e, 0xac, 0x39, 0x0e, 0x58, 0x1f, 0xea, 0x09, 0x1c, 0xe2, 0x02, 0xba, 0x94} + }; + + secp256k1_rfc6979_hmac_sha256 rng; + unsigned char out[32]; + int i; + + secp256k1_rfc6979_hmac_sha256_initialize(&rng, key1, 64); + for (i = 0; i < 3; i++) { + secp256k1_rfc6979_hmac_sha256_generate(&rng, out, 32); + CHECK(memcmp(out, out1[i], 32) == 0); + } + secp256k1_rfc6979_hmac_sha256_finalize(&rng); + + secp256k1_rfc6979_hmac_sha256_initialize(&rng, key1, 65); + for (i = 0; i < 3; i++) { + secp256k1_rfc6979_hmac_sha256_generate(&rng, out, 32); + CHECK(memcmp(out, out1[i], 32) != 0); + } + secp256k1_rfc6979_hmac_sha256_finalize(&rng); + + secp256k1_rfc6979_hmac_sha256_initialize(&rng, key2, 64); + for (i = 0; i < 3; i++) { + secp256k1_rfc6979_hmac_sha256_generate(&rng, out, 32); + CHECK(memcmp(out, out2[i], 32) == 0); + } + secp256k1_rfc6979_hmac_sha256_finalize(&rng); +} + +/***** RANDOM TESTS *****/ + +void test_rand_bits(int rand32, int bits) { + /* (1-1/2^B)^rounds[B] < 1/10^9, so rounds is the number of iterations to + * get a false negative chance below once in a billion */ + static const unsigned int rounds[7] = {1, 30, 73, 156, 322, 653, 1316}; + /* We try multiplying the results with various odd numbers, which shouldn't + * influence the uniform distribution modulo a power of 2. */ + static const uint32_t mults[6] = {1, 3, 21, 289, 0x9999, 0x80402011}; + /* We only select up to 6 bits from the output to analyse */ + unsigned int usebits = bits > 6 ? 6 : bits; + unsigned int maxshift = bits - usebits; + /* For each of the maxshift+1 usebits-bit sequences inside a bits-bit + number, track all observed outcomes, one per bit in a uint64_t. */ + uint64_t x[6][27] = {{0}}; + unsigned int i, shift, m; + /* Multiply the output of all rand calls with the odd number m, which + should not change the uniformity of its distribution. */ + for (i = 0; i < rounds[usebits]; i++) { + uint32_t r = (rand32 ? secp256k1_rand32() : secp256k1_rand_bits(bits)); + CHECK((((uint64_t)r) >> bits) == 0); + for (m = 0; m < sizeof(mults) / sizeof(mults[0]); m++) { + uint32_t rm = r * mults[m]; + for (shift = 0; shift <= maxshift; shift++) { + x[m][shift] |= (((uint64_t)1) << ((rm >> shift) & ((1 << usebits) - 1))); + } + } + } + for (m = 0; m < sizeof(mults) / sizeof(mults[0]); m++) { + for (shift = 0; shift <= maxshift; shift++) { + /* Test that the lower usebits bits of x[shift] are 1 */ + CHECK(((~x[m][shift]) << (64 - (1 << usebits))) == 0); + } + } +} + +/* Subrange must be a whole divisor of range, and at most 64 */ +void test_rand_int(uint32_t range, uint32_t subrange) { + /* (1-1/subrange)^rounds < 1/10^9 */ + int rounds = (subrange * 2073) / 100; + int i; + uint64_t x = 0; + CHECK((range % subrange) == 0); + for (i = 0; i < rounds; i++) { + uint32_t r = secp256k1_rand_int(range); + CHECK(r < range); + r = r % subrange; + x |= (((uint64_t)1) << r); + } + /* Test that the lower subrange bits of x are 1. */ + CHECK(((~x) << (64 - subrange)) == 0); +} + +void run_rand_bits(void) { + size_t b; + test_rand_bits(1, 32); + for (b = 1; b <= 32; b++) { + test_rand_bits(0, b); + } +} + +void run_rand_int(void) { + static const uint32_t ms[] = {1, 3, 17, 1000, 13771, 999999, 33554432}; + static const uint32_t ss[] = {1, 3, 6, 9, 13, 31, 64}; + unsigned int m, s; + for (m = 0; m < sizeof(ms) / sizeof(ms[0]); m++) { + for (s = 0; s < sizeof(ss) / sizeof(ss[0]); s++) { + test_rand_int(ms[m] * ss[s], ss[s]); + } + } +} + +/***** NUM TESTS *****/ + +#ifndef USE_NUM_NONE +void random_num_negate(secp256k1_num *num) { + if (secp256k1_rand_bits(1)) { + secp256k1_num_negate(num); + } +} + +void random_num_order_test(secp256k1_num *num) { + secp256k1_scalar sc; + random_scalar_order_test(&sc); + secp256k1_scalar_get_num(num, &sc); +} + +void random_num_order(secp256k1_num *num) { + secp256k1_scalar sc; + random_scalar_order(&sc); + secp256k1_scalar_get_num(num, &sc); +} + +void test_num_negate(void) { + secp256k1_num n1; + secp256k1_num n2; + random_num_order_test(&n1); /* n1 = R */ + random_num_negate(&n1); + secp256k1_num_copy(&n2, &n1); /* n2 = R */ + secp256k1_num_sub(&n1, &n2, &n1); /* n1 = n2-n1 = 0 */ + CHECK(secp256k1_num_is_zero(&n1)); + secp256k1_num_copy(&n1, &n2); /* n1 = R */ + secp256k1_num_negate(&n1); /* n1 = -R */ + CHECK(!secp256k1_num_is_zero(&n1)); + secp256k1_num_add(&n1, &n2, &n1); /* n1 = n2+n1 = 0 */ + CHECK(secp256k1_num_is_zero(&n1)); + secp256k1_num_copy(&n1, &n2); /* n1 = R */ + secp256k1_num_negate(&n1); /* n1 = -R */ + CHECK(secp256k1_num_is_neg(&n1) != secp256k1_num_is_neg(&n2)); + secp256k1_num_negate(&n1); /* n1 = R */ + CHECK(secp256k1_num_eq(&n1, &n2)); +} + +void test_num_add_sub(void) { + int i; + secp256k1_scalar s; + secp256k1_num n1; + secp256k1_num n2; + secp256k1_num n1p2, n2p1, n1m2, n2m1; + random_num_order_test(&n1); /* n1 = R1 */ + if (secp256k1_rand_bits(1)) { + random_num_negate(&n1); + } + random_num_order_test(&n2); /* n2 = R2 */ + if (secp256k1_rand_bits(1)) { + random_num_negate(&n2); + } + secp256k1_num_add(&n1p2, &n1, &n2); /* n1p2 = R1 + R2 */ + secp256k1_num_add(&n2p1, &n2, &n1); /* n2p1 = R2 + R1 */ + secp256k1_num_sub(&n1m2, &n1, &n2); /* n1m2 = R1 - R2 */ + secp256k1_num_sub(&n2m1, &n2, &n1); /* n2m1 = R2 - R1 */ + CHECK(secp256k1_num_eq(&n1p2, &n2p1)); + CHECK(!secp256k1_num_eq(&n1p2, &n1m2)); + secp256k1_num_negate(&n2m1); /* n2m1 = -R2 + R1 */ + CHECK(secp256k1_num_eq(&n2m1, &n1m2)); + CHECK(!secp256k1_num_eq(&n2m1, &n1)); + secp256k1_num_add(&n2m1, &n2m1, &n2); /* n2m1 = -R2 + R1 + R2 = R1 */ + CHECK(secp256k1_num_eq(&n2m1, &n1)); + CHECK(!secp256k1_num_eq(&n2p1, &n1)); + secp256k1_num_sub(&n2p1, &n2p1, &n2); /* n2p1 = R2 + R1 - R2 = R1 */ + CHECK(secp256k1_num_eq(&n2p1, &n1)); + + /* check is_one */ + secp256k1_scalar_set_int(&s, 1); + secp256k1_scalar_get_num(&n1, &s); + CHECK(secp256k1_num_is_one(&n1)); + /* check that 2^n + 1 is never 1 */ + secp256k1_scalar_get_num(&n2, &s); + for (i = 0; i < 250; ++i) { + secp256k1_num_add(&n1, &n1, &n1); /* n1 *= 2 */ + secp256k1_num_add(&n1p2, &n1, &n2); /* n1p2 = n1 + 1 */ + CHECK(!secp256k1_num_is_one(&n1p2)); + } +} + +void test_num_mod(void) { + int i; + secp256k1_scalar s; + secp256k1_num order, n; + + /* check that 0 mod anything is 0 */ + random_scalar_order_test(&s); + secp256k1_scalar_get_num(&order, &s); + secp256k1_scalar_set_int(&s, 0); + secp256k1_scalar_get_num(&n, &s); + secp256k1_num_mod(&n, &order); + CHECK(secp256k1_num_is_zero(&n)); + + /* check that anything mod 1 is 0 */ + secp256k1_scalar_set_int(&s, 1); + secp256k1_scalar_get_num(&order, &s); + secp256k1_scalar_get_num(&n, &s); + secp256k1_num_mod(&n, &order); + CHECK(secp256k1_num_is_zero(&n)); + + /* check that increasing the number past 2^256 does not break this */ + random_scalar_order_test(&s); + secp256k1_scalar_get_num(&n, &s); + /* multiply by 2^8, which'll test this case with high probability */ + for (i = 0; i < 8; ++i) { + secp256k1_num_add(&n, &n, &n); + } + secp256k1_num_mod(&n, &order); + CHECK(secp256k1_num_is_zero(&n)); +} + +void test_num_jacobi(void) { + secp256k1_scalar sqr; + secp256k1_scalar small; + secp256k1_scalar five; /* five is not a quadratic residue */ + secp256k1_num order, n; + int i; + /* squares mod 5 are 1, 4 */ + const int jacobi5[10] = { 0, 1, -1, -1, 1, 0, 1, -1, -1, 1 }; + + /* check some small values with 5 as the order */ + secp256k1_scalar_set_int(&five, 5); + secp256k1_scalar_get_num(&order, &five); + for (i = 0; i < 10; ++i) { + secp256k1_scalar_set_int(&small, i); + secp256k1_scalar_get_num(&n, &small); + CHECK(secp256k1_num_jacobi(&n, &order) == jacobi5[i]); + } + + /** test large values with 5 as group order */ + secp256k1_scalar_get_num(&order, &five); + /* we first need a scalar which is not a multiple of 5 */ + do { + secp256k1_num fiven; + random_scalar_order_test(&sqr); + secp256k1_scalar_get_num(&fiven, &five); + secp256k1_scalar_get_num(&n, &sqr); + secp256k1_num_mod(&n, &fiven); + } while (secp256k1_num_is_zero(&n)); + /* next force it to be a residue. 2 is a nonresidue mod 5 so we can + * just multiply by two, i.e. add the number to itself */ + if (secp256k1_num_jacobi(&n, &order) == -1) { + secp256k1_num_add(&n, &n, &n); + } + + /* test residue */ + CHECK(secp256k1_num_jacobi(&n, &order) == 1); + /* test nonresidue */ + secp256k1_num_add(&n, &n, &n); + CHECK(secp256k1_num_jacobi(&n, &order) == -1); + + /** test with secp group order as order */ + secp256k1_scalar_order_get_num(&order); + random_scalar_order_test(&sqr); + secp256k1_scalar_sqr(&sqr, &sqr); + /* test residue */ + secp256k1_scalar_get_num(&n, &sqr); + CHECK(secp256k1_num_jacobi(&n, &order) == 1); + /* test nonresidue */ + secp256k1_scalar_mul(&sqr, &sqr, &five); + secp256k1_scalar_get_num(&n, &sqr); + CHECK(secp256k1_num_jacobi(&n, &order) == -1); + /* test multiple of the order*/ + CHECK(secp256k1_num_jacobi(&order, &order) == 0); + + /* check one less than the order */ + secp256k1_scalar_set_int(&small, 1); + secp256k1_scalar_get_num(&n, &small); + secp256k1_num_sub(&n, &order, &n); + CHECK(secp256k1_num_jacobi(&n, &order) == 1); /* sage confirms this is 1 */ +} + +void run_num_smalltests(void) { + int i; + for (i = 0; i < 100*count; i++) { + test_num_negate(); + test_num_add_sub(); + test_num_mod(); + test_num_jacobi(); + } +} +#endif + +/***** SCALAR TESTS *****/ + +void scalar_test(void) { + secp256k1_scalar s; + secp256k1_scalar s1; + secp256k1_scalar s2; +#ifndef USE_NUM_NONE + secp256k1_num snum, s1num, s2num; + secp256k1_num order, half_order; +#endif + unsigned char c[32]; + + /* Set 's' to a random scalar, with value 'snum'. */ + random_scalar_order_test(&s); + + /* Set 's1' to a random scalar, with value 's1num'. */ + random_scalar_order_test(&s1); + + /* Set 's2' to a random scalar, with value 'snum2', and byte array representation 'c'. */ + random_scalar_order_test(&s2); + secp256k1_scalar_get_b32(c, &s2); + +#ifndef USE_NUM_NONE + secp256k1_scalar_get_num(&snum, &s); + secp256k1_scalar_get_num(&s1num, &s1); + secp256k1_scalar_get_num(&s2num, &s2); + + secp256k1_scalar_order_get_num(&order); + half_order = order; + secp256k1_num_shift(&half_order, 1); +#endif + + { + int i; + /* Test that fetching groups of 4 bits from a scalar and recursing n(i)=16*n(i-1)+p(i) reconstructs it. */ + secp256k1_scalar n; + secp256k1_scalar_set_int(&n, 0); + for (i = 0; i < 256; i += 4) { + secp256k1_scalar t; + int j; + secp256k1_scalar_set_int(&t, secp256k1_scalar_get_bits(&s, 256 - 4 - i, 4)); + for (j = 0; j < 4; j++) { + secp256k1_scalar_add(&n, &n, &n); + } + secp256k1_scalar_add(&n, &n, &t); + } + CHECK(secp256k1_scalar_eq(&n, &s)); + } + + { + /* Test that fetching groups of randomly-sized bits from a scalar and recursing n(i)=b*n(i-1)+p(i) reconstructs it. */ + secp256k1_scalar n; + int i = 0; + secp256k1_scalar_set_int(&n, 0); + while (i < 256) { + secp256k1_scalar t; + int j; + int now = secp256k1_rand_int(15) + 1; + if (now + i > 256) { + now = 256 - i; + } + secp256k1_scalar_set_int(&t, secp256k1_scalar_get_bits_var(&s, 256 - now - i, now)); + for (j = 0; j < now; j++) { + secp256k1_scalar_add(&n, &n, &n); + } + secp256k1_scalar_add(&n, &n, &t); + i += now; + } + CHECK(secp256k1_scalar_eq(&n, &s)); + } + +#ifndef USE_NUM_NONE + { + /* Test that adding the scalars together is equal to adding their numbers together modulo the order. */ + secp256k1_num rnum; + secp256k1_num r2num; + secp256k1_scalar r; + secp256k1_num_add(&rnum, &snum, &s2num); + secp256k1_num_mod(&rnum, &order); + secp256k1_scalar_add(&r, &s, &s2); + secp256k1_scalar_get_num(&r2num, &r); + CHECK(secp256k1_num_eq(&rnum, &r2num)); + } + + { + /* Test that multiplying the scalars is equal to multiplying their numbers modulo the order. */ + secp256k1_scalar r; + secp256k1_num r2num; + secp256k1_num rnum; + secp256k1_num_mul(&rnum, &snum, &s2num); + secp256k1_num_mod(&rnum, &order); + secp256k1_scalar_mul(&r, &s, &s2); + secp256k1_scalar_get_num(&r2num, &r); + CHECK(secp256k1_num_eq(&rnum, &r2num)); + /* The result can only be zero if at least one of the factors was zero. */ + CHECK(secp256k1_scalar_is_zero(&r) == (secp256k1_scalar_is_zero(&s) || secp256k1_scalar_is_zero(&s2))); + /* The results can only be equal to one of the factors if that factor was zero, or the other factor was one. */ + CHECK(secp256k1_num_eq(&rnum, &snum) == (secp256k1_scalar_is_zero(&s) || secp256k1_scalar_is_one(&s2))); + CHECK(secp256k1_num_eq(&rnum, &s2num) == (secp256k1_scalar_is_zero(&s2) || secp256k1_scalar_is_one(&s))); + } + + { + secp256k1_scalar neg; + secp256k1_num negnum; + secp256k1_num negnum2; + /* Check that comparison with zero matches comparison with zero on the number. */ + CHECK(secp256k1_num_is_zero(&snum) == secp256k1_scalar_is_zero(&s)); + /* Check that comparison with the half order is equal to testing for high scalar. */ + CHECK(secp256k1_scalar_is_high(&s) == (secp256k1_num_cmp(&snum, &half_order) > 0)); + secp256k1_scalar_negate(&neg, &s); + secp256k1_num_sub(&negnum, &order, &snum); + secp256k1_num_mod(&negnum, &order); + /* Check that comparison with the half order is equal to testing for high scalar after negation. */ + CHECK(secp256k1_scalar_is_high(&neg) == (secp256k1_num_cmp(&negnum, &half_order) > 0)); + /* Negating should change the high property, unless the value was already zero. */ + CHECK((secp256k1_scalar_is_high(&s) == secp256k1_scalar_is_high(&neg)) == secp256k1_scalar_is_zero(&s)); + secp256k1_scalar_get_num(&negnum2, &neg); + /* Negating a scalar should be equal to (order - n) mod order on the number. */ + CHECK(secp256k1_num_eq(&negnum, &negnum2)); + secp256k1_scalar_add(&neg, &neg, &s); + /* Adding a number to its negation should result in zero. */ + CHECK(secp256k1_scalar_is_zero(&neg)); + secp256k1_scalar_negate(&neg, &neg); + /* Negating zero should still result in zero. */ + CHECK(secp256k1_scalar_is_zero(&neg)); + } + + { + /* Test secp256k1_scalar_mul_shift_var. */ + secp256k1_scalar r; + secp256k1_num one; + secp256k1_num rnum; + secp256k1_num rnum2; + unsigned char cone[1] = {0x01}; + unsigned int shift = 256 + secp256k1_rand_int(257); + secp256k1_scalar_mul_shift_var(&r, &s1, &s2, shift); + secp256k1_num_mul(&rnum, &s1num, &s2num); + secp256k1_num_shift(&rnum, shift - 1); + secp256k1_num_set_bin(&one, cone, 1); + secp256k1_num_add(&rnum, &rnum, &one); + secp256k1_num_shift(&rnum, 1); + secp256k1_scalar_get_num(&rnum2, &r); + CHECK(secp256k1_num_eq(&rnum, &rnum2)); + } + + { + /* test secp256k1_scalar_shr_int */ + secp256k1_scalar r; + int i; + random_scalar_order_test(&r); + for (i = 0; i < 100; ++i) { + int low; + int shift = 1 + secp256k1_rand_int(15); + int expected = r.d[0] % (1 << shift); + low = secp256k1_scalar_shr_int(&r, shift); + CHECK(expected == low); + } + } +#endif + + { + /* Test that scalar inverses are equal to the inverse of their number modulo the order. */ + if (!secp256k1_scalar_is_zero(&s)) { + secp256k1_scalar inv; +#ifndef USE_NUM_NONE + secp256k1_num invnum; + secp256k1_num invnum2; +#endif + secp256k1_scalar_inverse(&inv, &s); +#ifndef USE_NUM_NONE + secp256k1_num_mod_inverse(&invnum, &snum, &order); + secp256k1_scalar_get_num(&invnum2, &inv); + CHECK(secp256k1_num_eq(&invnum, &invnum2)); +#endif + secp256k1_scalar_mul(&inv, &inv, &s); + /* Multiplying a scalar with its inverse must result in one. */ + CHECK(secp256k1_scalar_is_one(&inv)); + secp256k1_scalar_inverse(&inv, &inv); + /* Inverting one must result in one. */ + CHECK(secp256k1_scalar_is_one(&inv)); +#ifndef USE_NUM_NONE + secp256k1_scalar_get_num(&invnum, &inv); + CHECK(secp256k1_num_is_one(&invnum)); +#endif + } + } + + { + /* Test commutativity of add. */ + secp256k1_scalar r1, r2; + secp256k1_scalar_add(&r1, &s1, &s2); + secp256k1_scalar_add(&r2, &s2, &s1); + CHECK(secp256k1_scalar_eq(&r1, &r2)); + } + + { + secp256k1_scalar r1, r2; + secp256k1_scalar b; + int i; + /* Test add_bit. */ + int bit = secp256k1_rand_bits(8); + secp256k1_scalar_set_int(&b, 1); + CHECK(secp256k1_scalar_is_one(&b)); + for (i = 0; i < bit; i++) { + secp256k1_scalar_add(&b, &b, &b); + } + r1 = s1; + r2 = s1; + if (!secp256k1_scalar_add(&r1, &r1, &b)) { + /* No overflow happened. */ + secp256k1_scalar_cadd_bit(&r2, bit, 1); + CHECK(secp256k1_scalar_eq(&r1, &r2)); + /* cadd is a noop when flag is zero */ + secp256k1_scalar_cadd_bit(&r2, bit, 0); + CHECK(secp256k1_scalar_eq(&r1, &r2)); + } + } + + { + /* Test commutativity of mul. */ + secp256k1_scalar r1, r2; + secp256k1_scalar_mul(&r1, &s1, &s2); + secp256k1_scalar_mul(&r2, &s2, &s1); + CHECK(secp256k1_scalar_eq(&r1, &r2)); + } + + { + /* Test associativity of add. */ + secp256k1_scalar r1, r2; + secp256k1_scalar_add(&r1, &s1, &s2); + secp256k1_scalar_add(&r1, &r1, &s); + secp256k1_scalar_add(&r2, &s2, &s); + secp256k1_scalar_add(&r2, &s1, &r2); + CHECK(secp256k1_scalar_eq(&r1, &r2)); + } + + { + /* Test associativity of mul. */ + secp256k1_scalar r1, r2; + secp256k1_scalar_mul(&r1, &s1, &s2); + secp256k1_scalar_mul(&r1, &r1, &s); + secp256k1_scalar_mul(&r2, &s2, &s); + secp256k1_scalar_mul(&r2, &s1, &r2); + CHECK(secp256k1_scalar_eq(&r1, &r2)); + } + + { + /* Test distributitivity of mul over add. */ + secp256k1_scalar r1, r2, t; + secp256k1_scalar_add(&r1, &s1, &s2); + secp256k1_scalar_mul(&r1, &r1, &s); + secp256k1_scalar_mul(&r2, &s1, &s); + secp256k1_scalar_mul(&t, &s2, &s); + secp256k1_scalar_add(&r2, &r2, &t); + CHECK(secp256k1_scalar_eq(&r1, &r2)); + } + + { + /* Test square. */ + secp256k1_scalar r1, r2; + secp256k1_scalar_sqr(&r1, &s1); + secp256k1_scalar_mul(&r2, &s1, &s1); + CHECK(secp256k1_scalar_eq(&r1, &r2)); + } + + { + /* Test multiplicative identity. */ + secp256k1_scalar r1, v1; + secp256k1_scalar_set_int(&v1,1); + secp256k1_scalar_mul(&r1, &s1, &v1); + CHECK(secp256k1_scalar_eq(&r1, &s1)); + } + + { + /* Test additive identity. */ + secp256k1_scalar r1, v0; + secp256k1_scalar_set_int(&v0,0); + secp256k1_scalar_add(&r1, &s1, &v0); + CHECK(secp256k1_scalar_eq(&r1, &s1)); + } + + { + /* Test zero product property. */ + secp256k1_scalar r1, v0; + secp256k1_scalar_set_int(&v0,0); + secp256k1_scalar_mul(&r1, &s1, &v0); + CHECK(secp256k1_scalar_eq(&r1, &v0)); + } + +} + +void run_scalar_tests(void) { + int i; + for (i = 0; i < 128 * count; i++) { + scalar_test(); + } + + { + /* (-1)+1 should be zero. */ + secp256k1_scalar s, o; + secp256k1_scalar_set_int(&s, 1); + CHECK(secp256k1_scalar_is_one(&s)); + secp256k1_scalar_negate(&o, &s); + secp256k1_scalar_add(&o, &o, &s); + CHECK(secp256k1_scalar_is_zero(&o)); + secp256k1_scalar_negate(&o, &o); + CHECK(secp256k1_scalar_is_zero(&o)); + } + +#ifndef USE_NUM_NONE + { + /* A scalar with value of the curve order should be 0. */ + secp256k1_num order; + secp256k1_scalar zero; + unsigned char bin[32]; + int overflow = 0; + secp256k1_scalar_order_get_num(&order); + secp256k1_num_get_bin(bin, 32, &order); + secp256k1_scalar_set_b32(&zero, bin, &overflow); + CHECK(overflow == 1); + CHECK(secp256k1_scalar_is_zero(&zero)); + } +#endif + + { + /* Does check_overflow check catch all ones? */ + static const secp256k1_scalar overflowed = SECP256K1_SCALAR_CONST( + 0xFFFFFFFFUL, 0xFFFFFFFFUL, 0xFFFFFFFFUL, 0xFFFFFFFFUL, + 0xFFFFFFFFUL, 0xFFFFFFFFUL, 0xFFFFFFFFUL, 0xFFFFFFFFUL + ); + CHECK(secp256k1_scalar_check_overflow(&overflowed)); + } + + { + /* Static test vectors. + * These were reduced from ~10^12 random vectors based on comparison-decision + * and edge-case coverage on 32-bit and 64-bit implementations. + * The responses were generated with Sage 5.9. + */ + secp256k1_scalar x; + secp256k1_scalar y; + secp256k1_scalar z; + secp256k1_scalar zz; + secp256k1_scalar one; + secp256k1_scalar r1; + secp256k1_scalar r2; +#if defined(USE_SCALAR_INV_NUM) + secp256k1_scalar zzv; +#endif + int overflow; + unsigned char chal[33][2][32] = { + {{0xff, 0xff, 0x03, 0x07, 0x00, 0x00, 0x00, 0x00, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x03, + 0x00, 0x00, 0x00, 0x00, 0x00, 0xf8, 0xff, 0xff, + 0xff, 0xff, 0x03, 0x00, 0xc0, 0xff, 0xff, 0xff}, + {0xff, 0xff, 0xff, 0xff, 0xff, 0x0f, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xf8, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0x03, 0x00, 0x00, 0x00, 0x00, 0xe0, 0xff}}, + {{0xef, 0xff, 0x1f, 0x00, 0x00, 0x00, 0x00, 0x00, + 0xfe, 0xff, 0xff, 0xff, 0xff, 0xff, 0x3f, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xe0, + 0xff, 0xff, 0xff, 0xff, 0xfc, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0x7f, 0x00, 0x80, 0xff}}, + {{0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x06, 0x00, 0x00, + 0x80, 0x00, 0x00, 0x80, 0xff, 0x3f, 0x00, 0x00, + 0x00, 0x00, 0x00, 0xf8, 0xff, 0xff, 0xff, 0x00}, + {0x00, 0x00, 0xfc, 0xff, 0xff, 0xff, 0xff, 0x80, + 0xff, 0xff, 0xff, 0xff, 0xff, 0x0f, 0x00, 0xe0, + 0xff, 0xff, 0xff, 0xff, 0xff, 0x7f, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x7f, 0xff, 0xff, 0xff}}, + {{0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x80, 0x00, 0x00, 0x80, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, + 0x00, 0x1e, 0xf8, 0xff, 0xff, 0xff, 0xfd, 0xff}, + {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x1f, + 0x00, 0x00, 0x00, 0xf8, 0xff, 0x03, 0x00, 0xe0, + 0xff, 0x0f, 0x00, 0x00, 0x00, 0x00, 0xf0, 0xff, + 0xf3, 0xff, 0x03, 0x00, 0x00, 0x00, 0x00, 0x00}}, + {{0x80, 0x00, 0x00, 0x80, 0xff, 0xff, 0xff, 0x00, + 0x00, 0x1c, 0x00, 0x00, 0x00, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xe0, 0xff, 0xff, 0xff, 0x00, + 0x00, 0x00, 0x00, 0x00, 0xe0, 0xff, 0xff, 0xff}, + {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x03, 0x00, + 0xf8, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0x1f, 0x00, 0x00, 0x80, 0xff, 0xff, 0x3f, + 0x00, 0xfe, 0xff, 0xff, 0xff, 0xdf, 0xff, 0xff}}, + {{0xff, 0xff, 0xff, 0xff, 0x00, 0x0f, 0xfc, 0x9f, + 0xff, 0xff, 0xff, 0x00, 0x80, 0x00, 0x00, 0x80, + 0xff, 0x0f, 0xfc, 0xff, 0x7f, 0x00, 0x00, 0x00, + 0x00, 0xf8, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00}, + {0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, + 0x00, 0x00, 0xf8, 0xff, 0x0f, 0xc0, 0xff, 0xff, + 0xff, 0x1f, 0x00, 0x00, 0x00, 0xc0, 0xff, 0xff, + 0xff, 0xff, 0xff, 0x07, 0x80, 0xff, 0xff, 0xff}}, + {{0xff, 0xff, 0xff, 0xff, 0xff, 0x3f, 0x00, 0x00, + 0x80, 0x00, 0x00, 0x80, 0xff, 0xff, 0xff, 0xff, + 0xf7, 0xff, 0xff, 0xef, 0xff, 0xff, 0xff, 0x00, + 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0xf0}, + {0x00, 0x00, 0x00, 0x00, 0xf8, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0x01, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x80, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}}, + {{0x00, 0xf8, 0xff, 0x03, 0xff, 0xff, 0xff, 0x00, + 0x00, 0xfe, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, + 0x80, 0x00, 0x00, 0x80, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0x03, 0xc0, 0xff, 0x0f, 0xfc, 0xff}, + {0xff, 0xff, 0xff, 0xff, 0xff, 0xe0, 0xff, 0xff, + 0xff, 0x01, 0x00, 0x00, 0x00, 0x3f, 0x00, 0xc0, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}}, + {{0x8f, 0x0f, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0xf8, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0x7f, 0x00, 0x00, 0x80, 0x00, 0x00, 0x80, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00}, + {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0x0f, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}}, + {{0x00, 0x00, 0x00, 0xc0, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0x03, 0x00, 0x80, 0x00, 0x00, 0x80, + 0xff, 0xff, 0xff, 0x00, 0x00, 0x80, 0xff, 0x7f}, + {0xff, 0xcf, 0xff, 0xff, 0x01, 0x00, 0x00, 0x00, + 0x00, 0xc0, 0xff, 0xcf, 0xff, 0xff, 0xff, 0xff, + 0xbf, 0xff, 0x0e, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x80, 0xff, 0xff, 0xff, 0xff, 0x00, 0x00, 0x00}}, + {{0x00, 0x00, 0x00, 0x00, 0x00, 0x80, 0xff, 0xff, + 0xff, 0xff, 0x00, 0xfc, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0x00, 0x80, 0x00, 0x00, 0x80, + 0xff, 0x01, 0xfc, 0xff, 0x01, 0x00, 0xfe, 0xff}, + {0xff, 0xff, 0xff, 0x03, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xc0, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x03, 0x00}}, + {{0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, + 0xe0, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0x00, 0xf8, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0x7f, 0x00, 0x00, 0x00, 0x80, 0x00, 0x00, 0x80}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0xf8, 0xff, 0x01, 0x00, 0xf0, 0xff, 0xff, + 0xe0, 0xff, 0x0f, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}}, + {{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0xf8, 0xff, 0x00}, + {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, 0x00, + 0xfc, 0xff, 0xff, 0x3f, 0xf0, 0xff, 0xff, 0x3f, + 0x00, 0x00, 0xf8, 0x07, 0x00, 0x00, 0x00, 0xff, + 0xff, 0xff, 0xff, 0xff, 0x0f, 0x7e, 0x00, 0x00}}, + {{0x00, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x80, 0x00, 0x00, 0x80, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0x1f, 0x00, 0x00, 0xfe, 0x07, 0x00}, + {0x00, 0x00, 0x00, 0xf0, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xfb, 0xff, 0x07, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x60}}, + {{0xff, 0x01, 0x00, 0xff, 0xff, 0xff, 0x0f, 0x00, + 0x80, 0x7f, 0xfe, 0xff, 0xff, 0xff, 0xff, 0x03, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x80, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}, + {0xff, 0xff, 0x1f, 0x00, 0xf0, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0x3f, 0x00, 0x00, 0x00, 0x00}}, + {{0x80, 0x00, 0x00, 0x00, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}, + {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xf1, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x03, + 0x00, 0x00, 0x00, 0xe0, 0xff, 0xff, 0xff, 0xff}}, + {{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, + 0x7e, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0xc0, 0xff, 0xff, 0xcf, 0xff, 0x1f, 0x00, 0x00, + 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0xe0, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0x3f, 0x00, 0x7e, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}}, + {{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0xfc, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7c, 0x00}, + {0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, + 0xff, 0xff, 0x7f, 0x00, 0x80, 0x00, 0x00, 0x00, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, + 0x00, 0x00, 0xe0, 0xff, 0xff, 0xff, 0xff, 0xff}}, + {{0xff, 0xff, 0xff, 0xff, 0xff, 0x1f, 0x00, 0x80, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, + 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00}, + {0xf0, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0x3f, 0x00, 0x00, 0x80, + 0xff, 0x01, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, + 0xff, 0x7f, 0xf8, 0xff, 0xff, 0x1f, 0x00, 0xfe}}, + {{0xff, 0xff, 0xff, 0x3f, 0xf8, 0xff, 0xff, 0xff, + 0xff, 0x03, 0xfe, 0x01, 0x00, 0x00, 0x00, 0x00, + 0xf0, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x07}, + {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, + 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, + 0xff, 0xff, 0xff, 0xff, 0x01, 0x80, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00}}, + {{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, + 0xba, 0xae, 0xdc, 0xe6, 0xaf, 0x48, 0xa0, 0x3b, + 0xbf, 0xd2, 0x5e, 0x8c, 0xd0, 0x36, 0x41, 0x40}}, + {{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}}, + {{0x7f, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}, + {0x7f, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}}, + {{0xff, 0xff, 0xff, 0xff, 0xff, 0x00, 0x00, 0xc0, + 0xff, 0x0f, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0xf0, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x7f}, + {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x01, 0x00, + 0xf0, 0xff, 0xff, 0xff, 0xff, 0x07, 0x00, 0x00, + 0x00, 0x00, 0x00, 0xfe, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0x01, 0xff, 0xff, 0xff}}, + {{0x7f, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02}}, + {{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, + 0xba, 0xae, 0xdc, 0xe6, 0xaf, 0x48, 0xa0, 0x3b, + 0xbf, 0xd2, 0x5e, 0x8c, 0xd0, 0x36, 0x41, 0x40}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01}}, + {{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0x7e, 0x00, 0x00, 0xc0, 0xff, 0xff, 0x07, 0x00, + 0x80, 0x00, 0x00, 0x00, 0x80, 0x00, 0x00, 0x00, + 0xfc, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}, + {0xff, 0x01, 0x00, 0x00, 0x00, 0xe0, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0x1f, 0x00, 0x80, + 0xff, 0xff, 0xff, 0xff, 0xff, 0x03, 0x00, 0x00, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}}, + {{0xff, 0xff, 0xf0, 0xff, 0xff, 0xff, 0xff, 0x00, + 0xf0, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, + 0x00, 0xe0, 0xff, 0xff, 0xff, 0xff, 0xff, 0x01, + 0x80, 0x00, 0x00, 0x80, 0xff, 0xff, 0xff, 0xff}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0xe0, 0xff, 0xff, + 0xff, 0xff, 0x3f, 0x00, 0xf8, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0x3f, 0x00, 0x00, 0xc0, 0xf1, 0x7f, 0x00}}, + {{0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0xc0, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x80, 0x00, 0x00, 0x80, 0xff, 0xff, 0xff, 0x00}, + {0x00, 0xf8, 0xff, 0xff, 0xff, 0xff, 0xff, 0x01, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xf8, 0xff, + 0xff, 0x7f, 0x00, 0x00, 0x00, 0x00, 0x80, 0x1f, + 0x00, 0x00, 0xfc, 0xff, 0xff, 0x01, 0xff, 0xff}}, + {{0x00, 0xfe, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, + 0x80, 0x00, 0x00, 0x80, 0xff, 0x03, 0xe0, 0x01, + 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0xfc, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00}, + {0xff, 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, + 0xfe, 0xff, 0xff, 0xf0, 0x07, 0x00, 0x3c, 0x80, + 0xff, 0xff, 0xff, 0xff, 0xfc, 0xff, 0xff, 0xff, + 0xff, 0xff, 0x07, 0xe0, 0xff, 0x00, 0x00, 0x00}}, + {{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, + 0xfc, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x07, 0xf8, + 0x00, 0x00, 0x00, 0x00, 0x80, 0x00, 0x00, 0x80}, + {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0x0c, 0x80, 0x00, + 0x00, 0x00, 0x00, 0xc0, 0x7f, 0xfe, 0xff, 0x1f, + 0x00, 0xfe, 0xff, 0x03, 0x00, 0x00, 0xfe, 0xff}}, + {{0xff, 0xff, 0x81, 0xff, 0xff, 0xff, 0xff, 0x00, + 0x80, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x83, + 0xff, 0xff, 0x00, 0x00, 0x80, 0x00, 0x00, 0x80, + 0xff, 0xff, 0x7f, 0x00, 0x00, 0x00, 0x00, 0xf0}, + {0xff, 0x01, 0x00, 0x00, 0x00, 0x00, 0xf8, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0x1f, 0x00, 0x00, + 0xf8, 0x07, 0x00, 0x80, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xc7, 0xff, 0xff, 0xe0, 0xff, 0xff, 0xff}}, + {{0x82, 0xc9, 0xfa, 0xb0, 0x68, 0x04, 0xa0, 0x00, + 0x82, 0xc9, 0xfa, 0xb0, 0x68, 0x04, 0xa0, 0x00, + 0xff, 0xff, 0xff, 0xff, 0xff, 0x6f, 0x03, 0xfb, + 0xfa, 0x8a, 0x7d, 0xdf, 0x13, 0x86, 0xe2, 0x03}, + {0x82, 0xc9, 0xfa, 0xb0, 0x68, 0x04, 0xa0, 0x00, + 0x82, 0xc9, 0xfa, 0xb0, 0x68, 0x04, 0xa0, 0x00, + 0xff, 0xff, 0xff, 0xff, 0xff, 0x6f, 0x03, 0xfb, + 0xfa, 0x8a, 0x7d, 0xdf, 0x13, 0x86, 0xe2, 0x03}} + }; + unsigned char res[33][2][32] = { + {{0x0c, 0x3b, 0x0a, 0xca, 0x8d, 0x1a, 0x2f, 0xb9, + 0x8a, 0x7b, 0x53, 0x5a, 0x1f, 0xc5, 0x22, 0xa1, + 0x07, 0x2a, 0x48, 0xea, 0x02, 0xeb, 0xb3, 0xd6, + 0x20, 0x1e, 0x86, 0xd0, 0x95, 0xf6, 0x92, 0x35}, + {0xdc, 0x90, 0x7a, 0x07, 0x2e, 0x1e, 0x44, 0x6d, + 0xf8, 0x15, 0x24, 0x5b, 0x5a, 0x96, 0x37, 0x9c, + 0x37, 0x7b, 0x0d, 0xac, 0x1b, 0x65, 0x58, 0x49, + 0x43, 0xb7, 0x31, 0xbb, 0xa7, 0xf4, 0x97, 0x15}}, + {{0xf1, 0xf7, 0x3a, 0x50, 0xe6, 0x10, 0xba, 0x22, + 0x43, 0x4d, 0x1f, 0x1f, 0x7c, 0x27, 0xca, 0x9c, + 0xb8, 0xb6, 0xa0, 0xfc, 0xd8, 0xc0, 0x05, 0x2f, + 0xf7, 0x08, 0xe1, 0x76, 0xdd, 0xd0, 0x80, 0xc8}, + {0xe3, 0x80, 0x80, 0xb8, 0xdb, 0xe3, 0xa9, 0x77, + 0x00, 0xb0, 0xf5, 0x2e, 0x27, 0xe2, 0x68, 0xc4, + 0x88, 0xe8, 0x04, 0xc1, 0x12, 0xbf, 0x78, 0x59, + 0xe6, 0xa9, 0x7c, 0xe1, 0x81, 0xdd, 0xb9, 0xd5}}, + {{0x96, 0xe2, 0xee, 0x01, 0xa6, 0x80, 0x31, 0xef, + 0x5c, 0xd0, 0x19, 0xb4, 0x7d, 0x5f, 0x79, 0xab, + 0xa1, 0x97, 0xd3, 0x7e, 0x33, 0xbb, 0x86, 0x55, + 0x60, 0x20, 0x10, 0x0d, 0x94, 0x2d, 0x11, 0x7c}, + {0xcc, 0xab, 0xe0, 0xe8, 0x98, 0x65, 0x12, 0x96, + 0x38, 0x5a, 0x1a, 0xf2, 0x85, 0x23, 0x59, 0x5f, + 0xf9, 0xf3, 0xc2, 0x81, 0x70, 0x92, 0x65, 0x12, + 0x9c, 0x65, 0x1e, 0x96, 0x00, 0xef, 0xe7, 0x63}}, + {{0xac, 0x1e, 0x62, 0xc2, 0x59, 0xfc, 0x4e, 0x5c, + 0x83, 0xb0, 0xd0, 0x6f, 0xce, 0x19, 0xf6, 0xbf, + 0xa4, 0xb0, 0xe0, 0x53, 0x66, 0x1f, 0xbf, 0xc9, + 0x33, 0x47, 0x37, 0xa9, 0x3d, 0x5d, 0xb0, 0x48}, + {0x86, 0xb9, 0x2a, 0x7f, 0x8e, 0xa8, 0x60, 0x42, + 0x26, 0x6d, 0x6e, 0x1c, 0xa2, 0xec, 0xe0, 0xe5, + 0x3e, 0x0a, 0x33, 0xbb, 0x61, 0x4c, 0x9f, 0x3c, + 0xd1, 0xdf, 0x49, 0x33, 0xcd, 0x72, 0x78, 0x18}}, + {{0xf7, 0xd3, 0xcd, 0x49, 0x5c, 0x13, 0x22, 0xfb, + 0x2e, 0xb2, 0x2f, 0x27, 0xf5, 0x8a, 0x5d, 0x74, + 0xc1, 0x58, 0xc5, 0xc2, 0x2d, 0x9f, 0x52, 0xc6, + 0x63, 0x9f, 0xba, 0x05, 0x76, 0x45, 0x7a, 0x63}, + {0x8a, 0xfa, 0x55, 0x4d, 0xdd, 0xa3, 0xb2, 0xc3, + 0x44, 0xfd, 0xec, 0x72, 0xde, 0xef, 0xc0, 0x99, + 0xf5, 0x9f, 0xe2, 0x52, 0xb4, 0x05, 0x32, 0x58, + 0x57, 0xc1, 0x8f, 0xea, 0xc3, 0x24, 0x5b, 0x94}}, + {{0x05, 0x83, 0xee, 0xdd, 0x64, 0xf0, 0x14, 0x3b, + 0xa0, 0x14, 0x4a, 0x3a, 0x41, 0x82, 0x7c, 0xa7, + 0x2c, 0xaa, 0xb1, 0x76, 0xbb, 0x59, 0x64, 0x5f, + 0x52, 0xad, 0x25, 0x29, 0x9d, 0x8f, 0x0b, 0xb0}, + {0x7e, 0xe3, 0x7c, 0xca, 0xcd, 0x4f, 0xb0, 0x6d, + 0x7a, 0xb2, 0x3e, 0xa0, 0x08, 0xb9, 0xa8, 0x2d, + 0xc2, 0xf4, 0x99, 0x66, 0xcc, 0xac, 0xd8, 0xb9, + 0x72, 0x2a, 0x4a, 0x3e, 0x0f, 0x7b, 0xbf, 0xf4}}, + {{0x8c, 0x9c, 0x78, 0x2b, 0x39, 0x61, 0x7e, 0xf7, + 0x65, 0x37, 0x66, 0x09, 0x38, 0xb9, 0x6f, 0x70, + 0x78, 0x87, 0xff, 0xcf, 0x93, 0xca, 0x85, 0x06, + 0x44, 0x84, 0xa7, 0xfe, 0xd3, 0xa4, 0xe3, 0x7e}, + {0xa2, 0x56, 0x49, 0x23, 0x54, 0xa5, 0x50, 0xe9, + 0x5f, 0xf0, 0x4d, 0xe7, 0xdc, 0x38, 0x32, 0x79, + 0x4f, 0x1c, 0xb7, 0xe4, 0xbb, 0xf8, 0xbb, 0x2e, + 0x40, 0x41, 0x4b, 0xcc, 0xe3, 0x1e, 0x16, 0x36}}, + {{0x0c, 0x1e, 0xd7, 0x09, 0x25, 0x40, 0x97, 0xcb, + 0x5c, 0x46, 0xa8, 0xda, 0xef, 0x25, 0xd5, 0xe5, + 0x92, 0x4d, 0xcf, 0xa3, 0xc4, 0x5d, 0x35, 0x4a, + 0xe4, 0x61, 0x92, 0xf3, 0xbf, 0x0e, 0xcd, 0xbe}, + {0xe4, 0xaf, 0x0a, 0xb3, 0x30, 0x8b, 0x9b, 0x48, + 0x49, 0x43, 0xc7, 0x64, 0x60, 0x4a, 0x2b, 0x9e, + 0x95, 0x5f, 0x56, 0xe8, 0x35, 0xdc, 0xeb, 0xdc, + 0xc7, 0xc4, 0xfe, 0x30, 0x40, 0xc7, 0xbf, 0xa4}}, + {{0xd4, 0xa0, 0xf5, 0x81, 0x49, 0x6b, 0xb6, 0x8b, + 0x0a, 0x69, 0xf9, 0xfe, 0xa8, 0x32, 0xe5, 0xe0, + 0xa5, 0xcd, 0x02, 0x53, 0xf9, 0x2c, 0xe3, 0x53, + 0x83, 0x36, 0xc6, 0x02, 0xb5, 0xeb, 0x64, 0xb8}, + {0x1d, 0x42, 0xb9, 0xf9, 0xe9, 0xe3, 0x93, 0x2c, + 0x4c, 0xee, 0x6c, 0x5a, 0x47, 0x9e, 0x62, 0x01, + 0x6b, 0x04, 0xfe, 0xa4, 0x30, 0x2b, 0x0d, 0x4f, + 0x71, 0x10, 0xd3, 0x55, 0xca, 0xf3, 0x5e, 0x80}}, + {{0x77, 0x05, 0xf6, 0x0c, 0x15, 0x9b, 0x45, 0xe7, + 0xb9, 0x11, 0xb8, 0xf5, 0xd6, 0xda, 0x73, 0x0c, + 0xda, 0x92, 0xea, 0xd0, 0x9d, 0xd0, 0x18, 0x92, + 0xce, 0x9a, 0xaa, 0xee, 0x0f, 0xef, 0xde, 0x30}, + {0xf1, 0xf1, 0xd6, 0x9b, 0x51, 0xd7, 0x77, 0x62, + 0x52, 0x10, 0xb8, 0x7a, 0x84, 0x9d, 0x15, 0x4e, + 0x07, 0xdc, 0x1e, 0x75, 0x0d, 0x0c, 0x3b, 0xdb, + 0x74, 0x58, 0x62, 0x02, 0x90, 0x54, 0x8b, 0x43}}, + {{0xa6, 0xfe, 0x0b, 0x87, 0x80, 0x43, 0x67, 0x25, + 0x57, 0x5d, 0xec, 0x40, 0x50, 0x08, 0xd5, 0x5d, + 0x43, 0xd7, 0xe0, 0xaa, 0xe0, 0x13, 0xb6, 0xb0, + 0xc0, 0xd4, 0xe5, 0x0d, 0x45, 0x83, 0xd6, 0x13}, + {0x40, 0x45, 0x0a, 0x92, 0x31, 0xea, 0x8c, 0x60, + 0x8c, 0x1f, 0xd8, 0x76, 0x45, 0xb9, 0x29, 0x00, + 0x26, 0x32, 0xd8, 0xa6, 0x96, 0x88, 0xe2, 0xc4, + 0x8b, 0xdb, 0x7f, 0x17, 0x87, 0xcc, 0xc8, 0xf2}}, + {{0xc2, 0x56, 0xe2, 0xb6, 0x1a, 0x81, 0xe7, 0x31, + 0x63, 0x2e, 0xbb, 0x0d, 0x2f, 0x81, 0x67, 0xd4, + 0x22, 0xe2, 0x38, 0x02, 0x25, 0x97, 0xc7, 0x88, + 0x6e, 0xdf, 0xbe, 0x2a, 0xa5, 0x73, 0x63, 0xaa}, + {0x50, 0x45, 0xe2, 0xc3, 0xbd, 0x89, 0xfc, 0x57, + 0xbd, 0x3c, 0xa3, 0x98, 0x7e, 0x7f, 0x36, 0x38, + 0x92, 0x39, 0x1f, 0x0f, 0x81, 0x1a, 0x06, 0x51, + 0x1f, 0x8d, 0x6a, 0xff, 0x47, 0x16, 0x06, 0x9c}}, + {{0x33, 0x95, 0xa2, 0x6f, 0x27, 0x5f, 0x9c, 0x9c, + 0x64, 0x45, 0xcb, 0xd1, 0x3c, 0xee, 0x5e, 0x5f, + 0x48, 0xa6, 0xaf, 0xe3, 0x79, 0xcf, 0xb1, 0xe2, + 0xbf, 0x55, 0x0e, 0xa2, 0x3b, 0x62, 0xf0, 0xe4}, + {0x14, 0xe8, 0x06, 0xe3, 0xbe, 0x7e, 0x67, 0x01, + 0xc5, 0x21, 0x67, 0xd8, 0x54, 0xb5, 0x7f, 0xa4, + 0xf9, 0x75, 0x70, 0x1c, 0xfd, 0x79, 0xdb, 0x86, + 0xad, 0x37, 0x85, 0x83, 0x56, 0x4e, 0xf0, 0xbf}}, + {{0xbc, 0xa6, 0xe0, 0x56, 0x4e, 0xef, 0xfa, 0xf5, + 0x1d, 0x5d, 0x3f, 0x2a, 0x5b, 0x19, 0xab, 0x51, + 0xc5, 0x8b, 0xdd, 0x98, 0x28, 0x35, 0x2f, 0xc3, + 0x81, 0x4f, 0x5c, 0xe5, 0x70, 0xb9, 0xeb, 0x62}, + {0xc4, 0x6d, 0x26, 0xb0, 0x17, 0x6b, 0xfe, 0x6c, + 0x12, 0xf8, 0xe7, 0xc1, 0xf5, 0x2f, 0xfa, 0x91, + 0x13, 0x27, 0xbd, 0x73, 0xcc, 0x33, 0x31, 0x1c, + 0x39, 0xe3, 0x27, 0x6a, 0x95, 0xcf, 0xc5, 0xfb}}, + {{0x30, 0xb2, 0x99, 0x84, 0xf0, 0x18, 0x2a, 0x6e, + 0x1e, 0x27, 0xed, 0xa2, 0x29, 0x99, 0x41, 0x56, + 0xe8, 0xd4, 0x0d, 0xef, 0x99, 0x9c, 0xf3, 0x58, + 0x29, 0x55, 0x1a, 0xc0, 0x68, 0xd6, 0x74, 0xa4}, + {0x07, 0x9c, 0xe7, 0xec, 0xf5, 0x36, 0x73, 0x41, + 0xa3, 0x1c, 0xe5, 0x93, 0x97, 0x6a, 0xfd, 0xf7, + 0x53, 0x18, 0xab, 0xaf, 0xeb, 0x85, 0xbd, 0x92, + 0x90, 0xab, 0x3c, 0xbf, 0x30, 0x82, 0xad, 0xf6}}, + {{0xc6, 0x87, 0x8a, 0x2a, 0xea, 0xc0, 0xa9, 0xec, + 0x6d, 0xd3, 0xdc, 0x32, 0x23, 0xce, 0x62, 0x19, + 0xa4, 0x7e, 0xa8, 0xdd, 0x1c, 0x33, 0xae, 0xd3, + 0x4f, 0x62, 0x9f, 0x52, 0xe7, 0x65, 0x46, 0xf4}, + {0x97, 0x51, 0x27, 0x67, 0x2d, 0xa2, 0x82, 0x87, + 0x98, 0xd3, 0xb6, 0x14, 0x7f, 0x51, 0xd3, 0x9a, + 0x0b, 0xd0, 0x76, 0x81, 0xb2, 0x4f, 0x58, 0x92, + 0xa4, 0x86, 0xa1, 0xa7, 0x09, 0x1d, 0xef, 0x9b}}, + {{0xb3, 0x0f, 0x2b, 0x69, 0x0d, 0x06, 0x90, 0x64, + 0xbd, 0x43, 0x4c, 0x10, 0xe8, 0x98, 0x1c, 0xa3, + 0xe1, 0x68, 0xe9, 0x79, 0x6c, 0x29, 0x51, 0x3f, + 0x41, 0xdc, 0xdf, 0x1f, 0xf3, 0x60, 0xbe, 0x33}, + {0xa1, 0x5f, 0xf7, 0x1d, 0xb4, 0x3e, 0x9b, 0x3c, + 0xe7, 0xbd, 0xb6, 0x06, 0xd5, 0x60, 0x06, 0x6d, + 0x50, 0xd2, 0xf4, 0x1a, 0x31, 0x08, 0xf2, 0xea, + 0x8e, 0xef, 0x5f, 0x7d, 0xb6, 0xd0, 0xc0, 0x27}}, + {{0x62, 0x9a, 0xd9, 0xbb, 0x38, 0x36, 0xce, 0xf7, + 0x5d, 0x2f, 0x13, 0xec, 0xc8, 0x2d, 0x02, 0x8a, + 0x2e, 0x72, 0xf0, 0xe5, 0x15, 0x9d, 0x72, 0xae, + 0xfc, 0xb3, 0x4f, 0x02, 0xea, 0xe1, 0x09, 0xfe}, + {0x00, 0x00, 0x00, 0x00, 0xfa, 0x0a, 0x3d, 0xbc, + 0xad, 0x16, 0x0c, 0xb6, 0xe7, 0x7c, 0x8b, 0x39, + 0x9a, 0x43, 0xbb, 0xe3, 0xc2, 0x55, 0x15, 0x14, + 0x75, 0xac, 0x90, 0x9b, 0x7f, 0x9a, 0x92, 0x00}}, + {{0x8b, 0xac, 0x70, 0x86, 0x29, 0x8f, 0x00, 0x23, + 0x7b, 0x45, 0x30, 0xaa, 0xb8, 0x4c, 0xc7, 0x8d, + 0x4e, 0x47, 0x85, 0xc6, 0x19, 0xe3, 0x96, 0xc2, + 0x9a, 0xa0, 0x12, 0xed, 0x6f, 0xd7, 0x76, 0x16}, + {0x45, 0xaf, 0x7e, 0x33, 0xc7, 0x7f, 0x10, 0x6c, + 0x7c, 0x9f, 0x29, 0xc1, 0xa8, 0x7e, 0x15, 0x84, + 0xe7, 0x7d, 0xc0, 0x6d, 0xab, 0x71, 0x5d, 0xd0, + 0x6b, 0x9f, 0x97, 0xab, 0xcb, 0x51, 0x0c, 0x9f}}, + {{0x9e, 0xc3, 0x92, 0xb4, 0x04, 0x9f, 0xc8, 0xbb, + 0xdd, 0x9e, 0xc6, 0x05, 0xfd, 0x65, 0xec, 0x94, + 0x7f, 0x2c, 0x16, 0xc4, 0x40, 0xac, 0x63, 0x7b, + 0x7d, 0xb8, 0x0c, 0xe4, 0x5b, 0xe3, 0xa7, 0x0e}, + {0x43, 0xf4, 0x44, 0xe8, 0xcc, 0xc8, 0xd4, 0x54, + 0x33, 0x37, 0x50, 0xf2, 0x87, 0x42, 0x2e, 0x00, + 0x49, 0x60, 0x62, 0x02, 0xfd, 0x1a, 0x7c, 0xdb, + 0x29, 0x6c, 0x6d, 0x54, 0x53, 0x08, 0xd1, 0xc8}}, + {{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}}, + {{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01}}, + {{0x27, 0x59, 0xc7, 0x35, 0x60, 0x71, 0xa6, 0xf1, + 0x79, 0xa5, 0xfd, 0x79, 0x16, 0xf3, 0x41, 0xf0, + 0x57, 0xb4, 0x02, 0x97, 0x32, 0xe7, 0xde, 0x59, + 0xe2, 0x2d, 0x9b, 0x11, 0xea, 0x2c, 0x35, 0x92}, + {0x27, 0x59, 0xc7, 0x35, 0x60, 0x71, 0xa6, 0xf1, + 0x79, 0xa5, 0xfd, 0x79, 0x16, 0xf3, 0x41, 0xf0, + 0x57, 0xb4, 0x02, 0x97, 0x32, 0xe7, 0xde, 0x59, + 0xe2, 0x2d, 0x9b, 0x11, 0xea, 0x2c, 0x35, 0x92}}, + {{0x28, 0x56, 0xac, 0x0e, 0x4f, 0x98, 0x09, 0xf0, + 0x49, 0xfa, 0x7f, 0x84, 0xac, 0x7e, 0x50, 0x5b, + 0x17, 0x43, 0x14, 0x89, 0x9c, 0x53, 0xa8, 0x94, + 0x30, 0xf2, 0x11, 0x4d, 0x92, 0x14, 0x27, 0xe8}, + {0x39, 0x7a, 0x84, 0x56, 0x79, 0x9d, 0xec, 0x26, + 0x2c, 0x53, 0xc1, 0x94, 0xc9, 0x8d, 0x9e, 0x9d, + 0x32, 0x1f, 0xdd, 0x84, 0x04, 0xe8, 0xe2, 0x0a, + 0x6b, 0xbe, 0xbb, 0x42, 0x40, 0x67, 0x30, 0x6c}}, + {{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + 0x45, 0x51, 0x23, 0x19, 0x50, 0xb7, 0x5f, 0xc4, + 0x40, 0x2d, 0xa1, 0x73, 0x2f, 0xc9, 0xbe, 0xbd}, + {0x27, 0x59, 0xc7, 0x35, 0x60, 0x71, 0xa6, 0xf1, + 0x79, 0xa5, 0xfd, 0x79, 0x16, 0xf3, 0x41, 0xf0, + 0x57, 0xb4, 0x02, 0x97, 0x32, 0xe7, 0xde, 0x59, + 0xe2, 0x2d, 0x9b, 0x11, 0xea, 0x2c, 0x35, 0x92}}, + {{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, + 0xba, 0xae, 0xdc, 0xe6, 0xaf, 0x48, 0xa0, 0x3b, + 0xbf, 0xd2, 0x5e, 0x8c, 0xd0, 0x36, 0x41, 0x40}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01}}, + {{0x1c, 0xc4, 0xf7, 0xda, 0x0f, 0x65, 0xca, 0x39, + 0x70, 0x52, 0x92, 0x8e, 0xc3, 0xc8, 0x15, 0xea, + 0x7f, 0x10, 0x9e, 0x77, 0x4b, 0x6e, 0x2d, 0xdf, + 0xe8, 0x30, 0x9d, 0xda, 0xe8, 0x9a, 0x65, 0xae}, + {0x02, 0xb0, 0x16, 0xb1, 0x1d, 0xc8, 0x57, 0x7b, + 0xa2, 0x3a, 0xa2, 0xa3, 0x38, 0x5c, 0x8f, 0xeb, + 0x66, 0x37, 0x91, 0xa8, 0x5f, 0xef, 0x04, 0xf6, + 0x59, 0x75, 0xe1, 0xee, 0x92, 0xf6, 0x0e, 0x30}}, + {{0x8d, 0x76, 0x14, 0xa4, 0x14, 0x06, 0x9f, 0x9a, + 0xdf, 0x4a, 0x85, 0xa7, 0x6b, 0xbf, 0x29, 0x6f, + 0xbc, 0x34, 0x87, 0x5d, 0xeb, 0xbb, 0x2e, 0xa9, + 0xc9, 0x1f, 0x58, 0xd6, 0x9a, 0x82, 0xa0, 0x56}, + {0xd4, 0xb9, 0xdb, 0x88, 0x1d, 0x04, 0xe9, 0x93, + 0x8d, 0x3f, 0x20, 0xd5, 0x86, 0xa8, 0x83, 0x07, + 0xdb, 0x09, 0xd8, 0x22, 0x1f, 0x7f, 0xf1, 0x71, + 0xc8, 0xe7, 0x5d, 0x47, 0xaf, 0x8b, 0x72, 0xe9}}, + {{0x83, 0xb9, 0x39, 0xb2, 0xa4, 0xdf, 0x46, 0x87, + 0xc2, 0xb8, 0xf1, 0xe6, 0x4c, 0xd1, 0xe2, 0xa9, + 0xe4, 0x70, 0x30, 0x34, 0xbc, 0x52, 0x7c, 0x55, + 0xa6, 0xec, 0x80, 0xa4, 0xe5, 0xd2, 0xdc, 0x73}, + {0x08, 0xf1, 0x03, 0xcf, 0x16, 0x73, 0xe8, 0x7d, + 0xb6, 0x7e, 0x9b, 0xc0, 0xb4, 0xc2, 0xa5, 0x86, + 0x02, 0x77, 0xd5, 0x27, 0x86, 0xa5, 0x15, 0xfb, + 0xae, 0x9b, 0x8c, 0xa9, 0xf9, 0xf8, 0xa8, 0x4a}}, + {{0x8b, 0x00, 0x49, 0xdb, 0xfa, 0xf0, 0x1b, 0xa2, + 0xed, 0x8a, 0x9a, 0x7a, 0x36, 0x78, 0x4a, 0xc7, + 0xf7, 0xad, 0x39, 0xd0, 0x6c, 0x65, 0x7a, 0x41, + 0xce, 0xd6, 0xd6, 0x4c, 0x20, 0x21, 0x6b, 0xc7}, + {0xc6, 0xca, 0x78, 0x1d, 0x32, 0x6c, 0x6c, 0x06, + 0x91, 0xf2, 0x1a, 0xe8, 0x43, 0x16, 0xea, 0x04, + 0x3c, 0x1f, 0x07, 0x85, 0xf7, 0x09, 0x22, 0x08, + 0xba, 0x13, 0xfd, 0x78, 0x1e, 0x3f, 0x6f, 0x62}}, + {{0x25, 0x9b, 0x7c, 0xb0, 0xac, 0x72, 0x6f, 0xb2, + 0xe3, 0x53, 0x84, 0x7a, 0x1a, 0x9a, 0x98, 0x9b, + 0x44, 0xd3, 0x59, 0xd0, 0x8e, 0x57, 0x41, 0x40, + 0x78, 0xa7, 0x30, 0x2f, 0x4c, 0x9c, 0xb9, 0x68}, + {0xb7, 0x75, 0x03, 0x63, 0x61, 0xc2, 0x48, 0x6e, + 0x12, 0x3d, 0xbf, 0x4b, 0x27, 0xdf, 0xb1, 0x7a, + 0xff, 0x4e, 0x31, 0x07, 0x83, 0xf4, 0x62, 0x5b, + 0x19, 0xa5, 0xac, 0xa0, 0x32, 0x58, 0x0d, 0xa7}}, + {{0x43, 0x4f, 0x10, 0xa4, 0xca, 0xdb, 0x38, 0x67, + 0xfa, 0xae, 0x96, 0xb5, 0x6d, 0x97, 0xff, 0x1f, + 0xb6, 0x83, 0x43, 0xd3, 0xa0, 0x2d, 0x70, 0x7a, + 0x64, 0x05, 0x4c, 0xa7, 0xc1, 0xa5, 0x21, 0x51}, + {0xe4, 0xf1, 0x23, 0x84, 0xe1, 0xb5, 0x9d, 0xf2, + 0xb8, 0x73, 0x8b, 0x45, 0x2b, 0x35, 0x46, 0x38, + 0x10, 0x2b, 0x50, 0xf8, 0x8b, 0x35, 0xcd, 0x34, + 0xc8, 0x0e, 0xf6, 0xdb, 0x09, 0x35, 0xf0, 0xda}}, + {{0xdb, 0x21, 0x5c, 0x8d, 0x83, 0x1d, 0xb3, 0x34, + 0xc7, 0x0e, 0x43, 0xa1, 0x58, 0x79, 0x67, 0x13, + 0x1e, 0x86, 0x5d, 0x89, 0x63, 0xe6, 0x0a, 0x46, + 0x5c, 0x02, 0x97, 0x1b, 0x62, 0x43, 0x86, 0xf5}, + {0xdb, 0x21, 0x5c, 0x8d, 0x83, 0x1d, 0xb3, 0x34, + 0xc7, 0x0e, 0x43, 0xa1, 0x58, 0x79, 0x67, 0x13, + 0x1e, 0x86, 0x5d, 0x89, 0x63, 0xe6, 0x0a, 0x46, + 0x5c, 0x02, 0x97, 0x1b, 0x62, 0x43, 0x86, 0xf5}} + }; + secp256k1_scalar_set_int(&one, 1); + for (i = 0; i < 33; i++) { + secp256k1_scalar_set_b32(&x, chal[i][0], &overflow); + CHECK(!overflow); + secp256k1_scalar_set_b32(&y, chal[i][1], &overflow); + CHECK(!overflow); + secp256k1_scalar_set_b32(&r1, res[i][0], &overflow); + CHECK(!overflow); + secp256k1_scalar_set_b32(&r2, res[i][1], &overflow); + CHECK(!overflow); + secp256k1_scalar_mul(&z, &x, &y); + CHECK(!secp256k1_scalar_check_overflow(&z)); + CHECK(secp256k1_scalar_eq(&r1, &z)); + if (!secp256k1_scalar_is_zero(&y)) { + secp256k1_scalar_inverse(&zz, &y); + CHECK(!secp256k1_scalar_check_overflow(&zz)); +#if defined(USE_SCALAR_INV_NUM) + secp256k1_scalar_inverse_var(&zzv, &y); + CHECK(secp256k1_scalar_eq(&zzv, &zz)); +#endif + secp256k1_scalar_mul(&z, &z, &zz); + CHECK(!secp256k1_scalar_check_overflow(&z)); + CHECK(secp256k1_scalar_eq(&x, &z)); + secp256k1_scalar_mul(&zz, &zz, &y); + CHECK(!secp256k1_scalar_check_overflow(&zz)); + CHECK(secp256k1_scalar_eq(&one, &zz)); + } + secp256k1_scalar_mul(&z, &x, &x); + CHECK(!secp256k1_scalar_check_overflow(&z)); + secp256k1_scalar_sqr(&zz, &x); + CHECK(!secp256k1_scalar_check_overflow(&zz)); + CHECK(secp256k1_scalar_eq(&zz, &z)); + CHECK(secp256k1_scalar_eq(&r2, &zz)); + } + } +} + +/***** FIELD TESTS *****/ + +void random_fe(secp256k1_fe *x) { + unsigned char bin[32]; + do { + secp256k1_rand256(bin); + if (secp256k1_fe_set_b32(x, bin)) { + return; + } + } while(1); +} + +void random_fe_test(secp256k1_fe *x) { + unsigned char bin[32]; + do { + secp256k1_rand256_test(bin); + if (secp256k1_fe_set_b32(x, bin)) { + return; + } + } while(1); +} + +void random_fe_non_zero(secp256k1_fe *nz) { + int tries = 10; + while (--tries >= 0) { + random_fe(nz); + secp256k1_fe_normalize(nz); + if (!secp256k1_fe_is_zero(nz)) { + break; + } + } + /* Infinitesimal probability of spurious failure here */ + CHECK(tries >= 0); +} + +void random_fe_non_square(secp256k1_fe *ns) { + secp256k1_fe r; + random_fe_non_zero(ns); + if (secp256k1_fe_sqrt(&r, ns)) { + secp256k1_fe_negate(ns, ns, 1); + } +} + +int check_fe_equal(const secp256k1_fe *a, const secp256k1_fe *b) { + secp256k1_fe an = *a; + secp256k1_fe bn = *b; + secp256k1_fe_normalize_weak(&an); + secp256k1_fe_normalize_var(&bn); + return secp256k1_fe_equal_var(&an, &bn); +} + +int check_fe_inverse(const secp256k1_fe *a, const secp256k1_fe *ai) { + secp256k1_fe x; + secp256k1_fe one = SECP256K1_FE_CONST(0, 0, 0, 0, 0, 0, 0, 1); + secp256k1_fe_mul(&x, a, ai); + return check_fe_equal(&x, &one); +} + +void run_field_convert(void) { + static const unsigned char b32[32] = { + 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, + 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, + 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, + 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x40 + }; + static const secp256k1_fe_storage fes = SECP256K1_FE_STORAGE_CONST( + 0x00010203UL, 0x04050607UL, 0x11121314UL, 0x15161718UL, + 0x22232425UL, 0x26272829UL, 0x33343536UL, 0x37383940UL + ); + static const secp256k1_fe fe = SECP256K1_FE_CONST( + 0x00010203UL, 0x04050607UL, 0x11121314UL, 0x15161718UL, + 0x22232425UL, 0x26272829UL, 0x33343536UL, 0x37383940UL + ); + secp256k1_fe fe2; + unsigned char b322[32]; + secp256k1_fe_storage fes2; + /* Check conversions to fe. */ + CHECK(secp256k1_fe_set_b32(&fe2, b32)); + CHECK(secp256k1_fe_equal_var(&fe, &fe2)); + secp256k1_fe_from_storage(&fe2, &fes); + CHECK(secp256k1_fe_equal_var(&fe, &fe2)); + /* Check conversion from fe. */ + secp256k1_fe_get_b32(b322, &fe); + CHECK(memcmp(b322, b32, 32) == 0); + secp256k1_fe_to_storage(&fes2, &fe); + CHECK(memcmp(&fes2, &fes, sizeof(fes)) == 0); +} + +int fe_memcmp(const secp256k1_fe *a, const secp256k1_fe *b) { + secp256k1_fe t = *b; +#ifdef VERIFY + t.magnitude = a->magnitude; + t.normalized = a->normalized; +#endif + return memcmp(a, &t, sizeof(secp256k1_fe)); +} + +void run_field_misc(void) { + secp256k1_fe x; + secp256k1_fe y; + secp256k1_fe z; + secp256k1_fe q; + secp256k1_fe fe5 = SECP256K1_FE_CONST(0, 0, 0, 0, 0, 0, 0, 5); + int i, j; + for (i = 0; i < 5*count; i++) { + secp256k1_fe_storage xs, ys, zs; + random_fe(&x); + random_fe_non_zero(&y); + /* Test the fe equality and comparison operations. */ + CHECK(secp256k1_fe_cmp_var(&x, &x) == 0); + CHECK(secp256k1_fe_equal_var(&x, &x)); + z = x; + secp256k1_fe_add(&z,&y); + /* Test fe conditional move; z is not normalized here. */ + q = x; + secp256k1_fe_cmov(&x, &z, 0); +#ifdef VERIFY + CHECK(!x.normalized && x.magnitude == z.magnitude); +#endif + secp256k1_fe_cmov(&x, &x, 1); + CHECK(fe_memcmp(&x, &z) != 0); + CHECK(fe_memcmp(&x, &q) == 0); + secp256k1_fe_cmov(&q, &z, 1); +#ifdef VERIFY + CHECK(!q.normalized && q.magnitude == z.magnitude); +#endif + CHECK(fe_memcmp(&q, &z) == 0); + secp256k1_fe_normalize_var(&x); + secp256k1_fe_normalize_var(&z); + CHECK(!secp256k1_fe_equal_var(&x, &z)); + secp256k1_fe_normalize_var(&q); + secp256k1_fe_cmov(&q, &z, (i&1)); +#ifdef VERIFY + CHECK(q.normalized && q.magnitude == 1); +#endif + for (j = 0; j < 6; j++) { + secp256k1_fe_negate(&z, &z, j+1); + secp256k1_fe_normalize_var(&q); + secp256k1_fe_cmov(&q, &z, (j&1)); +#ifdef VERIFY + CHECK(!q.normalized && q.magnitude == (j+2)); +#endif + } + secp256k1_fe_normalize_var(&z); + /* Test storage conversion and conditional moves. */ + secp256k1_fe_to_storage(&xs, &x); + secp256k1_fe_to_storage(&ys, &y); + secp256k1_fe_to_storage(&zs, &z); + secp256k1_fe_storage_cmov(&zs, &xs, 0); + secp256k1_fe_storage_cmov(&zs, &zs, 1); + CHECK(memcmp(&xs, &zs, sizeof(xs)) != 0); + secp256k1_fe_storage_cmov(&ys, &xs, 1); + CHECK(memcmp(&xs, &ys, sizeof(xs)) == 0); + secp256k1_fe_from_storage(&x, &xs); + secp256k1_fe_from_storage(&y, &ys); + secp256k1_fe_from_storage(&z, &zs); + /* Test that mul_int, mul, and add agree. */ + secp256k1_fe_add(&y, &x); + secp256k1_fe_add(&y, &x); + z = x; + secp256k1_fe_mul_int(&z, 3); + CHECK(check_fe_equal(&y, &z)); + secp256k1_fe_add(&y, &x); + secp256k1_fe_add(&z, &x); + CHECK(check_fe_equal(&z, &y)); + z = x; + secp256k1_fe_mul_int(&z, 5); + secp256k1_fe_mul(&q, &x, &fe5); + CHECK(check_fe_equal(&z, &q)); + secp256k1_fe_negate(&x, &x, 1); + secp256k1_fe_add(&z, &x); + secp256k1_fe_add(&q, &x); + CHECK(check_fe_equal(&y, &z)); + CHECK(check_fe_equal(&q, &y)); + } +} + +void run_field_inv(void) { + secp256k1_fe x, xi, xii; + int i; + for (i = 0; i < 10*count; i++) { + random_fe_non_zero(&x); + secp256k1_fe_inv(&xi, &x); + CHECK(check_fe_inverse(&x, &xi)); + secp256k1_fe_inv(&xii, &xi); + CHECK(check_fe_equal(&x, &xii)); + } +} + +void run_field_inv_var(void) { + secp256k1_fe x, xi, xii; + int i; + for (i = 0; i < 10*count; i++) { + random_fe_non_zero(&x); + secp256k1_fe_inv_var(&xi, &x); + CHECK(check_fe_inverse(&x, &xi)); + secp256k1_fe_inv_var(&xii, &xi); + CHECK(check_fe_equal(&x, &xii)); + } +} + +void run_field_inv_all_var(void) { + secp256k1_fe x[16], xi[16], xii[16]; + int i; + /* Check it's safe to call for 0 elements */ + secp256k1_fe_inv_all_var(xi, x, 0); + for (i = 0; i < count; i++) { + size_t j; + size_t len = secp256k1_rand_int(15) + 1; + for (j = 0; j < len; j++) { + random_fe_non_zero(&x[j]); + } + secp256k1_fe_inv_all_var(xi, x, len); + for (j = 0; j < len; j++) { + CHECK(check_fe_inverse(&x[j], &xi[j])); + } + secp256k1_fe_inv_all_var(xii, xi, len); + for (j = 0; j < len; j++) { + CHECK(check_fe_equal(&x[j], &xii[j])); + } + } +} + +void run_sqr(void) { + secp256k1_fe x, s; + + { + int i; + secp256k1_fe_set_int(&x, 1); + secp256k1_fe_negate(&x, &x, 1); + + for (i = 1; i <= 512; ++i) { + secp256k1_fe_mul_int(&x, 2); + secp256k1_fe_normalize(&x); + secp256k1_fe_sqr(&s, &x); + } + } +} + +void test_sqrt(const secp256k1_fe *a, const secp256k1_fe *k) { + secp256k1_fe r1, r2; + int v = secp256k1_fe_sqrt(&r1, a); + CHECK((v == 0) == (k == NULL)); + + if (k != NULL) { + /* Check that the returned root is +/- the given known answer */ + secp256k1_fe_negate(&r2, &r1, 1); + secp256k1_fe_add(&r1, k); secp256k1_fe_add(&r2, k); + secp256k1_fe_normalize(&r1); secp256k1_fe_normalize(&r2); + CHECK(secp256k1_fe_is_zero(&r1) || secp256k1_fe_is_zero(&r2)); + } +} + +void run_sqrt(void) { + secp256k1_fe ns, x, s, t; + int i; + + /* Check sqrt(0) is 0 */ + secp256k1_fe_set_int(&x, 0); + secp256k1_fe_sqr(&s, &x); + test_sqrt(&s, &x); + + /* Check sqrt of small squares (and their negatives) */ + for (i = 1; i <= 100; i++) { + secp256k1_fe_set_int(&x, i); + secp256k1_fe_sqr(&s, &x); + test_sqrt(&s, &x); + secp256k1_fe_negate(&t, &s, 1); + test_sqrt(&t, NULL); + } + + /* Consistency checks for large random values */ + for (i = 0; i < 10; i++) { + int j; + random_fe_non_square(&ns); + for (j = 0; j < count; j++) { + random_fe(&x); + secp256k1_fe_sqr(&s, &x); + test_sqrt(&s, &x); + secp256k1_fe_negate(&t, &s, 1); + test_sqrt(&t, NULL); + secp256k1_fe_mul(&t, &s, &ns); + test_sqrt(&t, NULL); + } + } +} + +/***** GROUP TESTS *****/ + +void ge_equals_ge(const secp256k1_ge *a, const secp256k1_ge *b) { + CHECK(a->infinity == b->infinity); + if (a->infinity) { + return; + } + CHECK(secp256k1_fe_equal_var(&a->x, &b->x)); + CHECK(secp256k1_fe_equal_var(&a->y, &b->y)); +} + +/* This compares jacobian points including their Z, not just their geometric meaning. */ +int gej_xyz_equals_gej(const secp256k1_gej *a, const secp256k1_gej *b) { + secp256k1_gej a2; + secp256k1_gej b2; + int ret = 1; + ret &= a->infinity == b->infinity; + if (ret && !a->infinity) { + a2 = *a; + b2 = *b; + secp256k1_fe_normalize(&a2.x); + secp256k1_fe_normalize(&a2.y); + secp256k1_fe_normalize(&a2.z); + secp256k1_fe_normalize(&b2.x); + secp256k1_fe_normalize(&b2.y); + secp256k1_fe_normalize(&b2.z); + ret &= secp256k1_fe_cmp_var(&a2.x, &b2.x) == 0; + ret &= secp256k1_fe_cmp_var(&a2.y, &b2.y) == 0; + ret &= secp256k1_fe_cmp_var(&a2.z, &b2.z) == 0; + } + return ret; +} + +void ge_equals_gej(const secp256k1_ge *a, const secp256k1_gej *b) { + secp256k1_fe z2s; + secp256k1_fe u1, u2, s1, s2; + CHECK(a->infinity == b->infinity); + if (a->infinity) { + return; + } + /* Check a.x * b.z^2 == b.x && a.y * b.z^3 == b.y, to avoid inverses. */ + secp256k1_fe_sqr(&z2s, &b->z); + secp256k1_fe_mul(&u1, &a->x, &z2s); + u2 = b->x; secp256k1_fe_normalize_weak(&u2); + secp256k1_fe_mul(&s1, &a->y, &z2s); secp256k1_fe_mul(&s1, &s1, &b->z); + s2 = b->y; secp256k1_fe_normalize_weak(&s2); + CHECK(secp256k1_fe_equal_var(&u1, &u2)); + CHECK(secp256k1_fe_equal_var(&s1, &s2)); +} + +void test_ge(void) { + int i, i1; +#ifdef USE_ENDOMORPHISM + int runs = 6; +#else + int runs = 4; +#endif + /* Points: (infinity, p1, p1, -p1, -p1, p2, p2, -p2, -p2, p3, p3, -p3, -p3, p4, p4, -p4, -p4). + * The second in each pair of identical points uses a random Z coordinate in the Jacobian form. + * All magnitudes are randomized. + * All 17*17 combinations of points are added to each other, using all applicable methods. + * + * When the endomorphism code is compiled in, p5 = lambda*p1 and p6 = lambda^2*p1 are added as well. + */ + secp256k1_ge *ge = (secp256k1_ge *)checked_malloc(&ctx->error_callback, sizeof(secp256k1_ge) * (1 + 4 * runs)); + secp256k1_gej *gej = (secp256k1_gej *)checked_malloc(&ctx->error_callback, sizeof(secp256k1_gej) * (1 + 4 * runs)); + secp256k1_fe *zinv = (secp256k1_fe *)checked_malloc(&ctx->error_callback, sizeof(secp256k1_fe) * (1 + 4 * runs)); + secp256k1_fe zf; + secp256k1_fe zfi2, zfi3; + + secp256k1_gej_set_infinity(&gej[0]); + secp256k1_ge_clear(&ge[0]); + secp256k1_ge_set_gej_var(&ge[0], &gej[0]); + for (i = 0; i < runs; i++) { + int j; + secp256k1_ge g; + random_group_element_test(&g); +#ifdef USE_ENDOMORPHISM + if (i >= runs - 2) { + secp256k1_ge_mul_lambda(&g, &ge[1]); + } + if (i >= runs - 1) { + secp256k1_ge_mul_lambda(&g, &g); + } +#endif + ge[1 + 4 * i] = g; + ge[2 + 4 * i] = g; + secp256k1_ge_neg(&ge[3 + 4 * i], &g); + secp256k1_ge_neg(&ge[4 + 4 * i], &g); + secp256k1_gej_set_ge(&gej[1 + 4 * i], &ge[1 + 4 * i]); + random_group_element_jacobian_test(&gej[2 + 4 * i], &ge[2 + 4 * i]); + secp256k1_gej_set_ge(&gej[3 + 4 * i], &ge[3 + 4 * i]); + random_group_element_jacobian_test(&gej[4 + 4 * i], &ge[4 + 4 * i]); + for (j = 0; j < 4; j++) { + random_field_element_magnitude(&ge[1 + j + 4 * i].x); + random_field_element_magnitude(&ge[1 + j + 4 * i].y); + random_field_element_magnitude(&gej[1 + j + 4 * i].x); + random_field_element_magnitude(&gej[1 + j + 4 * i].y); + random_field_element_magnitude(&gej[1 + j + 4 * i].z); + } + } + + /* Compute z inverses. */ + { + secp256k1_fe *zs = checked_malloc(&ctx->error_callback, sizeof(secp256k1_fe) * (1 + 4 * runs)); + for (i = 0; i < 4 * runs + 1; i++) { + if (i == 0) { + /* The point at infinity does not have a meaningful z inverse. Any should do. */ + do { + random_field_element_test(&zs[i]); + } while(secp256k1_fe_is_zero(&zs[i])); + } else { + zs[i] = gej[i].z; + } + } + secp256k1_fe_inv_all_var(zinv, zs, 4 * runs + 1); + free(zs); + } + + /* Generate random zf, and zfi2 = 1/zf^2, zfi3 = 1/zf^3 */ + do { + random_field_element_test(&zf); + } while(secp256k1_fe_is_zero(&zf)); + random_field_element_magnitude(&zf); + secp256k1_fe_inv_var(&zfi3, &zf); + secp256k1_fe_sqr(&zfi2, &zfi3); + secp256k1_fe_mul(&zfi3, &zfi3, &zfi2); + + for (i1 = 0; i1 < 1 + 4 * runs; i1++) { + int i2; + for (i2 = 0; i2 < 1 + 4 * runs; i2++) { + /* Compute reference result using gej + gej (var). */ + secp256k1_gej refj, resj; + secp256k1_ge ref; + secp256k1_fe zr; + secp256k1_gej_add_var(&refj, &gej[i1], &gej[i2], secp256k1_gej_is_infinity(&gej[i1]) ? NULL : &zr); + /* Check Z ratio. */ + if (!secp256k1_gej_is_infinity(&gej[i1]) && !secp256k1_gej_is_infinity(&refj)) { + secp256k1_fe zrz; secp256k1_fe_mul(&zrz, &zr, &gej[i1].z); + CHECK(secp256k1_fe_equal_var(&zrz, &refj.z)); + } + secp256k1_ge_set_gej_var(&ref, &refj); + + /* Test gej + ge with Z ratio result (var). */ + secp256k1_gej_add_ge_var(&resj, &gej[i1], &ge[i2], secp256k1_gej_is_infinity(&gej[i1]) ? NULL : &zr); + ge_equals_gej(&ref, &resj); + if (!secp256k1_gej_is_infinity(&gej[i1]) && !secp256k1_gej_is_infinity(&resj)) { + secp256k1_fe zrz; secp256k1_fe_mul(&zrz, &zr, &gej[i1].z); + CHECK(secp256k1_fe_equal_var(&zrz, &resj.z)); + } + + /* Test gej + ge (var, with additional Z factor). */ + { + secp256k1_ge ge2_zfi = ge[i2]; /* the second term with x and y rescaled for z = 1/zf */ + secp256k1_fe_mul(&ge2_zfi.x, &ge2_zfi.x, &zfi2); + secp256k1_fe_mul(&ge2_zfi.y, &ge2_zfi.y, &zfi3); + random_field_element_magnitude(&ge2_zfi.x); + random_field_element_magnitude(&ge2_zfi.y); + secp256k1_gej_add_zinv_var(&resj, &gej[i1], &ge2_zfi, &zf); + ge_equals_gej(&ref, &resj); + } + + /* Test gej + ge (const). */ + if (i2 != 0) { + /* secp256k1_gej_add_ge does not support its second argument being infinity. */ + secp256k1_gej_add_ge(&resj, &gej[i1], &ge[i2]); + ge_equals_gej(&ref, &resj); + } + + /* Test doubling (var). */ + if ((i1 == 0 && i2 == 0) || ((i1 + 3)/4 == (i2 + 3)/4 && ((i1 + 3)%4)/2 == ((i2 + 3)%4)/2)) { + secp256k1_fe zr2; + /* Normal doubling with Z ratio result. */ + secp256k1_gej_double_var(&resj, &gej[i1], &zr2); + ge_equals_gej(&ref, &resj); + /* Check Z ratio. */ + secp256k1_fe_mul(&zr2, &zr2, &gej[i1].z); + CHECK(secp256k1_fe_equal_var(&zr2, &resj.z)); + /* Normal doubling. */ + secp256k1_gej_double_var(&resj, &gej[i2], NULL); + ge_equals_gej(&ref, &resj); + } + + /* Test adding opposites. */ + if ((i1 == 0 && i2 == 0) || ((i1 + 3)/4 == (i2 + 3)/4 && ((i1 + 3)%4)/2 != ((i2 + 3)%4)/2)) { + CHECK(secp256k1_ge_is_infinity(&ref)); + } + + /* Test adding infinity. */ + if (i1 == 0) { + CHECK(secp256k1_ge_is_infinity(&ge[i1])); + CHECK(secp256k1_gej_is_infinity(&gej[i1])); + ge_equals_gej(&ref, &gej[i2]); + } + if (i2 == 0) { + CHECK(secp256k1_ge_is_infinity(&ge[i2])); + CHECK(secp256k1_gej_is_infinity(&gej[i2])); + ge_equals_gej(&ref, &gej[i1]); + } + } + } + + /* Test adding all points together in random order equals infinity. */ + { + secp256k1_gej sum = SECP256K1_GEJ_CONST_INFINITY; + secp256k1_gej *gej_shuffled = (secp256k1_gej *)checked_malloc(&ctx->error_callback, (4 * runs + 1) * sizeof(secp256k1_gej)); + for (i = 0; i < 4 * runs + 1; i++) { + gej_shuffled[i] = gej[i]; + } + for (i = 0; i < 4 * runs + 1; i++) { + int swap = i + secp256k1_rand_int(4 * runs + 1 - i); + if (swap != i) { + secp256k1_gej t = gej_shuffled[i]; + gej_shuffled[i] = gej_shuffled[swap]; + gej_shuffled[swap] = t; + } + } + for (i = 0; i < 4 * runs + 1; i++) { + secp256k1_gej_add_var(&sum, &sum, &gej_shuffled[i], NULL); + } + CHECK(secp256k1_gej_is_infinity(&sum)); + free(gej_shuffled); + } + + /* Test batch gej -> ge conversion with and without known z ratios. */ + { + secp256k1_fe *zr = (secp256k1_fe *)checked_malloc(&ctx->error_callback, (4 * runs + 1) * sizeof(secp256k1_fe)); + secp256k1_ge *ge_set_all = (secp256k1_ge *)checked_malloc(&ctx->error_callback, (4 * runs + 1) * sizeof(secp256k1_ge)); + for (i = 0; i < 4 * runs + 1; i++) { + /* Compute gej[i + 1].z / gez[i].z (with gej[n].z taken to be 1). */ + if (i < 4 * runs) { + secp256k1_fe_mul(&zr[i + 1], &zinv[i], &gej[i + 1].z); + } + } + secp256k1_ge_set_all_gej_var(ge_set_all, gej, 4 * runs + 1); + for (i = 0; i < 4 * runs + 1; i++) { + secp256k1_fe s; + random_fe_non_zero(&s); + secp256k1_gej_rescale(&gej[i], &s); + ge_equals_gej(&ge_set_all[i], &gej[i]); + } + free(ge_set_all); + free(zr); + } + + /* Test batch gej -> ge conversion with many infinities. */ + for (i = 0; i < 4 * runs + 1; i++) { + random_group_element_test(&ge[i]); + /* randomly set half the points to infinity */ + if(secp256k1_fe_is_odd(&ge[i].x)) { + secp256k1_ge_set_infinity(&ge[i]); + } + secp256k1_gej_set_ge(&gej[i], &ge[i]); + } + /* batch invert */ + secp256k1_ge_set_all_gej_var(ge, gej, 4 * runs + 1); + /* check result */ + for (i = 0; i < 4 * runs + 1; i++) { + ge_equals_gej(&ge[i], &gej[i]); + } + + free(ge); + free(gej); + free(zinv); +} + +void test_add_neg_y_diff_x(void) { + /* The point of this test is to check that we can add two points + * whose y-coordinates are negatives of each other but whose x + * coordinates differ. If the x-coordinates were the same, these + * points would be negatives of each other and their sum is + * infinity. This is cool because it "covers up" any degeneracy + * in the addition algorithm that would cause the xy coordinates + * of the sum to be wrong (since infinity has no xy coordinates). + * HOWEVER, if the x-coordinates are different, infinity is the + * wrong answer, and such degeneracies are exposed. This is the + * root of https://github.com/bitcoin-core/secp256k1/issues/257 + * which this test is a regression test for. + * + * These points were generated in sage as + * # secp256k1 params + * F = FiniteField (0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFC2F) + * C = EllipticCurve ([F (0), F (7)]) + * G = C.lift_x(0x79BE667EF9DCBBAC55A06295CE870B07029BFCDB2DCE28D959F2815B16F81798) + * N = FiniteField(G.order()) + * + * # endomorphism values (lambda is 1^{1/3} in N, beta is 1^{1/3} in F) + * x = polygen(N) + * lam = (1 - x^3).roots()[1][0] + * + * # random "bad pair" + * P = C.random_element() + * Q = -int(lam) * P + * print " P: %x %x" % P.xy() + * print " Q: %x %x" % Q.xy() + * print "P + Q: %x %x" % (P + Q).xy() + */ + secp256k1_gej aj = SECP256K1_GEJ_CONST( + 0x8d24cd95, 0x0a355af1, 0x3c543505, 0x44238d30, + 0x0643d79f, 0x05a59614, 0x2f8ec030, 0xd58977cb, + 0x001e337a, 0x38093dcd, 0x6c0f386d, 0x0b1293a8, + 0x4d72c879, 0xd7681924, 0x44e6d2f3, 0x9190117d + ); + secp256k1_gej bj = SECP256K1_GEJ_CONST( + 0xc7b74206, 0x1f788cd9, 0xabd0937d, 0x164a0d86, + 0x95f6ff75, 0xf19a4ce9, 0xd013bd7b, 0xbf92d2a7, + 0xffe1cc85, 0xc7f6c232, 0x93f0c792, 0xf4ed6c57, + 0xb28d3786, 0x2897e6db, 0xbb192d0b, 0x6e6feab2 + ); + secp256k1_gej sumj = SECP256K1_GEJ_CONST( + 0x671a63c0, 0x3efdad4c, 0x389a7798, 0x24356027, + 0xb3d69010, 0x278625c3, 0x5c86d390, 0x184a8f7a, + 0x5f6409c2, 0x2ce01f2b, 0x511fd375, 0x25071d08, + 0xda651801, 0x70e95caf, 0x8f0d893c, 0xbed8fbbe + ); + secp256k1_ge b; + secp256k1_gej resj; + secp256k1_ge res; + secp256k1_ge_set_gej(&b, &bj); + + secp256k1_gej_add_var(&resj, &aj, &bj, NULL); + secp256k1_ge_set_gej(&res, &resj); + ge_equals_gej(&res, &sumj); + + secp256k1_gej_add_ge(&resj, &aj, &b); + secp256k1_ge_set_gej(&res, &resj); + ge_equals_gej(&res, &sumj); + + secp256k1_gej_add_ge_var(&resj, &aj, &b, NULL); + secp256k1_ge_set_gej(&res, &resj); + ge_equals_gej(&res, &sumj); +} + +void run_ge(void) { + int i; + for (i = 0; i < count * 32; i++) { + test_ge(); + } + test_add_neg_y_diff_x(); +} + +void test_ec_combine(void) { + secp256k1_scalar sum = SECP256K1_SCALAR_CONST(0, 0, 0, 0, 0, 0, 0, 0); + secp256k1_pubkey data[6]; + const secp256k1_pubkey* d[6]; + secp256k1_pubkey sd; + secp256k1_pubkey sd2; + secp256k1_gej Qj; + secp256k1_ge Q; + int i; + for (i = 1; i <= 6; i++) { + secp256k1_scalar s; + random_scalar_order_test(&s); + secp256k1_scalar_add(&sum, &sum, &s); + secp256k1_ecmult_gen(&ctx->ecmult_gen_ctx, &Qj, &s); + secp256k1_ge_set_gej(&Q, &Qj); + secp256k1_pubkey_save(&data[i - 1], &Q); + d[i - 1] = &data[i - 1]; + secp256k1_ecmult_gen(&ctx->ecmult_gen_ctx, &Qj, &sum); + secp256k1_ge_set_gej(&Q, &Qj); + secp256k1_pubkey_save(&sd, &Q); + CHECK(secp256k1_ec_pubkey_combine(ctx, &sd2, d, i) == 1); + CHECK(memcmp(&sd, &sd2, sizeof(sd)) == 0); + } +} + +void run_ec_combine(void) { + int i; + for (i = 0; i < count * 8; i++) { + test_ec_combine(); + } +} + +void test_group_decompress(const secp256k1_fe* x) { + /* The input itself, normalized. */ + secp256k1_fe fex = *x; + secp256k1_fe fez; + /* Results of set_xquad_var, set_xo_var(..., 0), set_xo_var(..., 1). */ + secp256k1_ge ge_quad, ge_even, ge_odd; + secp256k1_gej gej_quad; + /* Return values of the above calls. */ + int res_quad, res_even, res_odd; + + secp256k1_fe_normalize_var(&fex); + + res_quad = secp256k1_ge_set_xquad(&ge_quad, &fex); + res_even = secp256k1_ge_set_xo_var(&ge_even, &fex, 0); + res_odd = secp256k1_ge_set_xo_var(&ge_odd, &fex, 1); + + CHECK(res_quad == res_even); + CHECK(res_quad == res_odd); + + if (res_quad) { + secp256k1_fe_normalize_var(&ge_quad.x); + secp256k1_fe_normalize_var(&ge_odd.x); + secp256k1_fe_normalize_var(&ge_even.x); + secp256k1_fe_normalize_var(&ge_quad.y); + secp256k1_fe_normalize_var(&ge_odd.y); + secp256k1_fe_normalize_var(&ge_even.y); + + /* No infinity allowed. */ + CHECK(!ge_quad.infinity); + CHECK(!ge_even.infinity); + CHECK(!ge_odd.infinity); + + /* Check that the x coordinates check out. */ + CHECK(secp256k1_fe_equal_var(&ge_quad.x, x)); + CHECK(secp256k1_fe_equal_var(&ge_even.x, x)); + CHECK(secp256k1_fe_equal_var(&ge_odd.x, x)); + + /* Check that the Y coordinate result in ge_quad is a square. */ + CHECK(secp256k1_fe_is_quad_var(&ge_quad.y)); + + /* Check odd/even Y in ge_odd, ge_even. */ + CHECK(secp256k1_fe_is_odd(&ge_odd.y)); + CHECK(!secp256k1_fe_is_odd(&ge_even.y)); + + /* Check secp256k1_gej_has_quad_y_var. */ + secp256k1_gej_set_ge(&gej_quad, &ge_quad); + CHECK(secp256k1_gej_has_quad_y_var(&gej_quad)); + do { + random_fe_test(&fez); + } while (secp256k1_fe_is_zero(&fez)); + secp256k1_gej_rescale(&gej_quad, &fez); + CHECK(secp256k1_gej_has_quad_y_var(&gej_quad)); + secp256k1_gej_neg(&gej_quad, &gej_quad); + CHECK(!secp256k1_gej_has_quad_y_var(&gej_quad)); + do { + random_fe_test(&fez); + } while (secp256k1_fe_is_zero(&fez)); + secp256k1_gej_rescale(&gej_quad, &fez); + CHECK(!secp256k1_gej_has_quad_y_var(&gej_quad)); + secp256k1_gej_neg(&gej_quad, &gej_quad); + CHECK(secp256k1_gej_has_quad_y_var(&gej_quad)); + } +} + +void run_group_decompress(void) { + int i; + for (i = 0; i < count * 4; i++) { + secp256k1_fe fe; + random_fe_test(&fe); + test_group_decompress(&fe); + } +} + +/***** ECMULT TESTS *****/ + +void run_ecmult_chain(void) { + /* random starting point A (on the curve) */ + secp256k1_gej a = SECP256K1_GEJ_CONST( + 0x8b30bbe9, 0xae2a9906, 0x96b22f67, 0x0709dff3, + 0x727fd8bc, 0x04d3362c, 0x6c7bf458, 0xe2846004, + 0xa357ae91, 0x5c4a6528, 0x1309edf2, 0x0504740f, + 0x0eb33439, 0x90216b4f, 0x81063cb6, 0x5f2f7e0f + ); + /* two random initial factors xn and gn */ + secp256k1_scalar xn = SECP256K1_SCALAR_CONST( + 0x84cc5452, 0xf7fde1ed, 0xb4d38a8c, 0xe9b1b84c, + 0xcef31f14, 0x6e569be9, 0x705d357a, 0x42985407 + ); + secp256k1_scalar gn = SECP256K1_SCALAR_CONST( + 0xa1e58d22, 0x553dcd42, 0xb2398062, 0x5d4c57a9, + 0x6e9323d4, 0x2b3152e5, 0xca2c3990, 0xedc7c9de + ); + /* two small multipliers to be applied to xn and gn in every iteration: */ + static const secp256k1_scalar xf = SECP256K1_SCALAR_CONST(0, 0, 0, 0, 0, 0, 0, 0x1337); + static const secp256k1_scalar gf = SECP256K1_SCALAR_CONST(0, 0, 0, 0, 0, 0, 0, 0x7113); + /* accumulators with the resulting coefficients to A and G */ + secp256k1_scalar ae = SECP256K1_SCALAR_CONST(0, 0, 0, 0, 0, 0, 0, 1); + secp256k1_scalar ge = SECP256K1_SCALAR_CONST(0, 0, 0, 0, 0, 0, 0, 0); + /* actual points */ + secp256k1_gej x; + secp256k1_gej x2; + int i; + + /* the point being computed */ + x = a; + for (i = 0; i < 200*count; i++) { + /* in each iteration, compute X = xn*X + gn*G; */ + secp256k1_ecmult(&ctx->ecmult_ctx, &x, &x, &xn, &gn); + /* also compute ae and ge: the actual accumulated factors for A and G */ + /* if X was (ae*A+ge*G), xn*X + gn*G results in (xn*ae*A + (xn*ge+gn)*G) */ + secp256k1_scalar_mul(&ae, &ae, &xn); + secp256k1_scalar_mul(&ge, &ge, &xn); + secp256k1_scalar_add(&ge, &ge, &gn); + /* modify xn and gn */ + secp256k1_scalar_mul(&xn, &xn, &xf); + secp256k1_scalar_mul(&gn, &gn, &gf); + + /* verify */ + if (i == 19999) { + /* expected result after 19999 iterations */ + secp256k1_gej rp = SECP256K1_GEJ_CONST( + 0xD6E96687, 0xF9B10D09, 0x2A6F3543, 0x9D86CEBE, + 0xA4535D0D, 0x409F5358, 0x6440BD74, 0xB933E830, + 0xB95CBCA2, 0xC77DA786, 0x539BE8FD, 0x53354D2D, + 0x3B4F566A, 0xE6580454, 0x07ED6015, 0xEE1B2A88 + ); + + secp256k1_gej_neg(&rp, &rp); + secp256k1_gej_add_var(&rp, &rp, &x, NULL); + CHECK(secp256k1_gej_is_infinity(&rp)); + } + } + /* redo the computation, but directly with the resulting ae and ge coefficients: */ + secp256k1_ecmult(&ctx->ecmult_ctx, &x2, &a, &ae, &ge); + secp256k1_gej_neg(&x2, &x2); + secp256k1_gej_add_var(&x2, &x2, &x, NULL); + CHECK(secp256k1_gej_is_infinity(&x2)); +} + +void test_point_times_order(const secp256k1_gej *point) { + /* X * (point + G) + (order-X) * (pointer + G) = 0 */ + secp256k1_scalar x; + secp256k1_scalar nx; + secp256k1_scalar zero = SECP256K1_SCALAR_CONST(0, 0, 0, 0, 0, 0, 0, 0); + secp256k1_scalar one = SECP256K1_SCALAR_CONST(0, 0, 0, 0, 0, 0, 0, 1); + secp256k1_gej res1, res2; + secp256k1_ge res3; + unsigned char pub[65]; + size_t psize = 65; + random_scalar_order_test(&x); + secp256k1_scalar_negate(&nx, &x); + secp256k1_ecmult(&ctx->ecmult_ctx, &res1, point, &x, &x); /* calc res1 = x * point + x * G; */ + secp256k1_ecmult(&ctx->ecmult_ctx, &res2, point, &nx, &nx); /* calc res2 = (order - x) * point + (order - x) * G; */ + secp256k1_gej_add_var(&res1, &res1, &res2, NULL); + CHECK(secp256k1_gej_is_infinity(&res1)); + CHECK(secp256k1_gej_is_valid_var(&res1) == 0); + secp256k1_ge_set_gej(&res3, &res1); + CHECK(secp256k1_ge_is_infinity(&res3)); + CHECK(secp256k1_ge_is_valid_var(&res3) == 0); + CHECK(secp256k1_eckey_pubkey_serialize(&res3, pub, &psize, 0) == 0); + psize = 65; + CHECK(secp256k1_eckey_pubkey_serialize(&res3, pub, &psize, 1) == 0); + /* check zero/one edge cases */ + secp256k1_ecmult(&ctx->ecmult_ctx, &res1, point, &zero, &zero); + secp256k1_ge_set_gej(&res3, &res1); + CHECK(secp256k1_ge_is_infinity(&res3)); + secp256k1_ecmult(&ctx->ecmult_ctx, &res1, point, &one, &zero); + secp256k1_ge_set_gej(&res3, &res1); + ge_equals_gej(&res3, point); + secp256k1_ecmult(&ctx->ecmult_ctx, &res1, point, &zero, &one); + secp256k1_ge_set_gej(&res3, &res1); + ge_equals_ge(&res3, &secp256k1_ge_const_g); +} + +void run_point_times_order(void) { + int i; + secp256k1_fe x = SECP256K1_FE_CONST(0, 0, 0, 0, 0, 0, 0, 2); + static const secp256k1_fe xr = SECP256K1_FE_CONST( + 0x7603CB59, 0xB0EF6C63, 0xFE608479, 0x2A0C378C, + 0xDB3233A8, 0x0F8A9A09, 0xA877DEAD, 0x31B38C45 + ); + for (i = 0; i < 500; i++) { + secp256k1_ge p; + if (secp256k1_ge_set_xo_var(&p, &x, 1)) { + secp256k1_gej j; + CHECK(secp256k1_ge_is_valid_var(&p)); + secp256k1_gej_set_ge(&j, &p); + CHECK(secp256k1_gej_is_valid_var(&j)); + test_point_times_order(&j); + } + secp256k1_fe_sqr(&x, &x); + } + secp256k1_fe_normalize_var(&x); + CHECK(secp256k1_fe_equal_var(&x, &xr)); +} + +void ecmult_const_random_mult(void) { + /* random starting point A (on the curve) */ + secp256k1_ge a = SECP256K1_GE_CONST( + 0x6d986544, 0x57ff52b8, 0xcf1b8126, 0x5b802a5b, + 0xa97f9263, 0xb1e88044, 0x93351325, 0x91bc450a, + 0x535c59f7, 0x325e5d2b, 0xc391fbe8, 0x3c12787c, + 0x337e4a98, 0xe82a9011, 0x0123ba37, 0xdd769c7d + ); + /* random initial factor xn */ + secp256k1_scalar xn = SECP256K1_SCALAR_CONST( + 0x649d4f77, 0xc4242df7, 0x7f2079c9, 0x14530327, + 0xa31b876a, 0xd2d8ce2a, 0x2236d5c6, 0xd7b2029b + ); + /* expected xn * A (from sage) */ + secp256k1_ge expected_b = SECP256K1_GE_CONST( + 0x23773684, 0x4d209dc7, 0x098a786f, 0x20d06fcd, + 0x070a38bf, 0xc11ac651, 0x03004319, 0x1e2a8786, + 0xed8c3b8e, 0xc06dd57b, 0xd06ea66e, 0x45492b0f, + 0xb84e4e1b, 0xfb77e21f, 0x96baae2a, 0x63dec956 + ); + secp256k1_gej b; + secp256k1_ecmult_const(&b, &a, &xn, 256); + + CHECK(secp256k1_ge_is_valid_var(&a)); + ge_equals_gej(&expected_b, &b); +} + +void ecmult_const_commutativity(void) { + secp256k1_scalar a; + secp256k1_scalar b; + secp256k1_gej res1; + secp256k1_gej res2; + secp256k1_ge mid1; + secp256k1_ge mid2; + random_scalar_order_test(&a); + random_scalar_order_test(&b); + + secp256k1_ecmult_const(&res1, &secp256k1_ge_const_g, &a, 256); + secp256k1_ecmult_const(&res2, &secp256k1_ge_const_g, &b, 256); + secp256k1_ge_set_gej(&mid1, &res1); + secp256k1_ge_set_gej(&mid2, &res2); + secp256k1_ecmult_const(&res1, &mid1, &b, 256); + secp256k1_ecmult_const(&res2, &mid2, &a, 256); + secp256k1_ge_set_gej(&mid1, &res1); + secp256k1_ge_set_gej(&mid2, &res2); + ge_equals_ge(&mid1, &mid2); +} + +void ecmult_const_mult_zero_one(void) { + secp256k1_scalar zero = SECP256K1_SCALAR_CONST(0, 0, 0, 0, 0, 0, 0, 0); + secp256k1_scalar one = SECP256K1_SCALAR_CONST(0, 0, 0, 0, 0, 0, 0, 1); + secp256k1_scalar negone; + secp256k1_gej res1; + secp256k1_ge res2; + secp256k1_ge point; + secp256k1_scalar_negate(&negone, &one); + + random_group_element_test(&point); + secp256k1_ecmult_const(&res1, &point, &zero, 3); + secp256k1_ge_set_gej(&res2, &res1); + CHECK(secp256k1_ge_is_infinity(&res2)); + secp256k1_ecmult_const(&res1, &point, &one, 2); + secp256k1_ge_set_gej(&res2, &res1); + ge_equals_ge(&res2, &point); + secp256k1_ecmult_const(&res1, &point, &negone, 256); + secp256k1_gej_neg(&res1, &res1); + secp256k1_ge_set_gej(&res2, &res1); + ge_equals_ge(&res2, &point); +} + +void ecmult_const_chain_multiply(void) { + /* Check known result (randomly generated test problem from sage) */ + const secp256k1_scalar scalar = SECP256K1_SCALAR_CONST( + 0x4968d524, 0x2abf9b7a, 0x466abbcf, 0x34b11b6d, + 0xcd83d307, 0x827bed62, 0x05fad0ce, 0x18fae63b + ); + const secp256k1_gej expected_point = SECP256K1_GEJ_CONST( + 0x5494c15d, 0x32099706, 0xc2395f94, 0x348745fd, + 0x757ce30e, 0x4e8c90fb, 0xa2bad184, 0xf883c69f, + 0x5d195d20, 0xe191bf7f, 0x1be3e55f, 0x56a80196, + 0x6071ad01, 0xf1462f66, 0xc997fa94, 0xdb858435 + ); + secp256k1_gej point; + secp256k1_ge res; + int i; + + secp256k1_gej_set_ge(&point, &secp256k1_ge_const_g); + for (i = 0; i < 100; ++i) { + secp256k1_ge tmp; + secp256k1_ge_set_gej(&tmp, &point); + secp256k1_ecmult_const(&point, &tmp, &scalar, 256); + } + secp256k1_ge_set_gej(&res, &point); + ge_equals_gej(&res, &expected_point); +} + +void run_ecmult_const_tests(void) { + ecmult_const_mult_zero_one(); + ecmult_const_random_mult(); + ecmult_const_commutativity(); + ecmult_const_chain_multiply(); +} + +typedef struct { + secp256k1_scalar *sc; + secp256k1_ge *pt; +} ecmult_multi_data; + +static int ecmult_multi_callback(secp256k1_scalar *sc, secp256k1_ge *pt, size_t idx, void *cbdata) { + ecmult_multi_data *data = (ecmult_multi_data*) cbdata; + *sc = data->sc[idx]; + *pt = data->pt[idx]; + return 1; +} + +static int ecmult_multi_false_callback(secp256k1_scalar *sc, secp256k1_ge *pt, size_t idx, void *cbdata) { + (void)sc; + (void)pt; + (void)idx; + (void)cbdata; + return 0; +} + +void test_ecmult_multi(secp256k1_scratch *scratch, secp256k1_ecmult_multi_func ecmult_multi) { + int ncount; + secp256k1_scalar szero; + secp256k1_scalar sc[32]; + secp256k1_ge pt[32]; + secp256k1_gej r; + secp256k1_gej r2; + ecmult_multi_data data; + + data.sc = sc; + data.pt = pt; + secp256k1_scalar_set_int(&szero, 0); + + /* No points to multiply */ + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, NULL, ecmult_multi_callback, &data, 0)); + + /* Check 1- and 2-point multiplies against ecmult */ + for (ncount = 0; ncount < count; ncount++) { + secp256k1_ge ptg; + secp256k1_gej ptgj; + random_scalar_order(&sc[0]); + random_scalar_order(&sc[1]); + + random_group_element_test(&ptg); + secp256k1_gej_set_ge(&ptgj, &ptg); + pt[0] = ptg; + pt[1] = secp256k1_ge_const_g; + + /* only G scalar */ + secp256k1_ecmult(&ctx->ecmult_ctx, &r2, &ptgj, &szero, &sc[0]); + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &sc[0], ecmult_multi_callback, &data, 0)); + secp256k1_gej_neg(&r2, &r2); + secp256k1_gej_add_var(&r, &r, &r2, NULL); + CHECK(secp256k1_gej_is_infinity(&r)); + + /* 1-point */ + secp256k1_ecmult(&ctx->ecmult_ctx, &r2, &ptgj, &sc[0], &szero); + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_callback, &data, 1)); + secp256k1_gej_neg(&r2, &r2); + secp256k1_gej_add_var(&r, &r, &r2, NULL); + CHECK(secp256k1_gej_is_infinity(&r)); + + /* Try to multiply 1 point, but callback returns false */ + CHECK(!ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_false_callback, &data, 1)); + + /* 2-point */ + secp256k1_ecmult(&ctx->ecmult_ctx, &r2, &ptgj, &sc[0], &sc[1]); + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_callback, &data, 2)); + secp256k1_gej_neg(&r2, &r2); + secp256k1_gej_add_var(&r, &r, &r2, NULL); + CHECK(secp256k1_gej_is_infinity(&r)); + + /* 2-point with G scalar */ + secp256k1_ecmult(&ctx->ecmult_ctx, &r2, &ptgj, &sc[0], &sc[1]); + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &sc[1], ecmult_multi_callback, &data, 1)); + secp256k1_gej_neg(&r2, &r2); + secp256k1_gej_add_var(&r, &r, &r2, NULL); + CHECK(secp256k1_gej_is_infinity(&r)); + } + + /* Check infinite outputs of various forms */ + for (ncount = 0; ncount < count; ncount++) { + secp256k1_ge ptg; + size_t i, j; + size_t sizes[] = { 2, 10, 32 }; + + for (j = 0; j < 3; j++) { + for (i = 0; i < 32; i++) { + random_scalar_order(&sc[i]); + secp256k1_ge_set_infinity(&pt[i]); + } + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_callback, &data, sizes[j])); + CHECK(secp256k1_gej_is_infinity(&r)); + } + + for (j = 0; j < 3; j++) { + for (i = 0; i < 32; i++) { + random_group_element_test(&ptg); + pt[i] = ptg; + secp256k1_scalar_set_int(&sc[i], 0); + } + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_callback, &data, sizes[j])); + CHECK(secp256k1_gej_is_infinity(&r)); + } + + for (j = 0; j < 3; j++) { + random_group_element_test(&ptg); + for (i = 0; i < 16; i++) { + random_scalar_order(&sc[2*i]); + secp256k1_scalar_negate(&sc[2*i + 1], &sc[2*i]); + pt[2 * i] = ptg; + pt[2 * i + 1] = ptg; + } + + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_callback, &data, sizes[j])); + CHECK(secp256k1_gej_is_infinity(&r)); + + random_scalar_order(&sc[0]); + for (i = 0; i < 16; i++) { + random_group_element_test(&ptg); + + sc[2*i] = sc[0]; + sc[2*i+1] = sc[0]; + pt[2 * i] = ptg; + secp256k1_ge_neg(&pt[2*i+1], &pt[2*i]); + } + + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_callback, &data, sizes[j])); + CHECK(secp256k1_gej_is_infinity(&r)); + } + + random_group_element_test(&ptg); + secp256k1_scalar_set_int(&sc[0], 0); + pt[0] = ptg; + for (i = 1; i < 32; i++) { + pt[i] = ptg; + + random_scalar_order(&sc[i]); + secp256k1_scalar_add(&sc[0], &sc[0], &sc[i]); + secp256k1_scalar_negate(&sc[i], &sc[i]); + } + + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_callback, &data, 32)); + CHECK(secp256k1_gej_is_infinity(&r)); + } + + /* Check random points, constant scalar */ + for (ncount = 0; ncount < count; ncount++) { + size_t i; + secp256k1_gej_set_infinity(&r); + + random_scalar_order(&sc[0]); + for (i = 0; i < 20; i++) { + secp256k1_ge ptg; + sc[i] = sc[0]; + random_group_element_test(&ptg); + pt[i] = ptg; + secp256k1_gej_add_ge_var(&r, &r, &pt[i], NULL); + } + + secp256k1_ecmult(&ctx->ecmult_ctx, &r2, &r, &sc[0], &szero); + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_callback, &data, 20)); + secp256k1_gej_neg(&r2, &r2); + secp256k1_gej_add_var(&r, &r, &r2, NULL); + CHECK(secp256k1_gej_is_infinity(&r)); + } + + /* Check random scalars, constant point */ + for (ncount = 0; ncount < count; ncount++) { + size_t i; + secp256k1_ge ptg; + secp256k1_gej p0j; + secp256k1_scalar rs; + secp256k1_scalar_set_int(&rs, 0); + + random_group_element_test(&ptg); + for (i = 0; i < 20; i++) { + random_scalar_order(&sc[i]); + pt[i] = ptg; + secp256k1_scalar_add(&rs, &rs, &sc[i]); + } + + secp256k1_gej_set_ge(&p0j, &pt[0]); + secp256k1_ecmult(&ctx->ecmult_ctx, &r2, &p0j, &rs, &szero); + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_callback, &data, 20)); + secp256k1_gej_neg(&r2, &r2); + secp256k1_gej_add_var(&r, &r, &r2, NULL); + CHECK(secp256k1_gej_is_infinity(&r)); + } + + /* Sanity check that zero scalars don't cause problems */ + for (ncount = 0; ncount < 20; ncount++) { + random_scalar_order(&sc[ncount]); + random_group_element_test(&pt[ncount]); + } + + secp256k1_scalar_clear(&sc[0]); + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_callback, &data, 20)); + secp256k1_scalar_clear(&sc[1]); + secp256k1_scalar_clear(&sc[2]); + secp256k1_scalar_clear(&sc[3]); + secp256k1_scalar_clear(&sc[4]); + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_callback, &data, 6)); + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &szero, ecmult_multi_callback, &data, 5)); + CHECK(secp256k1_gej_is_infinity(&r)); + + /* Run through s0*(t0*P) + s1*(t1*P) exhaustively for many small values of s0, s1, t0, t1 */ + { + const size_t TOP = 8; + size_t s0i, s1i; + size_t t0i, t1i; + secp256k1_ge ptg; + secp256k1_gej ptgj; + + random_group_element_test(&ptg); + secp256k1_gej_set_ge(&ptgj, &ptg); + + for(t0i = 0; t0i < TOP; t0i++) { + for(t1i = 0; t1i < TOP; t1i++) { + secp256k1_gej t0p, t1p; + secp256k1_scalar t0, t1; + + secp256k1_scalar_set_int(&t0, (t0i + 1) / 2); + secp256k1_scalar_cond_negate(&t0, t0i & 1); + secp256k1_scalar_set_int(&t1, (t1i + 1) / 2); + secp256k1_scalar_cond_negate(&t1, t1i & 1); + + secp256k1_ecmult(&ctx->ecmult_ctx, &t0p, &ptgj, &t0, &szero); + secp256k1_ecmult(&ctx->ecmult_ctx, &t1p, &ptgj, &t1, &szero); + + for(s0i = 0; s0i < TOP; s0i++) { + for(s1i = 0; s1i < TOP; s1i++) { + secp256k1_scalar tmp1, tmp2; + secp256k1_gej expected, actual; + + secp256k1_ge_set_gej(&pt[0], &t0p); + secp256k1_ge_set_gej(&pt[1], &t1p); + + secp256k1_scalar_set_int(&sc[0], (s0i + 1) / 2); + secp256k1_scalar_cond_negate(&sc[0], s0i & 1); + secp256k1_scalar_set_int(&sc[1], (s1i + 1) / 2); + secp256k1_scalar_cond_negate(&sc[1], s1i & 1); + + secp256k1_scalar_mul(&tmp1, &t0, &sc[0]); + secp256k1_scalar_mul(&tmp2, &t1, &sc[1]); + secp256k1_scalar_add(&tmp1, &tmp1, &tmp2); + + secp256k1_ecmult(&ctx->ecmult_ctx, &expected, &ptgj, &tmp1, &szero); + CHECK(ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &actual, &szero, ecmult_multi_callback, &data, 2)); + secp256k1_gej_neg(&expected, &expected); + secp256k1_gej_add_var(&actual, &actual, &expected, NULL); + CHECK(secp256k1_gej_is_infinity(&actual)); + } + } + } + } + } +} + +void test_ecmult_multi_batch_single(secp256k1_ecmult_multi_func ecmult_multi) { + secp256k1_scalar szero; + secp256k1_scalar sc[32]; + secp256k1_ge pt[32]; + secp256k1_gej r; + ecmult_multi_data data; + secp256k1_scratch *scratch_empty; + + data.sc = sc; + data.pt = pt; + secp256k1_scalar_set_int(&szero, 0); + + /* Try to multiply 1 point, but scratch space is empty.*/ + scratch_empty = secp256k1_scratch_create(&ctx->error_callback, 0); + CHECK(!ecmult_multi(&ctx->error_callback, &ctx->ecmult_ctx, scratch_empty, &r, &szero, ecmult_multi_callback, &data, 1)); + secp256k1_scratch_destroy(&ctx->error_callback, scratch_empty); +} + +void test_secp256k1_pippenger_bucket_window_inv(void) { + int i; + + CHECK(secp256k1_pippenger_bucket_window_inv(0) == 0); + for(i = 1; i <= PIPPENGER_MAX_BUCKET_WINDOW; i++) { +#ifdef USE_ENDOMORPHISM + /* Bucket_window of 8 is not used with endo */ + if (i == 8) { + continue; + } +#endif + CHECK(secp256k1_pippenger_bucket_window(secp256k1_pippenger_bucket_window_inv(i)) == i); + if (i != PIPPENGER_MAX_BUCKET_WINDOW) { + CHECK(secp256k1_pippenger_bucket_window(secp256k1_pippenger_bucket_window_inv(i)+1) > i); + } + } +} + +/** + * Probabilistically test the function returning the maximum number of possible points + * for a given scratch space. + */ +void test_ecmult_multi_pippenger_max_points(void) { + size_t scratch_size = secp256k1_rand_int(256); + size_t max_size = secp256k1_pippenger_scratch_size(secp256k1_pippenger_bucket_window_inv(PIPPENGER_MAX_BUCKET_WINDOW-1)+512, 12); + secp256k1_scratch *scratch; + size_t n_points_supported; + int bucket_window = 0; + + for(; scratch_size < max_size; scratch_size+=256) { + size_t i; + size_t total_alloc; + size_t checkpoint; + scratch = secp256k1_scratch_create(&ctx->error_callback, scratch_size); + CHECK(scratch != NULL); + checkpoint = secp256k1_scratch_checkpoint(&ctx->error_callback, scratch); + n_points_supported = secp256k1_pippenger_max_points(&ctx->error_callback, scratch); + if (n_points_supported == 0) { + secp256k1_scratch_destroy(&ctx->error_callback, scratch); + continue; + } + bucket_window = secp256k1_pippenger_bucket_window(n_points_supported); + /* allocate `total_alloc` bytes over `PIPPENGER_SCRATCH_OBJECTS` many allocations */ + total_alloc = secp256k1_pippenger_scratch_size(n_points_supported, bucket_window); + for (i = 0; i < PIPPENGER_SCRATCH_OBJECTS - 1; i++) { + CHECK(secp256k1_scratch_alloc(&ctx->error_callback, scratch, 1)); + total_alloc--; + } + CHECK(secp256k1_scratch_alloc(&ctx->error_callback, scratch, total_alloc)); + secp256k1_scratch_apply_checkpoint(&ctx->error_callback, scratch, checkpoint); + secp256k1_scratch_destroy(&ctx->error_callback, scratch); + } + CHECK(bucket_window == PIPPENGER_MAX_BUCKET_WINDOW); +} + +void test_ecmult_multi_batch_size_helper(void) { + size_t n_batches, n_batch_points, max_n_batch_points, n; + + max_n_batch_points = 0; + n = 1; + CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 0); + + max_n_batch_points = 1; + n = 0; + CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 1); + CHECK(n_batches == 0); + CHECK(n_batch_points == 0); + + max_n_batch_points = 2; + n = 5; + CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 1); + CHECK(n_batches == 3); + CHECK(n_batch_points == 2); + + max_n_batch_points = ECMULT_MAX_POINTS_PER_BATCH; + n = ECMULT_MAX_POINTS_PER_BATCH; + CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 1); + CHECK(n_batches == 1); + CHECK(n_batch_points == ECMULT_MAX_POINTS_PER_BATCH); + + max_n_batch_points = ECMULT_MAX_POINTS_PER_BATCH + 1; + n = ECMULT_MAX_POINTS_PER_BATCH + 1; + CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 1); + CHECK(n_batches == 2); + CHECK(n_batch_points == ECMULT_MAX_POINTS_PER_BATCH/2 + 1); + + max_n_batch_points = 1; + n = SIZE_MAX; + CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 1); + CHECK(n_batches == SIZE_MAX); + CHECK(n_batch_points == 1); + + max_n_batch_points = 2; + n = SIZE_MAX; + CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 1); + CHECK(n_batches == SIZE_MAX/2 + 1); + CHECK(n_batch_points == 2); +} + +/** + * Run secp256k1_ecmult_multi_var with num points and a scratch space restricted to + * 1 <= i <= num points. + */ +void test_ecmult_multi_batching(void) { + static const int n_points = 2*ECMULT_PIPPENGER_THRESHOLD; + secp256k1_scalar scG; + secp256k1_scalar szero; + secp256k1_scalar *sc = (secp256k1_scalar *)checked_malloc(&ctx->error_callback, sizeof(secp256k1_scalar) * n_points); + secp256k1_ge *pt = (secp256k1_ge *)checked_malloc(&ctx->error_callback, sizeof(secp256k1_ge) * n_points); + secp256k1_gej r; + secp256k1_gej r2; + ecmult_multi_data data; + int i; + secp256k1_scratch *scratch; + + secp256k1_gej_set_infinity(&r2); + secp256k1_scalar_set_int(&szero, 0); + + /* Get random scalars and group elements and compute result */ + random_scalar_order(&scG); + secp256k1_ecmult(&ctx->ecmult_ctx, &r2, &r2, &szero, &scG); + for(i = 0; i < n_points; i++) { + secp256k1_ge ptg; + secp256k1_gej ptgj; + random_group_element_test(&ptg); + secp256k1_gej_set_ge(&ptgj, &ptg); + pt[i] = ptg; + random_scalar_order(&sc[i]); + secp256k1_ecmult(&ctx->ecmult_ctx, &ptgj, &ptgj, &sc[i], NULL); + secp256k1_gej_add_var(&r2, &r2, &ptgj, NULL); + } + data.sc = sc; + data.pt = pt; + secp256k1_gej_neg(&r2, &r2); + + /* Test with empty scratch space. It should compute the correct result using + * ecmult_mult_simple algorithm which doesn't require a scratch space. */ + scratch = secp256k1_scratch_create(&ctx->error_callback, 0); + CHECK(secp256k1_ecmult_multi_var(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &scG, ecmult_multi_callback, &data, n_points)); + secp256k1_gej_add_var(&r, &r, &r2, NULL); + CHECK(secp256k1_gej_is_infinity(&r)); + secp256k1_scratch_destroy(&ctx->error_callback, scratch); + + /* Test with space for 1 point in pippenger. That's not enough because + * ecmult_multi selects strauss which requires more memory. It should + * therefore select the simple algorithm. */ + scratch = secp256k1_scratch_create(&ctx->error_callback, secp256k1_pippenger_scratch_size(1, 1) + PIPPENGER_SCRATCH_OBJECTS*ALIGNMENT); + CHECK(secp256k1_ecmult_multi_var(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &scG, ecmult_multi_callback, &data, n_points)); + secp256k1_gej_add_var(&r, &r, &r2, NULL); + CHECK(secp256k1_gej_is_infinity(&r)); + secp256k1_scratch_destroy(&ctx->error_callback, scratch); + + for(i = 1; i <= n_points; i++) { + if (i > ECMULT_PIPPENGER_THRESHOLD) { + int bucket_window = secp256k1_pippenger_bucket_window(i); + size_t scratch_size = secp256k1_pippenger_scratch_size(i, bucket_window); + scratch = secp256k1_scratch_create(&ctx->error_callback, scratch_size + PIPPENGER_SCRATCH_OBJECTS*ALIGNMENT); + } else { + size_t scratch_size = secp256k1_strauss_scratch_size(i); + scratch = secp256k1_scratch_create(&ctx->error_callback, scratch_size + STRAUSS_SCRATCH_OBJECTS*ALIGNMENT); + } + CHECK(secp256k1_ecmult_multi_var(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &r, &scG, ecmult_multi_callback, &data, n_points)); + secp256k1_gej_add_var(&r, &r, &r2, NULL); + CHECK(secp256k1_gej_is_infinity(&r)); + secp256k1_scratch_destroy(&ctx->error_callback, scratch); + } + free(sc); + free(pt); +} + +void run_ecmult_multi_tests(void) { + secp256k1_scratch *scratch; + + test_secp256k1_pippenger_bucket_window_inv(); + test_ecmult_multi_pippenger_max_points(); + scratch = secp256k1_scratch_create(&ctx->error_callback, 819200); + test_ecmult_multi(scratch, secp256k1_ecmult_multi_var); + test_ecmult_multi(NULL, secp256k1_ecmult_multi_var); + test_ecmult_multi(scratch, secp256k1_ecmult_pippenger_batch_single); + test_ecmult_multi_batch_single(secp256k1_ecmult_pippenger_batch_single); + test_ecmult_multi(scratch, secp256k1_ecmult_strauss_batch_single); + test_ecmult_multi_batch_single(secp256k1_ecmult_strauss_batch_single); + secp256k1_scratch_destroy(&ctx->error_callback, scratch); + + /* Run test_ecmult_multi with space for exactly one point */ + scratch = secp256k1_scratch_create(&ctx->error_callback, secp256k1_strauss_scratch_size(1) + STRAUSS_SCRATCH_OBJECTS*ALIGNMENT); + test_ecmult_multi(scratch, secp256k1_ecmult_multi_var); + secp256k1_scratch_destroy(&ctx->error_callback, scratch); + + test_ecmult_multi_batch_size_helper(); + test_ecmult_multi_batching(); +} + +void test_wnaf(const secp256k1_scalar *number, int w) { + secp256k1_scalar x, two, t; + int wnaf[256]; + int zeroes = -1; + int i; + int bits; + secp256k1_scalar_set_int(&x, 0); + secp256k1_scalar_set_int(&two, 2); + bits = secp256k1_ecmult_wnaf(wnaf, 256, number, w); + CHECK(bits <= 256); + for (i = bits-1; i >= 0; i--) { + int v = wnaf[i]; + secp256k1_scalar_mul(&x, &x, &two); + if (v) { + CHECK(zeroes == -1 || zeroes >= w-1); /* check that distance between non-zero elements is at least w-1 */ + zeroes=0; + CHECK((v & 1) == 1); /* check non-zero elements are odd */ + CHECK(v <= (1 << (w-1)) - 1); /* check range below */ + CHECK(v >= -(1 << (w-1)) - 1); /* check range above */ + } else { + CHECK(zeroes != -1); /* check that no unnecessary zero padding exists */ + zeroes++; + } + if (v >= 0) { + secp256k1_scalar_set_int(&t, v); + } else { + secp256k1_scalar_set_int(&t, -v); + secp256k1_scalar_negate(&t, &t); + } + secp256k1_scalar_add(&x, &x, &t); + } + CHECK(secp256k1_scalar_eq(&x, number)); /* check that wnaf represents number */ +} + +void test_constant_wnaf_negate(const secp256k1_scalar *number) { + secp256k1_scalar neg1 = *number; + secp256k1_scalar neg2 = *number; + int sign1 = 1; + int sign2 = 1; + + if (!secp256k1_scalar_get_bits(&neg1, 0, 1)) { + secp256k1_scalar_negate(&neg1, &neg1); + sign1 = -1; + } + sign2 = secp256k1_scalar_cond_negate(&neg2, secp256k1_scalar_is_even(&neg2)); + CHECK(sign1 == sign2); + CHECK(secp256k1_scalar_eq(&neg1, &neg2)); +} + +void test_constant_wnaf(const secp256k1_scalar *number, int w) { + secp256k1_scalar x, shift; + int wnaf[256] = {0}; + int i; + int skew; + int bits = 256; + secp256k1_scalar num = *number; + + secp256k1_scalar_set_int(&x, 0); + secp256k1_scalar_set_int(&shift, 1 << w); + /* With USE_ENDOMORPHISM on we only consider 128-bit numbers */ +#ifdef USE_ENDOMORPHISM + for (i = 0; i < 16; ++i) { + secp256k1_scalar_shr_int(&num, 8); + } + bits = 128; +#endif + skew = secp256k1_wnaf_const(wnaf, &num, w, bits); + + for (i = WNAF_SIZE_BITS(bits, w); i >= 0; --i) { + secp256k1_scalar t; + int v = wnaf[i]; + CHECK(v != 0); /* check nonzero */ + CHECK(v & 1); /* check parity */ + CHECK(v > -(1 << w)); /* check range above */ + CHECK(v < (1 << w)); /* check range below */ + + secp256k1_scalar_mul(&x, &x, &shift); + if (v >= 0) { + secp256k1_scalar_set_int(&t, v); + } else { + secp256k1_scalar_set_int(&t, -v); + secp256k1_scalar_negate(&t, &t); + } + secp256k1_scalar_add(&x, &x, &t); + } + /* Skew num because when encoding numbers as odd we use an offset */ + secp256k1_scalar_cadd_bit(&num, skew == 2, 1); + CHECK(secp256k1_scalar_eq(&x, &num)); +} + +void test_fixed_wnaf(const secp256k1_scalar *number, int w) { + secp256k1_scalar x, shift; + int wnaf[256] = {0}; + int i; + int skew; + secp256k1_scalar num = *number; + + secp256k1_scalar_set_int(&x, 0); + secp256k1_scalar_set_int(&shift, 1 << w); + /* With USE_ENDOMORPHISM on we only consider 128-bit numbers */ +#ifdef USE_ENDOMORPHISM + for (i = 0; i < 16; ++i) { + secp256k1_scalar_shr_int(&num, 8); + } +#endif + skew = secp256k1_wnaf_fixed(wnaf, &num, w); + + for (i = WNAF_SIZE(w)-1; i >= 0; --i) { + secp256k1_scalar t; + int v = wnaf[i]; + CHECK(v == 0 || v & 1); /* check parity */ + CHECK(v > -(1 << w)); /* check range above */ + CHECK(v < (1 << w)); /* check range below */ + + secp256k1_scalar_mul(&x, &x, &shift); + if (v >= 0) { + secp256k1_scalar_set_int(&t, v); + } else { + secp256k1_scalar_set_int(&t, -v); + secp256k1_scalar_negate(&t, &t); + } + secp256k1_scalar_add(&x, &x, &t); + } + /* If skew is 1 then add 1 to num */ + secp256k1_scalar_cadd_bit(&num, 0, skew == 1); + CHECK(secp256k1_scalar_eq(&x, &num)); +} + +/* Checks that the first 8 elements of wnaf are equal to wnaf_expected and the + * rest is 0.*/ +void test_fixed_wnaf_small_helper(int *wnaf, int *wnaf_expected, int w) { + int i; + for (i = WNAF_SIZE(w)-1; i >= 8; --i) { + CHECK(wnaf[i] == 0); + } + for (i = 7; i >= 0; --i) { + CHECK(wnaf[i] == wnaf_expected[i]); + } +} + +void test_fixed_wnaf_small(void) { + int w = 4; + int wnaf[256] = {0}; + int i; + int skew; + secp256k1_scalar num; + + secp256k1_scalar_set_int(&num, 0); + skew = secp256k1_wnaf_fixed(wnaf, &num, w); + for (i = WNAF_SIZE(w)-1; i >= 0; --i) { + int v = wnaf[i]; + CHECK(v == 0); + } + CHECK(skew == 0); + + secp256k1_scalar_set_int(&num, 1); + skew = secp256k1_wnaf_fixed(wnaf, &num, w); + for (i = WNAF_SIZE(w)-1; i >= 1; --i) { + int v = wnaf[i]; + CHECK(v == 0); + } + CHECK(wnaf[0] == 1); + CHECK(skew == 0); + + { + int wnaf_expected[8] = { 0xf, 0xf, 0xf, 0xf, 0xf, 0xf, 0xf, 0xf }; + secp256k1_scalar_set_int(&num, 0xffffffff); + skew = secp256k1_wnaf_fixed(wnaf, &num, w); + test_fixed_wnaf_small_helper(wnaf, wnaf_expected, w); + CHECK(skew == 0); + } + { + int wnaf_expected[8] = { -1, -1, -1, -1, -1, -1, -1, 0xf }; + secp256k1_scalar_set_int(&num, 0xeeeeeeee); + skew = secp256k1_wnaf_fixed(wnaf, &num, w); + test_fixed_wnaf_small_helper(wnaf, wnaf_expected, w); + CHECK(skew == 1); + } + { + int wnaf_expected[8] = { 1, 0, 1, 0, 1, 0, 1, 0 }; + secp256k1_scalar_set_int(&num, 0x01010101); + skew = secp256k1_wnaf_fixed(wnaf, &num, w); + test_fixed_wnaf_small_helper(wnaf, wnaf_expected, w); + CHECK(skew == 0); + } + { + int wnaf_expected[8] = { -0xf, 0, 0xf, -0xf, 0, 0xf, 1, 0 }; + secp256k1_scalar_set_int(&num, 0x01ef1ef1); + skew = secp256k1_wnaf_fixed(wnaf, &num, w); + test_fixed_wnaf_small_helper(wnaf, wnaf_expected, w); + CHECK(skew == 0); + } +} + +void run_wnaf(void) { + int i; + secp256k1_scalar n = {{0}}; + + /* Sanity check: 1 and 2 are the smallest odd and even numbers and should + * have easier-to-diagnose failure modes */ + n.d[0] = 1; + test_constant_wnaf(&n, 4); + n.d[0] = 2; + test_constant_wnaf(&n, 4); + /* Test 0 */ + test_fixed_wnaf_small(); + /* Random tests */ + for (i = 0; i < count; i++) { + random_scalar_order(&n); + test_wnaf(&n, 4+(i%10)); + test_constant_wnaf_negate(&n); + test_constant_wnaf(&n, 4 + (i % 10)); + test_fixed_wnaf(&n, 4 + (i % 10)); + } + secp256k1_scalar_set_int(&n, 0); + CHECK(secp256k1_scalar_cond_negate(&n, 1) == -1); + CHECK(secp256k1_scalar_is_zero(&n)); + CHECK(secp256k1_scalar_cond_negate(&n, 0) == 1); + CHECK(secp256k1_scalar_is_zero(&n)); +} + +void test_ecmult_constants(void) { + /* Test ecmult_gen() for [0..36) and [order-36..0). */ + secp256k1_scalar x; + secp256k1_gej r; + secp256k1_ge ng; + int i; + int j; + secp256k1_ge_neg(&ng, &secp256k1_ge_const_g); + for (i = 0; i < 36; i++ ) { + secp256k1_scalar_set_int(&x, i); + secp256k1_ecmult_gen(&ctx->ecmult_gen_ctx, &r, &x); + for (j = 0; j < i; j++) { + if (j == i - 1) { + ge_equals_gej(&secp256k1_ge_const_g, &r); + } + secp256k1_gej_add_ge(&r, &r, &ng); + } + CHECK(secp256k1_gej_is_infinity(&r)); + } + for (i = 1; i <= 36; i++ ) { + secp256k1_scalar_set_int(&x, i); + secp256k1_scalar_negate(&x, &x); + secp256k1_ecmult_gen(&ctx->ecmult_gen_ctx, &r, &x); + for (j = 0; j < i; j++) { + if (j == i - 1) { + ge_equals_gej(&ng, &r); + } + secp256k1_gej_add_ge(&r, &r, &secp256k1_ge_const_g); + } + CHECK(secp256k1_gej_is_infinity(&r)); + } +} + +void run_ecmult_constants(void) { + test_ecmult_constants(); +} + +void test_ecmult_gen_blind(void) { + /* Test ecmult_gen() blinding and confirm that the blinding changes, the affine points match, and the z's don't match. */ + secp256k1_scalar key; + secp256k1_scalar b; + unsigned char seed32[32]; + secp256k1_gej pgej; + secp256k1_gej pgej2; + secp256k1_gej i; + secp256k1_ge pge; + random_scalar_order_test(&key); + secp256k1_ecmult_gen(&ctx->ecmult_gen_ctx, &pgej, &key); + secp256k1_rand256(seed32); + b = ctx->ecmult_gen_ctx.blind; + i = ctx->ecmult_gen_ctx.initial; + secp256k1_ecmult_gen_blind(&ctx->ecmult_gen_ctx, seed32); + CHECK(!secp256k1_scalar_eq(&b, &ctx->ecmult_gen_ctx.blind)); + secp256k1_ecmult_gen(&ctx->ecmult_gen_ctx, &pgej2, &key); + CHECK(!gej_xyz_equals_gej(&pgej, &pgej2)); + CHECK(!gej_xyz_equals_gej(&i, &ctx->ecmult_gen_ctx.initial)); + secp256k1_ge_set_gej(&pge, &pgej); + ge_equals_gej(&pge, &pgej2); +} + +void test_ecmult_gen_blind_reset(void) { + /* Test ecmult_gen() blinding reset and confirm that the blinding is consistent. */ + secp256k1_scalar b; + secp256k1_gej initial; + secp256k1_ecmult_gen_blind(&ctx->ecmult_gen_ctx, 0); + b = ctx->ecmult_gen_ctx.blind; + initial = ctx->ecmult_gen_ctx.initial; + secp256k1_ecmult_gen_blind(&ctx->ecmult_gen_ctx, 0); + CHECK(secp256k1_scalar_eq(&b, &ctx->ecmult_gen_ctx.blind)); + CHECK(gej_xyz_equals_gej(&initial, &ctx->ecmult_gen_ctx.initial)); +} + +void run_ecmult_gen_blind(void) { + int i; + test_ecmult_gen_blind_reset(); + for (i = 0; i < 10; i++) { + test_ecmult_gen_blind(); + } +} + +#ifdef USE_ENDOMORPHISM +/***** ENDOMORPHISH TESTS *****/ +void test_scalar_split(void) { + secp256k1_scalar full; + secp256k1_scalar s1, slam; + const unsigned char zero[32] = {0}; + unsigned char tmp[32]; + + random_scalar_order_test(&full); + secp256k1_scalar_split_lambda(&s1, &slam, &full); + + /* check that both are <= 128 bits in size */ + if (secp256k1_scalar_is_high(&s1)) { + secp256k1_scalar_negate(&s1, &s1); + } + if (secp256k1_scalar_is_high(&slam)) { + secp256k1_scalar_negate(&slam, &slam); + } + + secp256k1_scalar_get_b32(tmp, &s1); + CHECK(memcmp(zero, tmp, 16) == 0); + secp256k1_scalar_get_b32(tmp, &slam); + CHECK(memcmp(zero, tmp, 16) == 0); +} + +void run_endomorphism_tests(void) { + test_scalar_split(); +} +#endif + +void ec_pubkey_parse_pointtest(const unsigned char *input, int xvalid, int yvalid) { + unsigned char pubkeyc[65]; + secp256k1_pubkey pubkey; + secp256k1_ge ge; + size_t pubkeyclen; + int32_t ecount; + ecount = 0; + secp256k1_context_set_illegal_callback(ctx, counting_illegal_callback_fn, &ecount); + for (pubkeyclen = 3; pubkeyclen <= 65; pubkeyclen++) { + /* Smaller sizes are tested exhaustively elsewhere. */ + int32_t i; + memcpy(&pubkeyc[1], input, 64); + VG_UNDEF(&pubkeyc[pubkeyclen], 65 - pubkeyclen); + for (i = 0; i < 256; i++) { + /* Try all type bytes. */ + int xpass; + int ypass; + int ysign; + pubkeyc[0] = i; + /* What sign does this point have? */ + ysign = (input[63] & 1) + 2; + /* For the current type (i) do we expect parsing to work? Handled all of compressed/uncompressed/hybrid. */ + xpass = xvalid && (pubkeyclen == 33) && ((i & 254) == 2); + /* Do we expect a parse and re-serialize as uncompressed to give a matching y? */ + ypass = xvalid && yvalid && ((i & 4) == ((pubkeyclen == 65) << 2)) && + ((i == 4) || ((i & 251) == ysign)) && ((pubkeyclen == 33) || (pubkeyclen == 65)); + if (xpass || ypass) { + /* These cases must parse. */ + unsigned char pubkeyo[65]; + size_t outl; + memset(&pubkey, 0, sizeof(pubkey)); + VG_UNDEF(&pubkey, sizeof(pubkey)); + ecount = 0; + CHECK(secp256k1_ec_pubkey_parse(ctx, &pubkey, pubkeyc, pubkeyclen) == 1); + VG_CHECK(&pubkey, sizeof(pubkey)); + outl = 65; + VG_UNDEF(pubkeyo, 65); + CHECK(secp256k1_ec_pubkey_serialize(ctx, pubkeyo, &outl, &pubkey, SECP256K1_EC_COMPRESSED) == 1); + VG_CHECK(pubkeyo, outl); + CHECK(outl == 33); + CHECK(memcmp(&pubkeyo[1], &pubkeyc[1], 32) == 0); + CHECK((pubkeyclen != 33) || (pubkeyo[0] == pubkeyc[0])); + if (ypass) { + /* This test isn't always done because we decode with alternative signs, so the y won't match. */ + CHECK(pubkeyo[0] == ysign); + CHECK(secp256k1_pubkey_load(ctx, &ge, &pubkey) == 1); + memset(&pubkey, 0, sizeof(pubkey)); + VG_UNDEF(&pubkey, sizeof(pubkey)); + secp256k1_pubkey_save(&pubkey, &ge); + VG_CHECK(&pubkey, sizeof(pubkey)); + outl = 65; + VG_UNDEF(pubkeyo, 65); + CHECK(secp256k1_ec_pubkey_serialize(ctx, pubkeyo, &outl, &pubkey, SECP256K1_EC_UNCOMPRESSED) == 1); + VG_CHECK(pubkeyo, outl); + CHECK(outl == 65); + CHECK(pubkeyo[0] == 4); + CHECK(memcmp(&pubkeyo[1], input, 64) == 0); + } + CHECK(ecount == 0); + } else { + /* These cases must fail to parse. */ + memset(&pubkey, 0xfe, sizeof(pubkey)); + ecount = 0; + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_parse(ctx, &pubkey, pubkeyc, pubkeyclen) == 0); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(ecount == 0); + CHECK(secp256k1_pubkey_load(ctx, &ge, &pubkey) == 0); + CHECK(ecount == 1); + } + } + } + secp256k1_context_set_illegal_callback(ctx, NULL, NULL); +} + +void run_ec_pubkey_parse_test(void) { +#define SECP256K1_EC_PARSE_TEST_NVALID (12) + const unsigned char valid[SECP256K1_EC_PARSE_TEST_NVALID][64] = { + { + /* Point with leading and trailing zeros in x and y serialization. */ + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x42, 0x52, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x64, 0xef, 0xa1, 0x7b, 0x77, 0x61, 0xe1, 0xe4, 0x27, 0x06, 0x98, 0x9f, 0xb4, 0x83, + 0xb8, 0xd2, 0xd4, 0x9b, 0xf7, 0x8f, 0xae, 0x98, 0x03, 0xf0, 0x99, 0xb8, 0x34, 0xed, 0xeb, 0x00 + }, + { + /* Point with x equal to a 3rd root of unity.*/ + 0x7a, 0xe9, 0x6a, 0x2b, 0x65, 0x7c, 0x07, 0x10, 0x6e, 0x64, 0x47, 0x9e, 0xac, 0x34, 0x34, 0xe9, + 0x9c, 0xf0, 0x49, 0x75, 0x12, 0xf5, 0x89, 0x95, 0xc1, 0x39, 0x6c, 0x28, 0x71, 0x95, 0x01, 0xee, + 0x42, 0x18, 0xf2, 0x0a, 0xe6, 0xc6, 0x46, 0xb3, 0x63, 0xdb, 0x68, 0x60, 0x58, 0x22, 0xfb, 0x14, + 0x26, 0x4c, 0xa8, 0xd2, 0x58, 0x7f, 0xdd, 0x6f, 0xbc, 0x75, 0x0d, 0x58, 0x7e, 0x76, 0xa7, 0xee, + }, + { + /* Point with largest x. (1/2) */ + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xff, 0xff, 0xfc, 0x2c, + 0x0e, 0x99, 0x4b, 0x14, 0xea, 0x72, 0xf8, 0xc3, 0xeb, 0x95, 0xc7, 0x1e, 0xf6, 0x92, 0x57, 0x5e, + 0x77, 0x50, 0x58, 0x33, 0x2d, 0x7e, 0x52, 0xd0, 0x99, 0x5c, 0xf8, 0x03, 0x88, 0x71, 0xb6, 0x7d, + }, + { + /* Point with largest x. (2/2) */ + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xff, 0xff, 0xfc, 0x2c, + 0xf1, 0x66, 0xb4, 0xeb, 0x15, 0x8d, 0x07, 0x3c, 0x14, 0x6a, 0x38, 0xe1, 0x09, 0x6d, 0xa8, 0xa1, + 0x88, 0xaf, 0xa7, 0xcc, 0xd2, 0x81, 0xad, 0x2f, 0x66, 0xa3, 0x07, 0xfb, 0x77, 0x8e, 0x45, 0xb2, + }, + { + /* Point with smallest x. (1/2) */ + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + 0x42, 0x18, 0xf2, 0x0a, 0xe6, 0xc6, 0x46, 0xb3, 0x63, 0xdb, 0x68, 0x60, 0x58, 0x22, 0xfb, 0x14, + 0x26, 0x4c, 0xa8, 0xd2, 0x58, 0x7f, 0xdd, 0x6f, 0xbc, 0x75, 0x0d, 0x58, 0x7e, 0x76, 0xa7, 0xee, + }, + { + /* Point with smallest x. (2/2) */ + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + 0xbd, 0xe7, 0x0d, 0xf5, 0x19, 0x39, 0xb9, 0x4c, 0x9c, 0x24, 0x97, 0x9f, 0xa7, 0xdd, 0x04, 0xeb, + 0xd9, 0xb3, 0x57, 0x2d, 0xa7, 0x80, 0x22, 0x90, 0x43, 0x8a, 0xf2, 0xa6, 0x81, 0x89, 0x54, 0x41, + }, + { + /* Point with largest y. (1/3) */ + 0x1f, 0xe1, 0xe5, 0xef, 0x3f, 0xce, 0xb5, 0xc1, 0x35, 0xab, 0x77, 0x41, 0x33, 0x3c, 0xe5, 0xa6, + 0xe8, 0x0d, 0x68, 0x16, 0x76, 0x53, 0xf6, 0xb2, 0xb2, 0x4b, 0xcb, 0xcf, 0xaa, 0xaf, 0xf5, 0x07, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xff, 0xff, 0xfc, 0x2e, + }, + { + /* Point with largest y. (2/3) */ + 0xcb, 0xb0, 0xde, 0xab, 0x12, 0x57, 0x54, 0xf1, 0xfd, 0xb2, 0x03, 0x8b, 0x04, 0x34, 0xed, 0x9c, + 0xb3, 0xfb, 0x53, 0xab, 0x73, 0x53, 0x91, 0x12, 0x99, 0x94, 0xa5, 0x35, 0xd9, 0x25, 0xf6, 0x73, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xff, 0xff, 0xfc, 0x2e, + }, + { + /* Point with largest y. (3/3) */ + 0x14, 0x6d, 0x3b, 0x65, 0xad, 0xd9, 0xf5, 0x4c, 0xcc, 0xa2, 0x85, 0x33, 0xc8, 0x8e, 0x2c, 0xbc, + 0x63, 0xf7, 0x44, 0x3e, 0x16, 0x58, 0x78, 0x3a, 0xb4, 0x1f, 0x8e, 0xf9, 0x7c, 0x2a, 0x10, 0xb5, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xff, 0xff, 0xfc, 0x2e, + }, + { + /* Point with smallest y. (1/3) */ + 0x1f, 0xe1, 0xe5, 0xef, 0x3f, 0xce, 0xb5, 0xc1, 0x35, 0xab, 0x77, 0x41, 0x33, 0x3c, 0xe5, 0xa6, + 0xe8, 0x0d, 0x68, 0x16, 0x76, 0x53, 0xf6, 0xb2, 0xb2, 0x4b, 0xcb, 0xcf, 0xaa, 0xaf, 0xf5, 0x07, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + }, + { + /* Point with smallest y. (2/3) */ + 0xcb, 0xb0, 0xde, 0xab, 0x12, 0x57, 0x54, 0xf1, 0xfd, 0xb2, 0x03, 0x8b, 0x04, 0x34, 0xed, 0x9c, + 0xb3, 0xfb, 0x53, 0xab, 0x73, 0x53, 0x91, 0x12, 0x99, 0x94, 0xa5, 0x35, 0xd9, 0x25, 0xf6, 0x73, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + }, + { + /* Point with smallest y. (3/3) */ + 0x14, 0x6d, 0x3b, 0x65, 0xad, 0xd9, 0xf5, 0x4c, 0xcc, 0xa2, 0x85, 0x33, 0xc8, 0x8e, 0x2c, 0xbc, + 0x63, 0xf7, 0x44, 0x3e, 0x16, 0x58, 0x78, 0x3a, 0xb4, 0x1f, 0x8e, 0xf9, 0x7c, 0x2a, 0x10, 0xb5, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01 + } + }; +#define SECP256K1_EC_PARSE_TEST_NXVALID (4) + const unsigned char onlyxvalid[SECP256K1_EC_PARSE_TEST_NXVALID][64] = { + { + /* Valid if y overflow ignored (y = 1 mod p). (1/3) */ + 0x1f, 0xe1, 0xe5, 0xef, 0x3f, 0xce, 0xb5, 0xc1, 0x35, 0xab, 0x77, 0x41, 0x33, 0x3c, 0xe5, 0xa6, + 0xe8, 0x0d, 0x68, 0x16, 0x76, 0x53, 0xf6, 0xb2, 0xb2, 0x4b, 0xcb, 0xcf, 0xaa, 0xaf, 0xf5, 0x07, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xff, 0xff, 0xfc, 0x30, + }, + { + /* Valid if y overflow ignored (y = 1 mod p). (2/3) */ + 0xcb, 0xb0, 0xde, 0xab, 0x12, 0x57, 0x54, 0xf1, 0xfd, 0xb2, 0x03, 0x8b, 0x04, 0x34, 0xed, 0x9c, + 0xb3, 0xfb, 0x53, 0xab, 0x73, 0x53, 0x91, 0x12, 0x99, 0x94, 0xa5, 0x35, 0xd9, 0x25, 0xf6, 0x73, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xff, 0xff, 0xfc, 0x30, + }, + { + /* Valid if y overflow ignored (y = 1 mod p). (3/3)*/ + 0x14, 0x6d, 0x3b, 0x65, 0xad, 0xd9, 0xf5, 0x4c, 0xcc, 0xa2, 0x85, 0x33, 0xc8, 0x8e, 0x2c, 0xbc, + 0x63, 0xf7, 0x44, 0x3e, 0x16, 0x58, 0x78, 0x3a, 0xb4, 0x1f, 0x8e, 0xf9, 0x7c, 0x2a, 0x10, 0xb5, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xff, 0xff, 0xfc, 0x30, + }, + { + /* x on curve, y is from y^2 = x^3 + 8. */ + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03 + } + }; +#define SECP256K1_EC_PARSE_TEST_NINVALID (7) + const unsigned char invalid[SECP256K1_EC_PARSE_TEST_NINVALID][64] = { + { + /* x is third root of -8, y is -1 * (x^3+7); also on the curve for y^2 = x^3 + 9. */ + 0x0a, 0x2d, 0x2b, 0xa9, 0x35, 0x07, 0xf1, 0xdf, 0x23, 0x37, 0x70, 0xc2, 0xa7, 0x97, 0x96, 0x2c, + 0xc6, 0x1f, 0x6d, 0x15, 0xda, 0x14, 0xec, 0xd4, 0x7d, 0x8d, 0x27, 0xae, 0x1c, 0xd5, 0xf8, 0x53, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + }, + { + /* Valid if x overflow ignored (x = 1 mod p). */ + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xff, 0xff, 0xfc, 0x30, + 0x42, 0x18, 0xf2, 0x0a, 0xe6, 0xc6, 0x46, 0xb3, 0x63, 0xdb, 0x68, 0x60, 0x58, 0x22, 0xfb, 0x14, + 0x26, 0x4c, 0xa8, 0xd2, 0x58, 0x7f, 0xdd, 0x6f, 0xbc, 0x75, 0x0d, 0x58, 0x7e, 0x76, 0xa7, 0xee, + }, + { + /* Valid if x overflow ignored (x = 1 mod p). */ + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xff, 0xff, 0xfc, 0x30, + 0xbd, 0xe7, 0x0d, 0xf5, 0x19, 0x39, 0xb9, 0x4c, 0x9c, 0x24, 0x97, 0x9f, 0xa7, 0xdd, 0x04, 0xeb, + 0xd9, 0xb3, 0x57, 0x2d, 0xa7, 0x80, 0x22, 0x90, 0x43, 0x8a, 0xf2, 0xa6, 0x81, 0x89, 0x54, 0x41, + }, + { + /* x is -1, y is the result of the sqrt ladder; also on the curve for y^2 = x^3 - 5. */ + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xff, 0xff, 0xfc, 0x2e, + 0xf4, 0x84, 0x14, 0x5c, 0xb0, 0x14, 0x9b, 0x82, 0x5d, 0xff, 0x41, 0x2f, 0xa0, 0x52, 0xa8, 0x3f, + 0xcb, 0x72, 0xdb, 0x61, 0xd5, 0x6f, 0x37, 0x70, 0xce, 0x06, 0x6b, 0x73, 0x49, 0xa2, 0xaa, 0x28, + }, + { + /* x is -1, y is the result of the sqrt ladder; also on the curve for y^2 = x^3 - 5. */ + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xff, 0xff, 0xfc, 0x2e, + 0x0b, 0x7b, 0xeb, 0xa3, 0x4f, 0xeb, 0x64, 0x7d, 0xa2, 0x00, 0xbe, 0xd0, 0x5f, 0xad, 0x57, 0xc0, + 0x34, 0x8d, 0x24, 0x9e, 0x2a, 0x90, 0xc8, 0x8f, 0x31, 0xf9, 0x94, 0x8b, 0xb6, 0x5d, 0x52, 0x07, + }, + { + /* x is zero, y is the result of the sqrt ladder; also on the curve for y^2 = x^3 - 7. */ + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x8f, 0x53, 0x7e, 0xef, 0xdf, 0xc1, 0x60, 0x6a, 0x07, 0x27, 0xcd, 0x69, 0xb4, 0xa7, 0x33, 0x3d, + 0x38, 0xed, 0x44, 0xe3, 0x93, 0x2a, 0x71, 0x79, 0xee, 0xcb, 0x4b, 0x6f, 0xba, 0x93, 0x60, 0xdc, + }, + { + /* x is zero, y is the result of the sqrt ladder; also on the curve for y^2 = x^3 - 7. */ + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x70, 0xac, 0x81, 0x10, 0x20, 0x3e, 0x9f, 0x95, 0xf8, 0xd8, 0x32, 0x96, 0x4b, 0x58, 0xcc, 0xc2, + 0xc7, 0x12, 0xbb, 0x1c, 0x6c, 0xd5, 0x8e, 0x86, 0x11, 0x34, 0xb4, 0x8f, 0x45, 0x6c, 0x9b, 0x53 + } + }; + const unsigned char pubkeyc[66] = { + /* Serialization of G. */ + 0x04, 0x79, 0xBE, 0x66, 0x7E, 0xF9, 0xDC, 0xBB, 0xAC, 0x55, 0xA0, 0x62, 0x95, 0xCE, 0x87, 0x0B, + 0x07, 0x02, 0x9B, 0xFC, 0xDB, 0x2D, 0xCE, 0x28, 0xD9, 0x59, 0xF2, 0x81, 0x5B, 0x16, 0xF8, 0x17, + 0x98, 0x48, 0x3A, 0xDA, 0x77, 0x26, 0xA3, 0xC4, 0x65, 0x5D, 0xA4, 0xFB, 0xFC, 0x0E, 0x11, 0x08, + 0xA8, 0xFD, 0x17, 0xB4, 0x48, 0xA6, 0x85, 0x54, 0x19, 0x9C, 0x47, 0xD0, 0x8F, 0xFB, 0x10, 0xD4, + 0xB8, 0x00 + }; + unsigned char sout[65]; + unsigned char shortkey[2]; + secp256k1_ge ge; + secp256k1_pubkey pubkey; + size_t len; + int32_t i; + int32_t ecount; + int32_t ecount2; + ecount = 0; + /* Nothing should be reading this far into pubkeyc. */ + VG_UNDEF(&pubkeyc[65], 1); + secp256k1_context_set_illegal_callback(ctx, counting_illegal_callback_fn, &ecount); + /* Zero length claimed, fail, zeroize, no illegal arg error. */ + memset(&pubkey, 0xfe, sizeof(pubkey)); + ecount = 0; + VG_UNDEF(shortkey, 2); + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_parse(ctx, &pubkey, shortkey, 0) == 0); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(ecount == 0); + CHECK(secp256k1_pubkey_load(ctx, &ge, &pubkey) == 0); + CHECK(ecount == 1); + /* Length one claimed, fail, zeroize, no illegal arg error. */ + for (i = 0; i < 256 ; i++) { + memset(&pubkey, 0xfe, sizeof(pubkey)); + ecount = 0; + shortkey[0] = i; + VG_UNDEF(&shortkey[1], 1); + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_parse(ctx, &pubkey, shortkey, 1) == 0); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(ecount == 0); + CHECK(secp256k1_pubkey_load(ctx, &ge, &pubkey) == 0); + CHECK(ecount == 1); + } + /* Length two claimed, fail, zeroize, no illegal arg error. */ + for (i = 0; i < 65536 ; i++) { + memset(&pubkey, 0xfe, sizeof(pubkey)); + ecount = 0; + shortkey[0] = i & 255; + shortkey[1] = i >> 8; + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_parse(ctx, &pubkey, shortkey, 2) == 0); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(ecount == 0); + CHECK(secp256k1_pubkey_load(ctx, &ge, &pubkey) == 0); + CHECK(ecount == 1); + } + memset(&pubkey, 0xfe, sizeof(pubkey)); + ecount = 0; + VG_UNDEF(&pubkey, sizeof(pubkey)); + /* 33 bytes claimed on otherwise valid input starting with 0x04, fail, zeroize output, no illegal arg error. */ + CHECK(secp256k1_ec_pubkey_parse(ctx, &pubkey, pubkeyc, 33) == 0); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(ecount == 0); + CHECK(secp256k1_pubkey_load(ctx, &ge, &pubkey) == 0); + CHECK(ecount == 1); + /* NULL pubkey, illegal arg error. Pubkey isn't rewritten before this step, since it's NULL into the parser. */ + CHECK(secp256k1_ec_pubkey_parse(ctx, NULL, pubkeyc, 65) == 0); + CHECK(ecount == 2); + /* NULL input string. Illegal arg and zeroize output. */ + memset(&pubkey, 0xfe, sizeof(pubkey)); + ecount = 0; + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_parse(ctx, &pubkey, NULL, 65) == 0); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(ecount == 1); + CHECK(secp256k1_pubkey_load(ctx, &ge, &pubkey) == 0); + CHECK(ecount == 2); + /* 64 bytes claimed on input starting with 0x04, fail, zeroize output, no illegal arg error. */ + memset(&pubkey, 0xfe, sizeof(pubkey)); + ecount = 0; + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_parse(ctx, &pubkey, pubkeyc, 64) == 0); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(ecount == 0); + CHECK(secp256k1_pubkey_load(ctx, &ge, &pubkey) == 0); + CHECK(ecount == 1); + /* 66 bytes claimed, fail, zeroize output, no illegal arg error. */ + memset(&pubkey, 0xfe, sizeof(pubkey)); + ecount = 0; + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_parse(ctx, &pubkey, pubkeyc, 66) == 0); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(ecount == 0); + CHECK(secp256k1_pubkey_load(ctx, &ge, &pubkey) == 0); + CHECK(ecount == 1); + /* Valid parse. */ + memset(&pubkey, 0, sizeof(pubkey)); + ecount = 0; + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_parse(ctx, &pubkey, pubkeyc, 65) == 1); + CHECK(secp256k1_ec_pubkey_parse(secp256k1_context_no_precomp, &pubkey, pubkeyc, 65) == 1); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(ecount == 0); + VG_UNDEF(&ge, sizeof(ge)); + CHECK(secp256k1_pubkey_load(ctx, &ge, &pubkey) == 1); + VG_CHECK(&ge.x, sizeof(ge.x)); + VG_CHECK(&ge.y, sizeof(ge.y)); + VG_CHECK(&ge.infinity, sizeof(ge.infinity)); + ge_equals_ge(&secp256k1_ge_const_g, &ge); + CHECK(ecount == 0); + /* secp256k1_ec_pubkey_serialize illegal args. */ + ecount = 0; + len = 65; + CHECK(secp256k1_ec_pubkey_serialize(ctx, NULL, &len, &pubkey, SECP256K1_EC_UNCOMPRESSED) == 0); + CHECK(ecount == 1); + CHECK(len == 0); + CHECK(secp256k1_ec_pubkey_serialize(ctx, sout, NULL, &pubkey, SECP256K1_EC_UNCOMPRESSED) == 0); + CHECK(ecount == 2); + len = 65; + VG_UNDEF(sout, 65); + CHECK(secp256k1_ec_pubkey_serialize(ctx, sout, &len, NULL, SECP256K1_EC_UNCOMPRESSED) == 0); + VG_CHECK(sout, 65); + CHECK(ecount == 3); + CHECK(len == 0); + len = 65; + CHECK(secp256k1_ec_pubkey_serialize(ctx, sout, &len, &pubkey, ~0) == 0); + CHECK(ecount == 4); + CHECK(len == 0); + len = 65; + VG_UNDEF(sout, 65); + CHECK(secp256k1_ec_pubkey_serialize(ctx, sout, &len, &pubkey, SECP256K1_EC_UNCOMPRESSED) == 1); + VG_CHECK(sout, 65); + CHECK(ecount == 4); + CHECK(len == 65); + /* Multiple illegal args. Should still set arg error only once. */ + ecount = 0; + ecount2 = 11; + CHECK(secp256k1_ec_pubkey_parse(ctx, NULL, NULL, 65) == 0); + CHECK(ecount == 1); + /* Does the illegal arg callback actually change the behavior? */ + secp256k1_context_set_illegal_callback(ctx, uncounting_illegal_callback_fn, &ecount2); + CHECK(secp256k1_ec_pubkey_parse(ctx, NULL, NULL, 65) == 0); + CHECK(ecount == 1); + CHECK(ecount2 == 10); + secp256k1_context_set_illegal_callback(ctx, NULL, NULL); + /* Try a bunch of prefabbed points with all possible encodings. */ + for (i = 0; i < SECP256K1_EC_PARSE_TEST_NVALID; i++) { + ec_pubkey_parse_pointtest(valid[i], 1, 1); + } + for (i = 0; i < SECP256K1_EC_PARSE_TEST_NXVALID; i++) { + ec_pubkey_parse_pointtest(onlyxvalid[i], 1, 0); + } + for (i = 0; i < SECP256K1_EC_PARSE_TEST_NINVALID; i++) { + ec_pubkey_parse_pointtest(invalid[i], 0, 0); + } +} + +void run_eckey_edge_case_test(void) { + const unsigned char orderc[32] = { + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, + 0xba, 0xae, 0xdc, 0xe6, 0xaf, 0x48, 0xa0, 0x3b, + 0xbf, 0xd2, 0x5e, 0x8c, 0xd0, 0x36, 0x41, 0x41 + }; + const unsigned char zeros[sizeof(secp256k1_pubkey)] = {0x00}; + unsigned char ctmp[33]; + unsigned char ctmp2[33]; + secp256k1_pubkey pubkey; + secp256k1_pubkey pubkey2; + secp256k1_pubkey pubkey_one; + secp256k1_pubkey pubkey_negone; + const secp256k1_pubkey *pubkeys[3]; + size_t len; + int32_t ecount; + /* Group order is too large, reject. */ + CHECK(secp256k1_ec_seckey_verify(ctx, orderc) == 0); + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey, orderc) == 0); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) == 0); + /* Maximum value is too large, reject. */ + memset(ctmp, 255, 32); + CHECK(secp256k1_ec_seckey_verify(ctx, ctmp) == 0); + memset(&pubkey, 1, sizeof(pubkey)); + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey, ctmp) == 0); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) == 0); + /* Zero is too small, reject. */ + memset(ctmp, 0, 32); + CHECK(secp256k1_ec_seckey_verify(ctx, ctmp) == 0); + memset(&pubkey, 1, sizeof(pubkey)); + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey, ctmp) == 0); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) == 0); + /* One must be accepted. */ + ctmp[31] = 0x01; + CHECK(secp256k1_ec_seckey_verify(ctx, ctmp) == 1); + memset(&pubkey, 0, sizeof(pubkey)); + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey, ctmp) == 1); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) > 0); + pubkey_one = pubkey; + /* Group order + 1 is too large, reject. */ + memcpy(ctmp, orderc, 32); + ctmp[31] = 0x42; + CHECK(secp256k1_ec_seckey_verify(ctx, ctmp) == 0); + memset(&pubkey, 1, sizeof(pubkey)); + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey, ctmp) == 0); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) == 0); + /* -1 must be accepted. */ + ctmp[31] = 0x40; + CHECK(secp256k1_ec_seckey_verify(ctx, ctmp) == 1); + memset(&pubkey, 0, sizeof(pubkey)); + VG_UNDEF(&pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey, ctmp) == 1); + VG_CHECK(&pubkey, sizeof(pubkey)); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) > 0); + pubkey_negone = pubkey; + /* Tweak of zero leaves the value unchanged. */ + memset(ctmp2, 0, 32); + CHECK(secp256k1_ec_privkey_tweak_add(ctx, ctmp, ctmp2) == 1); + CHECK(memcmp(orderc, ctmp, 31) == 0 && ctmp[31] == 0x40); + memcpy(&pubkey2, &pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_tweak_add(ctx, &pubkey, ctmp2) == 1); + CHECK(memcmp(&pubkey, &pubkey2, sizeof(pubkey)) == 0); + /* Multiply tweak of zero zeroizes the output. */ + CHECK(secp256k1_ec_privkey_tweak_mul(ctx, ctmp, ctmp2) == 0); + CHECK(memcmp(zeros, ctmp, 32) == 0); + CHECK(secp256k1_ec_pubkey_tweak_mul(ctx, &pubkey, ctmp2) == 0); + CHECK(memcmp(&pubkey, zeros, sizeof(pubkey)) == 0); + memcpy(&pubkey, &pubkey2, sizeof(pubkey)); + /* Overflowing key tweak zeroizes. */ + memcpy(ctmp, orderc, 32); + ctmp[31] = 0x40; + CHECK(secp256k1_ec_privkey_tweak_add(ctx, ctmp, orderc) == 0); + CHECK(memcmp(zeros, ctmp, 32) == 0); + memcpy(ctmp, orderc, 32); + ctmp[31] = 0x40; + CHECK(secp256k1_ec_privkey_tweak_mul(ctx, ctmp, orderc) == 0); + CHECK(memcmp(zeros, ctmp, 32) == 0); + memcpy(ctmp, orderc, 32); + ctmp[31] = 0x40; + CHECK(secp256k1_ec_pubkey_tweak_add(ctx, &pubkey, orderc) == 0); + CHECK(memcmp(&pubkey, zeros, sizeof(pubkey)) == 0); + memcpy(&pubkey, &pubkey2, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_tweak_mul(ctx, &pubkey, orderc) == 0); + CHECK(memcmp(&pubkey, zeros, sizeof(pubkey)) == 0); + memcpy(&pubkey, &pubkey2, sizeof(pubkey)); + /* Private key tweaks results in a key of zero. */ + ctmp2[31] = 1; + CHECK(secp256k1_ec_privkey_tweak_add(ctx, ctmp2, ctmp) == 0); + CHECK(memcmp(zeros, ctmp2, 32) == 0); + ctmp2[31] = 1; + CHECK(secp256k1_ec_pubkey_tweak_add(ctx, &pubkey, ctmp2) == 0); + CHECK(memcmp(&pubkey, zeros, sizeof(pubkey)) == 0); + memcpy(&pubkey, &pubkey2, sizeof(pubkey)); + /* Tweak computation wraps and results in a key of 1. */ + ctmp2[31] = 2; + CHECK(secp256k1_ec_privkey_tweak_add(ctx, ctmp2, ctmp) == 1); + CHECK(memcmp(ctmp2, zeros, 31) == 0 && ctmp2[31] == 1); + ctmp2[31] = 2; + CHECK(secp256k1_ec_pubkey_tweak_add(ctx, &pubkey, ctmp2) == 1); + ctmp2[31] = 1; + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey2, ctmp2) == 1); + CHECK(memcmp(&pubkey, &pubkey2, sizeof(pubkey)) == 0); + /* Tweak mul * 2 = 1+1. */ + CHECK(secp256k1_ec_pubkey_tweak_add(ctx, &pubkey, ctmp2) == 1); + ctmp2[31] = 2; + CHECK(secp256k1_ec_pubkey_tweak_mul(ctx, &pubkey2, ctmp2) == 1); + CHECK(memcmp(&pubkey, &pubkey2, sizeof(pubkey)) == 0); + /* Test argument errors. */ + ecount = 0; + secp256k1_context_set_illegal_callback(ctx, counting_illegal_callback_fn, &ecount); + CHECK(ecount == 0); + /* Zeroize pubkey on parse error. */ + memset(&pubkey, 0, 32); + CHECK(secp256k1_ec_pubkey_tweak_add(ctx, &pubkey, ctmp2) == 0); + CHECK(ecount == 1); + CHECK(memcmp(&pubkey, zeros, sizeof(pubkey)) == 0); + memcpy(&pubkey, &pubkey2, sizeof(pubkey)); + memset(&pubkey2, 0, 32); + CHECK(secp256k1_ec_pubkey_tweak_mul(ctx, &pubkey2, ctmp2) == 0); + CHECK(ecount == 2); + CHECK(memcmp(&pubkey2, zeros, sizeof(pubkey2)) == 0); + /* Plain argument errors. */ + ecount = 0; + CHECK(secp256k1_ec_seckey_verify(ctx, ctmp) == 1); + CHECK(ecount == 0); + CHECK(secp256k1_ec_seckey_verify(ctx, NULL) == 0); + CHECK(ecount == 1); + ecount = 0; + memset(ctmp2, 0, 32); + ctmp2[31] = 4; + CHECK(secp256k1_ec_pubkey_tweak_add(ctx, NULL, ctmp2) == 0); + CHECK(ecount == 1); + CHECK(secp256k1_ec_pubkey_tweak_add(ctx, &pubkey, NULL) == 0); + CHECK(ecount == 2); + ecount = 0; + memset(ctmp2, 0, 32); + ctmp2[31] = 4; + CHECK(secp256k1_ec_pubkey_tweak_mul(ctx, NULL, ctmp2) == 0); + CHECK(ecount == 1); + CHECK(secp256k1_ec_pubkey_tweak_mul(ctx, &pubkey, NULL) == 0); + CHECK(ecount == 2); + ecount = 0; + memset(ctmp2, 0, 32); + CHECK(secp256k1_ec_privkey_tweak_add(ctx, NULL, ctmp2) == 0); + CHECK(ecount == 1); + CHECK(secp256k1_ec_privkey_tweak_add(ctx, ctmp, NULL) == 0); + CHECK(ecount == 2); + ecount = 0; + memset(ctmp2, 0, 32); + ctmp2[31] = 1; + CHECK(secp256k1_ec_privkey_tweak_mul(ctx, NULL, ctmp2) == 0); + CHECK(ecount == 1); + CHECK(secp256k1_ec_privkey_tweak_mul(ctx, ctmp, NULL) == 0); + CHECK(ecount == 2); + ecount = 0; + CHECK(secp256k1_ec_pubkey_create(ctx, NULL, ctmp) == 0); + CHECK(ecount == 1); + memset(&pubkey, 1, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey, NULL) == 0); + CHECK(ecount == 2); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) == 0); + /* secp256k1_ec_pubkey_combine tests. */ + ecount = 0; + pubkeys[0] = &pubkey_one; + VG_UNDEF(&pubkeys[0], sizeof(secp256k1_pubkey *)); + VG_UNDEF(&pubkeys[1], sizeof(secp256k1_pubkey *)); + VG_UNDEF(&pubkeys[2], sizeof(secp256k1_pubkey *)); + memset(&pubkey, 255, sizeof(secp256k1_pubkey)); + VG_UNDEF(&pubkey, sizeof(secp256k1_pubkey)); + CHECK(secp256k1_ec_pubkey_combine(ctx, &pubkey, pubkeys, 0) == 0); + VG_CHECK(&pubkey, sizeof(secp256k1_pubkey)); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) == 0); + CHECK(ecount == 1); + CHECK(secp256k1_ec_pubkey_combine(ctx, NULL, pubkeys, 1) == 0); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) == 0); + CHECK(ecount == 2); + memset(&pubkey, 255, sizeof(secp256k1_pubkey)); + VG_UNDEF(&pubkey, sizeof(secp256k1_pubkey)); + CHECK(secp256k1_ec_pubkey_combine(ctx, &pubkey, NULL, 1) == 0); + VG_CHECK(&pubkey, sizeof(secp256k1_pubkey)); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) == 0); + CHECK(ecount == 3); + pubkeys[0] = &pubkey_negone; + memset(&pubkey, 255, sizeof(secp256k1_pubkey)); + VG_UNDEF(&pubkey, sizeof(secp256k1_pubkey)); + CHECK(secp256k1_ec_pubkey_combine(ctx, &pubkey, pubkeys, 1) == 1); + VG_CHECK(&pubkey, sizeof(secp256k1_pubkey)); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) > 0); + CHECK(ecount == 3); + len = 33; + CHECK(secp256k1_ec_pubkey_serialize(ctx, ctmp, &len, &pubkey, SECP256K1_EC_COMPRESSED) == 1); + CHECK(secp256k1_ec_pubkey_serialize(ctx, ctmp2, &len, &pubkey_negone, SECP256K1_EC_COMPRESSED) == 1); + CHECK(memcmp(ctmp, ctmp2, 33) == 0); + /* Result is infinity. */ + pubkeys[0] = &pubkey_one; + pubkeys[1] = &pubkey_negone; + memset(&pubkey, 255, sizeof(secp256k1_pubkey)); + VG_UNDEF(&pubkey, sizeof(secp256k1_pubkey)); + CHECK(secp256k1_ec_pubkey_combine(ctx, &pubkey, pubkeys, 2) == 0); + VG_CHECK(&pubkey, sizeof(secp256k1_pubkey)); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) == 0); + CHECK(ecount == 3); + /* Passes through infinity but comes out one. */ + pubkeys[2] = &pubkey_one; + memset(&pubkey, 255, sizeof(secp256k1_pubkey)); + VG_UNDEF(&pubkey, sizeof(secp256k1_pubkey)); + CHECK(secp256k1_ec_pubkey_combine(ctx, &pubkey, pubkeys, 3) == 1); + VG_CHECK(&pubkey, sizeof(secp256k1_pubkey)); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) > 0); + CHECK(ecount == 3); + len = 33; + CHECK(secp256k1_ec_pubkey_serialize(ctx, ctmp, &len, &pubkey, SECP256K1_EC_COMPRESSED) == 1); + CHECK(secp256k1_ec_pubkey_serialize(ctx, ctmp2, &len, &pubkey_one, SECP256K1_EC_COMPRESSED) == 1); + CHECK(memcmp(ctmp, ctmp2, 33) == 0); + /* Adds to two. */ + pubkeys[1] = &pubkey_one; + memset(&pubkey, 255, sizeof(secp256k1_pubkey)); + VG_UNDEF(&pubkey, sizeof(secp256k1_pubkey)); + CHECK(secp256k1_ec_pubkey_combine(ctx, &pubkey, pubkeys, 2) == 1); + VG_CHECK(&pubkey, sizeof(secp256k1_pubkey)); + CHECK(memcmp(&pubkey, zeros, sizeof(secp256k1_pubkey)) > 0); + CHECK(ecount == 3); + secp256k1_context_set_illegal_callback(ctx, NULL, NULL); +} + +void random_sign(secp256k1_scalar *sigr, secp256k1_scalar *sigs, const secp256k1_scalar *key, const secp256k1_scalar *msg, int *recid) { + secp256k1_scalar nonce; + do { + random_scalar_order_test(&nonce); + } while(!secp256k1_ecdsa_sig_sign(&ctx->ecmult_gen_ctx, sigr, sigs, key, msg, &nonce, recid)); +} + +void test_ecdsa_sign_verify(void) { + secp256k1_gej pubj; + secp256k1_ge pub; + secp256k1_scalar one; + secp256k1_scalar msg, key; + secp256k1_scalar sigr, sigs; + int recid; + int getrec; + random_scalar_order_test(&msg); + random_scalar_order_test(&key); + secp256k1_ecmult_gen(&ctx->ecmult_gen_ctx, &pubj, &key); + secp256k1_ge_set_gej(&pub, &pubj); + getrec = secp256k1_rand_bits(1); + random_sign(&sigr, &sigs, &key, &msg, getrec?&recid:NULL); + if (getrec) { + CHECK(recid >= 0 && recid < 4); + } + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sigr, &sigs, &pub, &msg)); + secp256k1_scalar_set_int(&one, 1); + secp256k1_scalar_add(&msg, &msg, &one); + CHECK(!secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sigr, &sigs, &pub, &msg)); +} + +void run_ecdsa_sign_verify(void) { + int i; + for (i = 0; i < 10*count; i++) { + test_ecdsa_sign_verify(); + } +} + +/** Dummy nonce generation function that just uses a precomputed nonce, and fails if it is not accepted. Use only for testing. */ +static int precomputed_nonce_function(unsigned char *nonce32, const unsigned char *msg32, const unsigned char *key32, const unsigned char *algo16, void *data, unsigned int counter) { + (void)msg32; + (void)key32; + (void)algo16; + memcpy(nonce32, data, 32); + return (counter == 0); +} + +static int nonce_function_test_fail(unsigned char *nonce32, const unsigned char *msg32, const unsigned char *key32, const unsigned char *algo16, void *data, unsigned int counter) { + /* Dummy nonce generator that has a fatal error on the first counter value. */ + if (counter == 0) { + return 0; + } + return nonce_function_rfc6979(nonce32, msg32, key32, algo16, data, counter - 1); +} + +static int nonce_function_test_retry(unsigned char *nonce32, const unsigned char *msg32, const unsigned char *key32, const unsigned char *algo16, void *data, unsigned int counter) { + /* Dummy nonce generator that produces unacceptable nonces for the first several counter values. */ + if (counter < 3) { + memset(nonce32, counter==0 ? 0 : 255, 32); + if (counter == 2) { + nonce32[31]--; + } + return 1; + } + if (counter < 5) { + static const unsigned char order[] = { + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFE, + 0xBA,0xAE,0xDC,0xE6,0xAF,0x48,0xA0,0x3B, + 0xBF,0xD2,0x5E,0x8C,0xD0,0x36,0x41,0x41 + }; + memcpy(nonce32, order, 32); + if (counter == 4) { + nonce32[31]++; + } + return 1; + } + /* Retry rate of 6979 is negligible esp. as we only call this in deterministic tests. */ + /* If someone does fine a case where it retries for secp256k1, we'd like to know. */ + if (counter > 5) { + return 0; + } + return nonce_function_rfc6979(nonce32, msg32, key32, algo16, data, counter - 5); +} + +int is_empty_signature(const secp256k1_ecdsa_signature *sig) { + static const unsigned char res[sizeof(secp256k1_ecdsa_signature)] = {0}; + return memcmp(sig, res, sizeof(secp256k1_ecdsa_signature)) == 0; +} + +void test_ecdsa_end_to_end(void) { + unsigned char extra[32] = {0x00}; + unsigned char privkey[32]; + unsigned char message[32]; + unsigned char privkey2[32]; + secp256k1_ecdsa_signature signature[6]; + secp256k1_scalar r, s; + unsigned char sig[74]; + size_t siglen = 74; + unsigned char pubkeyc[65]; + size_t pubkeyclen = 65; + secp256k1_pubkey pubkey; + secp256k1_pubkey pubkey_tmp; + unsigned char seckey[300]; + size_t seckeylen = 300; + + /* Generate a random key and message. */ + { + secp256k1_scalar msg, key; + random_scalar_order_test(&msg); + random_scalar_order_test(&key); + secp256k1_scalar_get_b32(privkey, &key); + secp256k1_scalar_get_b32(message, &msg); + } + + /* Construct and verify corresponding public key. */ + CHECK(secp256k1_ec_seckey_verify(ctx, privkey) == 1); + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey, privkey) == 1); + + /* Verify exporting and importing public key. */ + CHECK(secp256k1_ec_pubkey_serialize(ctx, pubkeyc, &pubkeyclen, &pubkey, secp256k1_rand_bits(1) == 1 ? SECP256K1_EC_COMPRESSED : SECP256K1_EC_UNCOMPRESSED)); + memset(&pubkey, 0, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_parse(ctx, &pubkey, pubkeyc, pubkeyclen) == 1); + + /* Verify negation changes the key and changes it back */ + memcpy(&pubkey_tmp, &pubkey, sizeof(pubkey)); + CHECK(secp256k1_ec_pubkey_negate(ctx, &pubkey_tmp) == 1); + CHECK(memcmp(&pubkey_tmp, &pubkey, sizeof(pubkey)) != 0); + CHECK(secp256k1_ec_pubkey_negate(ctx, &pubkey_tmp) == 1); + CHECK(memcmp(&pubkey_tmp, &pubkey, sizeof(pubkey)) == 0); + + /* Verify private key import and export. */ + CHECK(ec_privkey_export_der(ctx, seckey, &seckeylen, privkey, secp256k1_rand_bits(1) == 1)); + CHECK(ec_privkey_import_der(ctx, privkey2, seckey, seckeylen) == 1); + CHECK(memcmp(privkey, privkey2, 32) == 0); + + /* Optionally tweak the keys using addition. */ + if (secp256k1_rand_int(3) == 0) { + int ret1; + int ret2; + unsigned char rnd[32]; + secp256k1_pubkey pubkey2; + secp256k1_rand256_test(rnd); + ret1 = secp256k1_ec_privkey_tweak_add(ctx, privkey, rnd); + ret2 = secp256k1_ec_pubkey_tweak_add(ctx, &pubkey, rnd); + CHECK(ret1 == ret2); + if (ret1 == 0) { + return; + } + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey2, privkey) == 1); + CHECK(memcmp(&pubkey, &pubkey2, sizeof(pubkey)) == 0); + } + + /* Optionally tweak the keys using multiplication. */ + if (secp256k1_rand_int(3) == 0) { + int ret1; + int ret2; + unsigned char rnd[32]; + secp256k1_pubkey pubkey2; + secp256k1_rand256_test(rnd); + ret1 = secp256k1_ec_privkey_tweak_mul(ctx, privkey, rnd); + ret2 = secp256k1_ec_pubkey_tweak_mul(ctx, &pubkey, rnd); + CHECK(ret1 == ret2); + if (ret1 == 0) { + return; + } + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey2, privkey) == 1); + CHECK(memcmp(&pubkey, &pubkey2, sizeof(pubkey)) == 0); + } + + /* Sign. */ + CHECK(secp256k1_ecdsa_sign(ctx, &signature[0], message, privkey, NULL, NULL) == 1); + CHECK(secp256k1_ecdsa_sign(ctx, &signature[4], message, privkey, NULL, NULL) == 1); + CHECK(secp256k1_ecdsa_sign(ctx, &signature[1], message, privkey, NULL, extra) == 1); + extra[31] = 1; + CHECK(secp256k1_ecdsa_sign(ctx, &signature[2], message, privkey, NULL, extra) == 1); + extra[31] = 0; + extra[0] = 1; + CHECK(secp256k1_ecdsa_sign(ctx, &signature[3], message, privkey, NULL, extra) == 1); + CHECK(memcmp(&signature[0], &signature[4], sizeof(signature[0])) == 0); + CHECK(memcmp(&signature[0], &signature[1], sizeof(signature[0])) != 0); + CHECK(memcmp(&signature[0], &signature[2], sizeof(signature[0])) != 0); + CHECK(memcmp(&signature[0], &signature[3], sizeof(signature[0])) != 0); + CHECK(memcmp(&signature[1], &signature[2], sizeof(signature[0])) != 0); + CHECK(memcmp(&signature[1], &signature[3], sizeof(signature[0])) != 0); + CHECK(memcmp(&signature[2], &signature[3], sizeof(signature[0])) != 0); + /* Verify. */ + CHECK(secp256k1_ecdsa_verify(ctx, &signature[0], message, &pubkey) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &signature[1], message, &pubkey) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &signature[2], message, &pubkey) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &signature[3], message, &pubkey) == 1); + /* Test lower-S form, malleate, verify and fail, test again, malleate again */ + CHECK(!secp256k1_ecdsa_signature_normalize(ctx, NULL, &signature[0])); + secp256k1_ecdsa_signature_load(ctx, &r, &s, &signature[0]); + secp256k1_scalar_negate(&s, &s); + secp256k1_ecdsa_signature_save(&signature[5], &r, &s); + CHECK(secp256k1_ecdsa_verify(ctx, &signature[5], message, &pubkey) == 0); + CHECK(secp256k1_ecdsa_signature_normalize(ctx, NULL, &signature[5])); + CHECK(secp256k1_ecdsa_signature_normalize(ctx, &signature[5], &signature[5])); + CHECK(!secp256k1_ecdsa_signature_normalize(ctx, NULL, &signature[5])); + CHECK(!secp256k1_ecdsa_signature_normalize(ctx, &signature[5], &signature[5])); + CHECK(secp256k1_ecdsa_verify(ctx, &signature[5], message, &pubkey) == 1); + secp256k1_scalar_negate(&s, &s); + secp256k1_ecdsa_signature_save(&signature[5], &r, &s); + CHECK(!secp256k1_ecdsa_signature_normalize(ctx, NULL, &signature[5])); + CHECK(secp256k1_ecdsa_verify(ctx, &signature[5], message, &pubkey) == 1); + CHECK(memcmp(&signature[5], &signature[0], 64) == 0); + + /* Serialize/parse DER and verify again */ + CHECK(secp256k1_ecdsa_signature_serialize_der(ctx, sig, &siglen, &signature[0]) == 1); + memset(&signature[0], 0, sizeof(signature[0])); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &signature[0], sig, siglen) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, &signature[0], message, &pubkey) == 1); + /* Serialize/destroy/parse DER and verify again. */ + siglen = 74; + CHECK(secp256k1_ecdsa_signature_serialize_der(ctx, sig, &siglen, &signature[0]) == 1); + sig[secp256k1_rand_int(siglen)] += 1 + secp256k1_rand_int(255); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &signature[0], sig, siglen) == 0 || + secp256k1_ecdsa_verify(ctx, &signature[0], message, &pubkey) == 0); +} + +void test_random_pubkeys(void) { + secp256k1_ge elem; + secp256k1_ge elem2; + unsigned char in[65]; + /* Generate some randomly sized pubkeys. */ + size_t len = secp256k1_rand_bits(2) == 0 ? 65 : 33; + if (secp256k1_rand_bits(2) == 0) { + len = secp256k1_rand_bits(6); + } + if (len == 65) { + in[0] = secp256k1_rand_bits(1) ? 4 : (secp256k1_rand_bits(1) ? 6 : 7); + } else { + in[0] = secp256k1_rand_bits(1) ? 2 : 3; + } + if (secp256k1_rand_bits(3) == 0) { + in[0] = secp256k1_rand_bits(8); + } + if (len > 1) { + secp256k1_rand256(&in[1]); + } + if (len > 33) { + secp256k1_rand256(&in[33]); + } + if (secp256k1_eckey_pubkey_parse(&elem, in, len)) { + unsigned char out[65]; + unsigned char firstb; + int res; + size_t size = len; + firstb = in[0]; + /* If the pubkey can be parsed, it should round-trip... */ + CHECK(secp256k1_eckey_pubkey_serialize(&elem, out, &size, len == 33)); + CHECK(size == len); + CHECK(memcmp(&in[1], &out[1], len-1) == 0); + /* ... except for the type of hybrid inputs. */ + if ((in[0] != 6) && (in[0] != 7)) { + CHECK(in[0] == out[0]); + } + size = 65; + CHECK(secp256k1_eckey_pubkey_serialize(&elem, in, &size, 0)); + CHECK(size == 65); + CHECK(secp256k1_eckey_pubkey_parse(&elem2, in, size)); + ge_equals_ge(&elem,&elem2); + /* Check that the X9.62 hybrid type is checked. */ + in[0] = secp256k1_rand_bits(1) ? 6 : 7; + res = secp256k1_eckey_pubkey_parse(&elem2, in, size); + if (firstb == 2 || firstb == 3) { + if (in[0] == firstb + 4) { + CHECK(res); + } else { + CHECK(!res); + } + } + if (res) { + ge_equals_ge(&elem,&elem2); + CHECK(secp256k1_eckey_pubkey_serialize(&elem, out, &size, 0)); + CHECK(memcmp(&in[1], &out[1], 64) == 0); + } + } +} + +void run_random_pubkeys(void) { + int i; + for (i = 0; i < 10*count; i++) { + test_random_pubkeys(); + } +} + +void run_ecdsa_end_to_end(void) { + int i; + for (i = 0; i < 64*count; i++) { + test_ecdsa_end_to_end(); + } +} + +int test_ecdsa_der_parse(const unsigned char *sig, size_t siglen, int certainly_der, int certainly_not_der) { + static const unsigned char zeroes[32] = {0}; +#ifdef ENABLE_OPENSSL_TESTS + static const unsigned char max_scalar[32] = { + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, + 0xba, 0xae, 0xdc, 0xe6, 0xaf, 0x48, 0xa0, 0x3b, + 0xbf, 0xd2, 0x5e, 0x8c, 0xd0, 0x36, 0x41, 0x40 + }; +#endif + + int ret = 0; + + secp256k1_ecdsa_signature sig_der; + unsigned char roundtrip_der[2048]; + unsigned char compact_der[64]; + size_t len_der = 2048; + int parsed_der = 0, valid_der = 0, roundtrips_der = 0; + + secp256k1_ecdsa_signature sig_der_lax; + unsigned char roundtrip_der_lax[2048]; + unsigned char compact_der_lax[64]; + size_t len_der_lax = 2048; + int parsed_der_lax = 0, valid_der_lax = 0, roundtrips_der_lax = 0; + +#ifdef ENABLE_OPENSSL_TESTS + ECDSA_SIG *sig_openssl; + const BIGNUM *r = NULL, *s = NULL; + const unsigned char *sigptr; + unsigned char roundtrip_openssl[2048]; + int len_openssl = 2048; + int parsed_openssl, valid_openssl = 0, roundtrips_openssl = 0; +#endif + + parsed_der = secp256k1_ecdsa_signature_parse_der(ctx, &sig_der, sig, siglen); + if (parsed_der) { + ret |= (!secp256k1_ecdsa_signature_serialize_compact(ctx, compact_der, &sig_der)) << 0; + valid_der = (memcmp(compact_der, zeroes, 32) != 0) && (memcmp(compact_der + 32, zeroes, 32) != 0); + } + if (valid_der) { + ret |= (!secp256k1_ecdsa_signature_serialize_der(ctx, roundtrip_der, &len_der, &sig_der)) << 1; + roundtrips_der = (len_der == siglen) && memcmp(roundtrip_der, sig, siglen) == 0; + } + + parsed_der_lax = ecdsa_signature_parse_der_lax(ctx, &sig_der_lax, sig, siglen); + if (parsed_der_lax) { + ret |= (!secp256k1_ecdsa_signature_serialize_compact(ctx, compact_der_lax, &sig_der_lax)) << 10; + valid_der_lax = (memcmp(compact_der_lax, zeroes, 32) != 0) && (memcmp(compact_der_lax + 32, zeroes, 32) != 0); + } + if (valid_der_lax) { + ret |= (!secp256k1_ecdsa_signature_serialize_der(ctx, roundtrip_der_lax, &len_der_lax, &sig_der_lax)) << 11; + roundtrips_der_lax = (len_der_lax == siglen) && memcmp(roundtrip_der_lax, sig, siglen) == 0; + } + + if (certainly_der) { + ret |= (!parsed_der) << 2; + } + if (certainly_not_der) { + ret |= (parsed_der) << 17; + } + if (valid_der) { + ret |= (!roundtrips_der) << 3; + } + + if (valid_der) { + ret |= (!roundtrips_der_lax) << 12; + ret |= (len_der != len_der_lax) << 13; + ret |= ((len_der != len_der_lax) || (memcmp(roundtrip_der_lax, roundtrip_der, len_der) != 0)) << 14; + } + ret |= (roundtrips_der != roundtrips_der_lax) << 15; + if (parsed_der) { + ret |= (!parsed_der_lax) << 16; + } + +#ifdef ENABLE_OPENSSL_TESTS + sig_openssl = ECDSA_SIG_new(); + sigptr = sig; + parsed_openssl = (d2i_ECDSA_SIG(&sig_openssl, &sigptr, siglen) != NULL); + if (parsed_openssl) { + ECDSA_SIG_get0(sig_openssl, &r, &s); + valid_openssl = !BN_is_negative(r) && !BN_is_negative(s) && BN_num_bits(r) > 0 && BN_num_bits(r) <= 256 && BN_num_bits(s) > 0 && BN_num_bits(s) <= 256; + if (valid_openssl) { + unsigned char tmp[32] = {0}; + BN_bn2bin(r, tmp + 32 - BN_num_bytes(r)); + valid_openssl = memcmp(tmp, max_scalar, 32) < 0; + } + if (valid_openssl) { + unsigned char tmp[32] = {0}; + BN_bn2bin(s, tmp + 32 - BN_num_bytes(s)); + valid_openssl = memcmp(tmp, max_scalar, 32) < 0; + } + } + len_openssl = i2d_ECDSA_SIG(sig_openssl, NULL); + if (len_openssl <= 2048) { + unsigned char *ptr = roundtrip_openssl; + CHECK(i2d_ECDSA_SIG(sig_openssl, &ptr) == len_openssl); + roundtrips_openssl = valid_openssl && ((size_t)len_openssl == siglen) && (memcmp(roundtrip_openssl, sig, siglen) == 0); + } else { + len_openssl = 0; + } + ECDSA_SIG_free(sig_openssl); + + ret |= (parsed_der && !parsed_openssl) << 4; + ret |= (valid_der && !valid_openssl) << 5; + ret |= (roundtrips_openssl && !parsed_der) << 6; + ret |= (roundtrips_der != roundtrips_openssl) << 7; + if (roundtrips_openssl) { + ret |= (len_der != (size_t)len_openssl) << 8; + ret |= ((len_der != (size_t)len_openssl) || (memcmp(roundtrip_der, roundtrip_openssl, len_der) != 0)) << 9; + } +#endif + return ret; +} + +static void assign_big_endian(unsigned char *ptr, size_t ptrlen, uint32_t val) { + size_t i; + for (i = 0; i < ptrlen; i++) { + int shift = ptrlen - 1 - i; + if (shift >= 4) { + ptr[i] = 0; + } else { + ptr[i] = (val >> shift) & 0xFF; + } + } +} + +static void damage_array(unsigned char *sig, size_t *len) { + int pos; + int action = secp256k1_rand_bits(3); + if (action < 1 && *len > 3) { + /* Delete a byte. */ + pos = secp256k1_rand_int(*len); + memmove(sig + pos, sig + pos + 1, *len - pos - 1); + (*len)--; + return; + } else if (action < 2 && *len < 2048) { + /* Insert a byte. */ + pos = secp256k1_rand_int(1 + *len); + memmove(sig + pos + 1, sig + pos, *len - pos); + sig[pos] = secp256k1_rand_bits(8); + (*len)++; + return; + } else if (action < 4) { + /* Modify a byte. */ + sig[secp256k1_rand_int(*len)] += 1 + secp256k1_rand_int(255); + return; + } else { /* action < 8 */ + /* Modify a bit. */ + sig[secp256k1_rand_int(*len)] ^= 1 << secp256k1_rand_bits(3); + return; + } +} + +static void random_ber_signature(unsigned char *sig, size_t *len, int* certainly_der, int* certainly_not_der) { + int der; + int nlow[2], nlen[2], nlenlen[2], nhbit[2], nhbyte[2], nzlen[2]; + size_t tlen, elen, glen; + int indet; + int n; + + *len = 0; + der = secp256k1_rand_bits(2) == 0; + *certainly_der = der; + *certainly_not_der = 0; + indet = der ? 0 : secp256k1_rand_int(10) == 0; + + for (n = 0; n < 2; n++) { + /* We generate two classes of numbers: nlow==1 "low" ones (up to 32 bytes), nlow==0 "high" ones (32 bytes with 129 top bits set, or larger than 32 bytes) */ + nlow[n] = der ? 1 : (secp256k1_rand_bits(3) != 0); + /* The length of the number in bytes (the first byte of which will always be nonzero) */ + nlen[n] = nlow[n] ? secp256k1_rand_int(33) : 32 + secp256k1_rand_int(200) * secp256k1_rand_int(8) / 8; + CHECK(nlen[n] <= 232); + /* The top bit of the number. */ + nhbit[n] = (nlow[n] == 0 && nlen[n] == 32) ? 1 : (nlen[n] == 0 ? 0 : secp256k1_rand_bits(1)); + /* The top byte of the number (after the potential hardcoded 16 0xFF characters for "high" 32 bytes numbers) */ + nhbyte[n] = nlen[n] == 0 ? 0 : (nhbit[n] ? 128 + secp256k1_rand_bits(7) : 1 + secp256k1_rand_int(127)); + /* The number of zero bytes in front of the number (which is 0 or 1 in case of DER, otherwise we extend up to 300 bytes) */ + nzlen[n] = der ? ((nlen[n] == 0 || nhbit[n]) ? 1 : 0) : (nlow[n] ? secp256k1_rand_int(3) : secp256k1_rand_int(300 - nlen[n]) * secp256k1_rand_int(8) / 8); + if (nzlen[n] > ((nlen[n] == 0 || nhbit[n]) ? 1 : 0)) { + *certainly_not_der = 1; + } + CHECK(nlen[n] + nzlen[n] <= 300); + /* The length of the length descriptor for the number. 0 means short encoding, anything else is long encoding. */ + nlenlen[n] = nlen[n] + nzlen[n] < 128 ? 0 : (nlen[n] + nzlen[n] < 256 ? 1 : 2); + if (!der) { + /* nlenlen[n] max 127 bytes */ + int add = secp256k1_rand_int(127 - nlenlen[n]) * secp256k1_rand_int(16) * secp256k1_rand_int(16) / 256; + nlenlen[n] += add; + if (add != 0) { + *certainly_not_der = 1; + } + } + CHECK(nlen[n] + nzlen[n] + nlenlen[n] <= 427); + } + + /* The total length of the data to go, so far */ + tlen = 2 + nlenlen[0] + nlen[0] + nzlen[0] + 2 + nlenlen[1] + nlen[1] + nzlen[1]; + CHECK(tlen <= 856); + + /* The length of the garbage inside the tuple. */ + elen = (der || indet) ? 0 : secp256k1_rand_int(980 - tlen) * secp256k1_rand_int(8) / 8; + if (elen != 0) { + *certainly_not_der = 1; + } + tlen += elen; + CHECK(tlen <= 980); + + /* The length of the garbage after the end of the tuple. */ + glen = der ? 0 : secp256k1_rand_int(990 - tlen) * secp256k1_rand_int(8) / 8; + if (glen != 0) { + *certainly_not_der = 1; + } + CHECK(tlen + glen <= 990); + + /* Write the tuple header. */ + sig[(*len)++] = 0x30; + if (indet) { + /* Indeterminate length */ + sig[(*len)++] = 0x80; + *certainly_not_der = 1; + } else { + int tlenlen = tlen < 128 ? 0 : (tlen < 256 ? 1 : 2); + if (!der) { + int add = secp256k1_rand_int(127 - tlenlen) * secp256k1_rand_int(16) * secp256k1_rand_int(16) / 256; + tlenlen += add; + if (add != 0) { + *certainly_not_der = 1; + } + } + if (tlenlen == 0) { + /* Short length notation */ + sig[(*len)++] = tlen; + } else { + /* Long length notation */ + sig[(*len)++] = 128 + tlenlen; + assign_big_endian(sig + *len, tlenlen, tlen); + *len += tlenlen; + } + tlen += tlenlen; + } + tlen += 2; + CHECK(tlen + glen <= 1119); + + for (n = 0; n < 2; n++) { + /* Write the integer header. */ + sig[(*len)++] = 0x02; + if (nlenlen[n] == 0) { + /* Short length notation */ + sig[(*len)++] = nlen[n] + nzlen[n]; + } else { + /* Long length notation. */ + sig[(*len)++] = 128 + nlenlen[n]; + assign_big_endian(sig + *len, nlenlen[n], nlen[n] + nzlen[n]); + *len += nlenlen[n]; + } + /* Write zero padding */ + while (nzlen[n] > 0) { + sig[(*len)++] = 0x00; + nzlen[n]--; + } + if (nlen[n] == 32 && !nlow[n]) { + /* Special extra 16 0xFF bytes in "high" 32-byte numbers */ + int i; + for (i = 0; i < 16; i++) { + sig[(*len)++] = 0xFF; + } + nlen[n] -= 16; + } + /* Write first byte of number */ + if (nlen[n] > 0) { + sig[(*len)++] = nhbyte[n]; + nlen[n]--; + } + /* Generate remaining random bytes of number */ + secp256k1_rand_bytes_test(sig + *len, nlen[n]); + *len += nlen[n]; + nlen[n] = 0; + } + + /* Generate random garbage inside tuple. */ + secp256k1_rand_bytes_test(sig + *len, elen); + *len += elen; + + /* Generate end-of-contents bytes. */ + if (indet) { + sig[(*len)++] = 0; + sig[(*len)++] = 0; + tlen += 2; + } + CHECK(tlen + glen <= 1121); + + /* Generate random garbage outside tuple. */ + secp256k1_rand_bytes_test(sig + *len, glen); + *len += glen; + tlen += glen; + CHECK(tlen <= 1121); + CHECK(tlen == *len); +} + +void run_ecdsa_der_parse(void) { + int i,j; + for (i = 0; i < 200 * count; i++) { + unsigned char buffer[2048]; + size_t buflen = 0; + int certainly_der = 0; + int certainly_not_der = 0; + random_ber_signature(buffer, &buflen, &certainly_der, &certainly_not_der); + CHECK(buflen <= 2048); + for (j = 0; j < 16; j++) { + int ret = 0; + if (j > 0) { + damage_array(buffer, &buflen); + /* We don't know anything anymore about the DERness of the result */ + certainly_der = 0; + certainly_not_der = 0; + } + ret = test_ecdsa_der_parse(buffer, buflen, certainly_der, certainly_not_der); + if (ret != 0) { + size_t k; + fprintf(stderr, "Failure %x on ", ret); + for (k = 0; k < buflen; k++) { + fprintf(stderr, "%02x ", buffer[k]); + } + fprintf(stderr, "\n"); + } + CHECK(ret == 0); + } + } +} + +/* Tests several edge cases. */ +void test_ecdsa_edge_cases(void) { + int t; + secp256k1_ecdsa_signature sig; + + /* Test the case where ECDSA recomputes a point that is infinity. */ + { + secp256k1_gej keyj; + secp256k1_ge key; + secp256k1_scalar msg; + secp256k1_scalar sr, ss; + secp256k1_scalar_set_int(&ss, 1); + secp256k1_scalar_negate(&ss, &ss); + secp256k1_scalar_inverse(&ss, &ss); + secp256k1_scalar_set_int(&sr, 1); + secp256k1_ecmult_gen(&ctx->ecmult_gen_ctx, &keyj, &sr); + secp256k1_ge_set_gej(&key, &keyj); + msg = ss; + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key, &msg) == 0); + } + + /* Verify signature with r of zero fails. */ + { + const unsigned char pubkey_mods_zero[33] = { + 0x02, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xfe, 0xba, 0xae, 0xdc, 0xe6, 0xaf, 0x48, 0xa0, + 0x3b, 0xbf, 0xd2, 0x5e, 0x8c, 0xd0, 0x36, 0x41, + 0x41 + }; + secp256k1_ge key; + secp256k1_scalar msg; + secp256k1_scalar sr, ss; + secp256k1_scalar_set_int(&ss, 1); + secp256k1_scalar_set_int(&msg, 0); + secp256k1_scalar_set_int(&sr, 0); + CHECK(secp256k1_eckey_pubkey_parse(&key, pubkey_mods_zero, 33)); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key, &msg) == 0); + } + + /* Verify signature with s of zero fails. */ + { + const unsigned char pubkey[33] = { + 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x01 + }; + secp256k1_ge key; + secp256k1_scalar msg; + secp256k1_scalar sr, ss; + secp256k1_scalar_set_int(&ss, 0); + secp256k1_scalar_set_int(&msg, 0); + secp256k1_scalar_set_int(&sr, 1); + CHECK(secp256k1_eckey_pubkey_parse(&key, pubkey, 33)); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key, &msg) == 0); + } + + /* Verify signature with message 0 passes. */ + { + const unsigned char pubkey[33] = { + 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x02 + }; + const unsigned char pubkey2[33] = { + 0x02, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xfe, 0xba, 0xae, 0xdc, 0xe6, 0xaf, 0x48, 0xa0, + 0x3b, 0xbf, 0xd2, 0x5e, 0x8c, 0xd0, 0x36, 0x41, + 0x43 + }; + secp256k1_ge key; + secp256k1_ge key2; + secp256k1_scalar msg; + secp256k1_scalar sr, ss; + secp256k1_scalar_set_int(&ss, 2); + secp256k1_scalar_set_int(&msg, 0); + secp256k1_scalar_set_int(&sr, 2); + CHECK(secp256k1_eckey_pubkey_parse(&key, pubkey, 33)); + CHECK(secp256k1_eckey_pubkey_parse(&key2, pubkey2, 33)); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key, &msg) == 1); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key2, &msg) == 1); + secp256k1_scalar_negate(&ss, &ss); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key, &msg) == 1); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key2, &msg) == 1); + secp256k1_scalar_set_int(&ss, 1); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key, &msg) == 0); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key2, &msg) == 0); + } + + /* Verify signature with message 1 passes. */ + { + const unsigned char pubkey[33] = { + 0x02, 0x14, 0x4e, 0x5a, 0x58, 0xef, 0x5b, 0x22, + 0x6f, 0xd2, 0xe2, 0x07, 0x6a, 0x77, 0xcf, 0x05, + 0xb4, 0x1d, 0xe7, 0x4a, 0x30, 0x98, 0x27, 0x8c, + 0x93, 0xe6, 0xe6, 0x3c, 0x0b, 0xc4, 0x73, 0x76, + 0x25 + }; + const unsigned char pubkey2[33] = { + 0x02, 0x8a, 0xd5, 0x37, 0xed, 0x73, 0xd9, 0x40, + 0x1d, 0xa0, 0x33, 0xd2, 0xdc, 0xf0, 0xaf, 0xae, + 0x34, 0xcf, 0x5f, 0x96, 0x4c, 0x73, 0x28, 0x0f, + 0x92, 0xc0, 0xf6, 0x9d, 0xd9, 0xb2, 0x09, 0x10, + 0x62 + }; + const unsigned char csr[32] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + 0x45, 0x51, 0x23, 0x19, 0x50, 0xb7, 0x5f, 0xc4, + 0x40, 0x2d, 0xa1, 0x72, 0x2f, 0xc9, 0xba, 0xeb + }; + secp256k1_ge key; + secp256k1_ge key2; + secp256k1_scalar msg; + secp256k1_scalar sr, ss; + secp256k1_scalar_set_int(&ss, 1); + secp256k1_scalar_set_int(&msg, 1); + secp256k1_scalar_set_b32(&sr, csr, NULL); + CHECK(secp256k1_eckey_pubkey_parse(&key, pubkey, 33)); + CHECK(secp256k1_eckey_pubkey_parse(&key2, pubkey2, 33)); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key, &msg) == 1); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key2, &msg) == 1); + secp256k1_scalar_negate(&ss, &ss); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key, &msg) == 1); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key2, &msg) == 1); + secp256k1_scalar_set_int(&ss, 2); + secp256k1_scalar_inverse_var(&ss, &ss); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key, &msg) == 0); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key2, &msg) == 0); + } + + /* Verify signature with message -1 passes. */ + { + const unsigned char pubkey[33] = { + 0x03, 0xaf, 0x97, 0xff, 0x7d, 0x3a, 0xf6, 0xa0, + 0x02, 0x94, 0xbd, 0x9f, 0x4b, 0x2e, 0xd7, 0x52, + 0x28, 0xdb, 0x49, 0x2a, 0x65, 0xcb, 0x1e, 0x27, + 0x57, 0x9c, 0xba, 0x74, 0x20, 0xd5, 0x1d, 0x20, + 0xf1 + }; + const unsigned char csr[32] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + 0x45, 0x51, 0x23, 0x19, 0x50, 0xb7, 0x5f, 0xc4, + 0x40, 0x2d, 0xa1, 0x72, 0x2f, 0xc9, 0xba, 0xee + }; + secp256k1_ge key; + secp256k1_scalar msg; + secp256k1_scalar sr, ss; + secp256k1_scalar_set_int(&ss, 1); + secp256k1_scalar_set_int(&msg, 1); + secp256k1_scalar_negate(&msg, &msg); + secp256k1_scalar_set_b32(&sr, csr, NULL); + CHECK(secp256k1_eckey_pubkey_parse(&key, pubkey, 33)); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key, &msg) == 1); + secp256k1_scalar_negate(&ss, &ss); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key, &msg) == 1); + secp256k1_scalar_set_int(&ss, 3); + secp256k1_scalar_inverse_var(&ss, &ss); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sr, &ss, &key, &msg) == 0); + } + + /* Signature where s would be zero. */ + { + secp256k1_pubkey pubkey; + size_t siglen; + int32_t ecount; + unsigned char signature[72]; + static const unsigned char nonce[32] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + }; + static const unsigned char nonce2[32] = { + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, + 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFE, + 0xBA,0xAE,0xDC,0xE6,0xAF,0x48,0xA0,0x3B, + 0xBF,0xD2,0x5E,0x8C,0xD0,0x36,0x41,0x40 + }; + const unsigned char key[32] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + }; + unsigned char msg[32] = { + 0x86, 0x41, 0x99, 0x81, 0x06, 0x23, 0x44, 0x53, + 0xaa, 0x5f, 0x9d, 0x6a, 0x31, 0x78, 0xf4, 0xf7, + 0xb8, 0x12, 0xe0, 0x0b, 0x81, 0x7a, 0x77, 0x62, + 0x65, 0xdf, 0xdd, 0x31, 0xb9, 0x3e, 0x29, 0xa9, + }; + ecount = 0; + secp256k1_context_set_illegal_callback(ctx, counting_illegal_callback_fn, &ecount); + CHECK(secp256k1_ecdsa_sign(ctx, &sig, msg, key, precomputed_nonce_function, nonce) == 0); + CHECK(secp256k1_ecdsa_sign(ctx, &sig, msg, key, precomputed_nonce_function, nonce2) == 0); + msg[31] = 0xaa; + CHECK(secp256k1_ecdsa_sign(ctx, &sig, msg, key, precomputed_nonce_function, nonce) == 1); + CHECK(ecount == 0); + CHECK(secp256k1_ecdsa_sign(ctx, NULL, msg, key, precomputed_nonce_function, nonce2) == 0); + CHECK(ecount == 1); + CHECK(secp256k1_ecdsa_sign(ctx, &sig, NULL, key, precomputed_nonce_function, nonce2) == 0); + CHECK(ecount == 2); + CHECK(secp256k1_ecdsa_sign(ctx, &sig, msg, NULL, precomputed_nonce_function, nonce2) == 0); + CHECK(ecount == 3); + CHECK(secp256k1_ecdsa_sign(ctx, &sig, msg, key, precomputed_nonce_function, nonce2) == 1); + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey, key) == 1); + CHECK(secp256k1_ecdsa_verify(ctx, NULL, msg, &pubkey) == 0); + CHECK(ecount == 4); + CHECK(secp256k1_ecdsa_verify(ctx, &sig, NULL, &pubkey) == 0); + CHECK(ecount == 5); + CHECK(secp256k1_ecdsa_verify(ctx, &sig, msg, NULL) == 0); + CHECK(ecount == 6); + CHECK(secp256k1_ecdsa_verify(ctx, &sig, msg, &pubkey) == 1); + CHECK(ecount == 6); + CHECK(secp256k1_ec_pubkey_create(ctx, &pubkey, NULL) == 0); + CHECK(ecount == 7); + /* That pubkeyload fails via an ARGCHECK is a little odd but makes sense because pubkeys are an opaque data type. */ + CHECK(secp256k1_ecdsa_verify(ctx, &sig, msg, &pubkey) == 0); + CHECK(ecount == 8); + siglen = 72; + CHECK(secp256k1_ecdsa_signature_serialize_der(ctx, NULL, &siglen, &sig) == 0); + CHECK(ecount == 9); + CHECK(secp256k1_ecdsa_signature_serialize_der(ctx, signature, NULL, &sig) == 0); + CHECK(ecount == 10); + CHECK(secp256k1_ecdsa_signature_serialize_der(ctx, signature, &siglen, NULL) == 0); + CHECK(ecount == 11); + CHECK(secp256k1_ecdsa_signature_serialize_der(ctx, signature, &siglen, &sig) == 1); + CHECK(ecount == 11); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, NULL, signature, siglen) == 0); + CHECK(ecount == 12); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, NULL, siglen) == 0); + CHECK(ecount == 13); + CHECK(secp256k1_ecdsa_signature_parse_der(ctx, &sig, signature, siglen) == 1); + CHECK(ecount == 13); + siglen = 10; + /* Too little room for a signature does not fail via ARGCHECK. */ + CHECK(secp256k1_ecdsa_signature_serialize_der(ctx, signature, &siglen, &sig) == 0); + CHECK(ecount == 13); + ecount = 0; + CHECK(secp256k1_ecdsa_signature_normalize(ctx, NULL, NULL) == 0); + CHECK(ecount == 1); + CHECK(secp256k1_ecdsa_signature_serialize_compact(ctx, NULL, &sig) == 0); + CHECK(ecount == 2); + CHECK(secp256k1_ecdsa_signature_serialize_compact(ctx, signature, NULL) == 0); + CHECK(ecount == 3); + CHECK(secp256k1_ecdsa_signature_serialize_compact(ctx, signature, &sig) == 1); + CHECK(ecount == 3); + CHECK(secp256k1_ecdsa_signature_parse_compact(ctx, NULL, signature) == 0); + CHECK(ecount == 4); + CHECK(secp256k1_ecdsa_signature_parse_compact(ctx, &sig, NULL) == 0); + CHECK(ecount == 5); + CHECK(secp256k1_ecdsa_signature_parse_compact(ctx, &sig, signature) == 1); + CHECK(ecount == 5); + memset(signature, 255, 64); + CHECK(secp256k1_ecdsa_signature_parse_compact(ctx, &sig, signature) == 0); + CHECK(ecount == 5); + secp256k1_context_set_illegal_callback(ctx, NULL, NULL); + } + + /* Nonce function corner cases. */ + for (t = 0; t < 2; t++) { + static const unsigned char zero[32] = {0x00}; + int i; + unsigned char key[32]; + unsigned char msg[32]; + secp256k1_ecdsa_signature sig2; + secp256k1_scalar sr[512], ss; + const unsigned char *extra; + extra = t == 0 ? NULL : zero; + memset(msg, 0, 32); + msg[31] = 1; + /* High key results in signature failure. */ + memset(key, 0xFF, 32); + CHECK(secp256k1_ecdsa_sign(ctx, &sig, msg, key, NULL, extra) == 0); + CHECK(is_empty_signature(&sig)); + /* Zero key results in signature failure. */ + memset(key, 0, 32); + CHECK(secp256k1_ecdsa_sign(ctx, &sig, msg, key, NULL, extra) == 0); + CHECK(is_empty_signature(&sig)); + /* Nonce function failure results in signature failure. */ + key[31] = 1; + CHECK(secp256k1_ecdsa_sign(ctx, &sig, msg, key, nonce_function_test_fail, extra) == 0); + CHECK(is_empty_signature(&sig)); + /* The retry loop successfully makes its way to the first good value. */ + CHECK(secp256k1_ecdsa_sign(ctx, &sig, msg, key, nonce_function_test_retry, extra) == 1); + CHECK(!is_empty_signature(&sig)); + CHECK(secp256k1_ecdsa_sign(ctx, &sig2, msg, key, nonce_function_rfc6979, extra) == 1); + CHECK(!is_empty_signature(&sig2)); + CHECK(memcmp(&sig, &sig2, sizeof(sig)) == 0); + /* The default nonce function is deterministic. */ + CHECK(secp256k1_ecdsa_sign(ctx, &sig2, msg, key, NULL, extra) == 1); + CHECK(!is_empty_signature(&sig2)); + CHECK(memcmp(&sig, &sig2, sizeof(sig)) == 0); + /* The default nonce function changes output with different messages. */ + for(i = 0; i < 256; i++) { + int j; + msg[0] = i; + CHECK(secp256k1_ecdsa_sign(ctx, &sig2, msg, key, NULL, extra) == 1); + CHECK(!is_empty_signature(&sig2)); + secp256k1_ecdsa_signature_load(ctx, &sr[i], &ss, &sig2); + for (j = 0; j < i; j++) { + CHECK(!secp256k1_scalar_eq(&sr[i], &sr[j])); + } + } + msg[0] = 0; + msg[31] = 2; + /* The default nonce function changes output with different keys. */ + for(i = 256; i < 512; i++) { + int j; + key[0] = i - 256; + CHECK(secp256k1_ecdsa_sign(ctx, &sig2, msg, key, NULL, extra) == 1); + CHECK(!is_empty_signature(&sig2)); + secp256k1_ecdsa_signature_load(ctx, &sr[i], &ss, &sig2); + for (j = 0; j < i; j++) { + CHECK(!secp256k1_scalar_eq(&sr[i], &sr[j])); + } + } + key[0] = 0; + } + + { + /* Check that optional nonce arguments do not have equivalent effect. */ + const unsigned char zeros[32] = {0}; + unsigned char nonce[32]; + unsigned char nonce2[32]; + unsigned char nonce3[32]; + unsigned char nonce4[32]; + VG_UNDEF(nonce,32); + VG_UNDEF(nonce2,32); + VG_UNDEF(nonce3,32); + VG_UNDEF(nonce4,32); + CHECK(nonce_function_rfc6979(nonce, zeros, zeros, NULL, NULL, 0) == 1); + VG_CHECK(nonce,32); + CHECK(nonce_function_rfc6979(nonce2, zeros, zeros, zeros, NULL, 0) == 1); + VG_CHECK(nonce2,32); + CHECK(nonce_function_rfc6979(nonce3, zeros, zeros, NULL, (void *)zeros, 0) == 1); + VG_CHECK(nonce3,32); + CHECK(nonce_function_rfc6979(nonce4, zeros, zeros, zeros, (void *)zeros, 0) == 1); + VG_CHECK(nonce4,32); + CHECK(memcmp(nonce, nonce2, 32) != 0); + CHECK(memcmp(nonce, nonce3, 32) != 0); + CHECK(memcmp(nonce, nonce4, 32) != 0); + CHECK(memcmp(nonce2, nonce3, 32) != 0); + CHECK(memcmp(nonce2, nonce4, 32) != 0); + CHECK(memcmp(nonce3, nonce4, 32) != 0); + } + + + /* Privkey export where pubkey is the point at infinity. */ + { + unsigned char privkey[300]; + unsigned char seckey[32] = { + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, + 0xba, 0xae, 0xdc, 0xe6, 0xaf, 0x48, 0xa0, 0x3b, + 0xbf, 0xd2, 0x5e, 0x8c, 0xd0, 0x36, 0x41, 0x41, + }; + size_t outlen = 300; + CHECK(!ec_privkey_export_der(ctx, privkey, &outlen, seckey, 0)); + outlen = 300; + CHECK(!ec_privkey_export_der(ctx, privkey, &outlen, seckey, 1)); + } +} + +void run_ecdsa_edge_cases(void) { + test_ecdsa_edge_cases(); +} + +#ifdef ENABLE_OPENSSL_TESTS +EC_KEY *get_openssl_key(const unsigned char *key32) { + unsigned char privkey[300]; + size_t privkeylen; + const unsigned char* pbegin = privkey; + int compr = secp256k1_rand_bits(1); + EC_KEY *ec_key = EC_KEY_new_by_curve_name(NID_secp256k1); + CHECK(ec_privkey_export_der(ctx, privkey, &privkeylen, key32, compr)); + CHECK(d2i_ECPrivateKey(&ec_key, &pbegin, privkeylen)); + CHECK(EC_KEY_check_key(ec_key)); + return ec_key; +} + +void test_ecdsa_openssl(void) { + secp256k1_gej qj; + secp256k1_ge q; + secp256k1_scalar sigr, sigs; + secp256k1_scalar one; + secp256k1_scalar msg2; + secp256k1_scalar key, msg; + EC_KEY *ec_key; + unsigned int sigsize = 80; + size_t secp_sigsize = 80; + unsigned char message[32]; + unsigned char signature[80]; + unsigned char key32[32]; + secp256k1_rand256_test(message); + secp256k1_scalar_set_b32(&msg, message, NULL); + random_scalar_order_test(&key); + secp256k1_scalar_get_b32(key32, &key); + secp256k1_ecmult_gen(&ctx->ecmult_gen_ctx, &qj, &key); + secp256k1_ge_set_gej(&q, &qj); + ec_key = get_openssl_key(key32); + CHECK(ec_key != NULL); + CHECK(ECDSA_sign(0, message, sizeof(message), signature, &sigsize, ec_key)); + CHECK(secp256k1_ecdsa_sig_parse(&sigr, &sigs, signature, sigsize)); + CHECK(secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sigr, &sigs, &q, &msg)); + secp256k1_scalar_set_int(&one, 1); + secp256k1_scalar_add(&msg2, &msg, &one); + CHECK(!secp256k1_ecdsa_sig_verify(&ctx->ecmult_ctx, &sigr, &sigs, &q, &msg2)); + + random_sign(&sigr, &sigs, &key, &msg, NULL); + CHECK(secp256k1_ecdsa_sig_serialize(signature, &secp_sigsize, &sigr, &sigs)); + CHECK(ECDSA_verify(0, message, sizeof(message), signature, secp_sigsize, ec_key) == 1); + + EC_KEY_free(ec_key); +} + +void run_ecdsa_openssl(void) { + int i; + for (i = 0; i < 10*count; i++) { + test_ecdsa_openssl(); + } +} +#endif + +#ifdef ENABLE_MODULE_ECDH +# include "modules/ecdh/tests_impl.h" +#endif + +#ifdef ENABLE_MODULE_RECOVERY +# include "modules/recovery/tests_impl.h" +#endif + +int main(int argc, char **argv) { + unsigned char seed16[16] = {0}; + unsigned char run32[32] = {0}; + + /* Disable buffering for stdout to improve reliability of getting + * diagnostic information. Happens right at the start of main because + * setbuf must be used before any other operation on the stream. */ + setbuf(stdout, NULL); + /* Also disable buffering for stderr because it's not guaranteed that it's + * unbuffered on all systems. */ + setbuf(stderr, NULL); + + /* find iteration count */ + if (argc > 1) { + count = strtol(argv[1], NULL, 0); + } + + /* find random seed */ + if (argc > 2) { + int pos = 0; + const char* ch = argv[2]; + while (pos < 16 && ch[0] != 0 && ch[1] != 0) { + unsigned short sh; + if ((sscanf(ch, "%2hx", &sh)) == 1) { + seed16[pos] = sh; + } else { + break; + } + ch += 2; + pos++; + } + } else { + FILE *frand = fopen("/dev/urandom", "r"); + if ((frand == NULL) || fread(&seed16, 1, sizeof(seed16), frand) != sizeof(seed16)) { + uint64_t t = time(NULL) * (uint64_t)1337; + fprintf(stderr, "WARNING: could not read 16 bytes from /dev/urandom; falling back to insecure PRNG\n"); + seed16[0] ^= t; + seed16[1] ^= t >> 8; + seed16[2] ^= t >> 16; + seed16[3] ^= t >> 24; + seed16[4] ^= t >> 32; + seed16[5] ^= t >> 40; + seed16[6] ^= t >> 48; + seed16[7] ^= t >> 56; + } + if (frand) { + fclose(frand); + } + } + secp256k1_rand_seed(seed16); + + printf("test count = %i\n", count); + printf("random seed = %02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x\n", seed16[0], seed16[1], seed16[2], seed16[3], seed16[4], seed16[5], seed16[6], seed16[7], seed16[8], seed16[9], seed16[10], seed16[11], seed16[12], seed16[13], seed16[14], seed16[15]); + + /* initialize */ + run_context_tests(0); + run_context_tests(1); + run_scratch_tests(); + ctx = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY); + if (secp256k1_rand_bits(1)) { + secp256k1_rand256(run32); + CHECK(secp256k1_context_randomize(ctx, secp256k1_rand_bits(1) ? run32 : NULL)); + } + + run_rand_bits(); + run_rand_int(); + + run_sha256_tests(); + run_hmac_sha256_tests(); + run_rfc6979_hmac_sha256_tests(); + +#ifndef USE_NUM_NONE + /* num tests */ + run_num_smalltests(); +#endif + + /* scalar tests */ + run_scalar_tests(); + + /* field tests */ + run_field_inv(); + run_field_inv_var(); + run_field_inv_all_var(); + run_field_misc(); + run_field_convert(); + run_sqr(); + run_sqrt(); + + /* group tests */ + run_ge(); + run_group_decompress(); + + /* ecmult tests */ + run_wnaf(); + run_point_times_order(); + run_ecmult_chain(); + run_ecmult_constants(); + run_ecmult_gen_blind(); + run_ecmult_const_tests(); + run_ecmult_multi_tests(); + run_ec_combine(); + + /* endomorphism tests */ +#ifdef USE_ENDOMORPHISM + run_endomorphism_tests(); +#endif + + /* EC point parser test */ + run_ec_pubkey_parse_test(); + + /* EC key edge cases */ + run_eckey_edge_case_test(); + +#ifdef ENABLE_MODULE_ECDH + /* ecdh tests */ + run_ecdh_tests(); +#endif + + /* ecdsa tests */ + run_random_pubkeys(); + run_ecdsa_der_parse(); + run_ecdsa_sign_verify(); + run_ecdsa_end_to_end(); + run_ecdsa_edge_cases(); +#ifdef ENABLE_OPENSSL_TESTS + run_ecdsa_openssl(); +#endif + +#ifdef ENABLE_MODULE_RECOVERY + /* ECDSA pubkey recovery tests */ + run_recovery_tests(); +#endif + + secp256k1_rand256(run32); + printf("random run = %02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x\n", run32[0], run32[1], run32[2], run32[3], run32[4], run32[5], run32[6], run32[7], run32[8], run32[9], run32[10], run32[11], run32[12], run32[13], run32[14], run32[15]); + + /* shutdown */ + secp256k1_context_destroy(ctx); + + printf("no problems found\n"); + return 0; +} diff --git a/node_modules/secp256k1/src/secp256k1/src/tests_exhaustive.c b/node_modules/secp256k1/src/secp256k1/src/tests_exhaustive.c new file mode 100644 index 0000000..b44e357 --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/tests_exhaustive.c @@ -0,0 +1,511 @@ +/*********************************************************************** + * Copyright (c) 2016 Andrew Poelstra * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#if defined HAVE_CONFIG_H +#include "libsecp256k1-config.h" +#endif + +#include +#include + +#include + +#undef USE_ECMULT_STATIC_PRECOMPUTATION + +#ifndef EXHAUSTIVE_TEST_ORDER +/* see group_impl.h for allowable values */ +#define EXHAUSTIVE_TEST_ORDER 13 +#define EXHAUSTIVE_TEST_LAMBDA 9 /* cube root of 1 mod 13 */ +#endif + +#include "include/secp256k1.h" +#include "group.h" +#include "secp256k1.c" +#include "testrand_impl.h" + +#ifdef ENABLE_MODULE_RECOVERY +#include "src/modules/recovery/main_impl.h" +#include "include/secp256k1_recovery.h" +#endif + +/** stolen from tests.c */ +void ge_equals_ge(const secp256k1_ge *a, const secp256k1_ge *b) { + CHECK(a->infinity == b->infinity); + if (a->infinity) { + return; + } + CHECK(secp256k1_fe_equal_var(&a->x, &b->x)); + CHECK(secp256k1_fe_equal_var(&a->y, &b->y)); +} + +void ge_equals_gej(const secp256k1_ge *a, const secp256k1_gej *b) { + secp256k1_fe z2s; + secp256k1_fe u1, u2, s1, s2; + CHECK(a->infinity == b->infinity); + if (a->infinity) { + return; + } + /* Check a.x * b.z^2 == b.x && a.y * b.z^3 == b.y, to avoid inverses. */ + secp256k1_fe_sqr(&z2s, &b->z); + secp256k1_fe_mul(&u1, &a->x, &z2s); + u2 = b->x; secp256k1_fe_normalize_weak(&u2); + secp256k1_fe_mul(&s1, &a->y, &z2s); secp256k1_fe_mul(&s1, &s1, &b->z); + s2 = b->y; secp256k1_fe_normalize_weak(&s2); + CHECK(secp256k1_fe_equal_var(&u1, &u2)); + CHECK(secp256k1_fe_equal_var(&s1, &s2)); +} + +void random_fe(secp256k1_fe *x) { + unsigned char bin[32]; + do { + secp256k1_rand256(bin); + if (secp256k1_fe_set_b32(x, bin)) { + return; + } + } while(1); +} +/** END stolen from tests.c */ + +int secp256k1_nonce_function_smallint(unsigned char *nonce32, const unsigned char *msg32, + const unsigned char *key32, const unsigned char *algo16, + void *data, unsigned int attempt) { + secp256k1_scalar s; + int *idata = data; + (void)msg32; + (void)key32; + (void)algo16; + /* Some nonces cannot be used because they'd cause s and/or r to be zero. + * The signing function has retry logic here that just re-calls the nonce + * function with an increased `attempt`. So if attempt > 0 this means we + * need to change the nonce to avoid an infinite loop. */ + if (attempt > 0) { + *idata = (*idata + 1) % EXHAUSTIVE_TEST_ORDER; + } + secp256k1_scalar_set_int(&s, *idata); + secp256k1_scalar_get_b32(nonce32, &s); + return 1; +} + +#ifdef USE_ENDOMORPHISM +void test_exhaustive_endomorphism(const secp256k1_ge *group, int order) { + int i; + for (i = 0; i < order; i++) { + secp256k1_ge res; + secp256k1_ge_mul_lambda(&res, &group[i]); + ge_equals_ge(&group[i * EXHAUSTIVE_TEST_LAMBDA % EXHAUSTIVE_TEST_ORDER], &res); + } +} +#endif + +void test_exhaustive_addition(const secp256k1_ge *group, const secp256k1_gej *groupj, int order) { + int i, j; + + /* Sanity-check (and check infinity functions) */ + CHECK(secp256k1_ge_is_infinity(&group[0])); + CHECK(secp256k1_gej_is_infinity(&groupj[0])); + for (i = 1; i < order; i++) { + CHECK(!secp256k1_ge_is_infinity(&group[i])); + CHECK(!secp256k1_gej_is_infinity(&groupj[i])); + } + + /* Check all addition formulae */ + for (j = 0; j < order; j++) { + secp256k1_fe fe_inv; + secp256k1_fe_inv(&fe_inv, &groupj[j].z); + for (i = 0; i < order; i++) { + secp256k1_ge zless_gej; + secp256k1_gej tmp; + /* add_var */ + secp256k1_gej_add_var(&tmp, &groupj[i], &groupj[j], NULL); + ge_equals_gej(&group[(i + j) % order], &tmp); + /* add_ge */ + if (j > 0) { + secp256k1_gej_add_ge(&tmp, &groupj[i], &group[j]); + ge_equals_gej(&group[(i + j) % order], &tmp); + } + /* add_ge_var */ + secp256k1_gej_add_ge_var(&tmp, &groupj[i], &group[j], NULL); + ge_equals_gej(&group[(i + j) % order], &tmp); + /* add_zinv_var */ + zless_gej.infinity = groupj[j].infinity; + zless_gej.x = groupj[j].x; + zless_gej.y = groupj[j].y; + secp256k1_gej_add_zinv_var(&tmp, &groupj[i], &zless_gej, &fe_inv); + ge_equals_gej(&group[(i + j) % order], &tmp); + } + } + + /* Check doubling */ + for (i = 0; i < order; i++) { + secp256k1_gej tmp; + if (i > 0) { + secp256k1_gej_double_nonzero(&tmp, &groupj[i], NULL); + ge_equals_gej(&group[(2 * i) % order], &tmp); + } + secp256k1_gej_double_var(&tmp, &groupj[i], NULL); + ge_equals_gej(&group[(2 * i) % order], &tmp); + } + + /* Check negation */ + for (i = 1; i < order; i++) { + secp256k1_ge tmp; + secp256k1_gej tmpj; + secp256k1_ge_neg(&tmp, &group[i]); + ge_equals_ge(&group[order - i], &tmp); + secp256k1_gej_neg(&tmpj, &groupj[i]); + ge_equals_gej(&group[order - i], &tmpj); + } +} + +void test_exhaustive_ecmult(const secp256k1_context *ctx, const secp256k1_ge *group, const secp256k1_gej *groupj, int order) { + int i, j, r_log; + for (r_log = 1; r_log < order; r_log++) { + for (j = 0; j < order; j++) { + for (i = 0; i < order; i++) { + secp256k1_gej tmp; + secp256k1_scalar na, ng; + secp256k1_scalar_set_int(&na, i); + secp256k1_scalar_set_int(&ng, j); + + secp256k1_ecmult(&ctx->ecmult_ctx, &tmp, &groupj[r_log], &na, &ng); + ge_equals_gej(&group[(i * r_log + j) % order], &tmp); + + if (i > 0) { + secp256k1_ecmult_const(&tmp, &group[i], &ng, 256); + ge_equals_gej(&group[(i * j) % order], &tmp); + } + } + } + } +} + +typedef struct { + secp256k1_scalar sc[2]; + secp256k1_ge pt[2]; +} ecmult_multi_data; + +static int ecmult_multi_callback(secp256k1_scalar *sc, secp256k1_ge *pt, size_t idx, void *cbdata) { + ecmult_multi_data *data = (ecmult_multi_data*) cbdata; + *sc = data->sc[idx]; + *pt = data->pt[idx]; + return 1; +} + +void test_exhaustive_ecmult_multi(const secp256k1_context *ctx, const secp256k1_ge *group, int order) { + int i, j, k, x, y; + secp256k1_scratch *scratch = secp256k1_scratch_create(&ctx->error_callback, 4096); + for (i = 0; i < order; i++) { + for (j = 0; j < order; j++) { + for (k = 0; k < order; k++) { + for (x = 0; x < order; x++) { + for (y = 0; y < order; y++) { + secp256k1_gej tmp; + secp256k1_scalar g_sc; + ecmult_multi_data data; + + secp256k1_scalar_set_int(&data.sc[0], i); + secp256k1_scalar_set_int(&data.sc[1], j); + secp256k1_scalar_set_int(&g_sc, k); + data.pt[0] = group[x]; + data.pt[1] = group[y]; + + secp256k1_ecmult_multi_var(&ctx->error_callback, &ctx->ecmult_ctx, scratch, &tmp, &g_sc, ecmult_multi_callback, &data, 2); + ge_equals_gej(&group[(i * x + j * y + k) % order], &tmp); + } + } + } + } + } + secp256k1_scratch_destroy(&ctx->error_callback, scratch); +} + +void r_from_k(secp256k1_scalar *r, const secp256k1_ge *group, int k) { + secp256k1_fe x; + unsigned char x_bin[32]; + k %= EXHAUSTIVE_TEST_ORDER; + x = group[k].x; + secp256k1_fe_normalize(&x); + secp256k1_fe_get_b32(x_bin, &x); + secp256k1_scalar_set_b32(r, x_bin, NULL); +} + +void test_exhaustive_verify(const secp256k1_context *ctx, const secp256k1_ge *group, int order) { + int s, r, msg, key; + for (s = 1; s < order; s++) { + for (r = 1; r < order; r++) { + for (msg = 1; msg < order; msg++) { + for (key = 1; key < order; key++) { + secp256k1_ge nonconst_ge; + secp256k1_ecdsa_signature sig; + secp256k1_pubkey pk; + secp256k1_scalar sk_s, msg_s, r_s, s_s; + secp256k1_scalar s_times_k_s, msg_plus_r_times_sk_s; + int k, should_verify; + unsigned char msg32[32]; + + secp256k1_scalar_set_int(&s_s, s); + secp256k1_scalar_set_int(&r_s, r); + secp256k1_scalar_set_int(&msg_s, msg); + secp256k1_scalar_set_int(&sk_s, key); + + /* Verify by hand */ + /* Run through every k value that gives us this r and check that *one* works. + * Note there could be none, there could be multiple, ECDSA is weird. */ + should_verify = 0; + for (k = 0; k < order; k++) { + secp256k1_scalar check_x_s; + r_from_k(&check_x_s, group, k); + if (r_s == check_x_s) { + secp256k1_scalar_set_int(&s_times_k_s, k); + secp256k1_scalar_mul(&s_times_k_s, &s_times_k_s, &s_s); + secp256k1_scalar_mul(&msg_plus_r_times_sk_s, &r_s, &sk_s); + secp256k1_scalar_add(&msg_plus_r_times_sk_s, &msg_plus_r_times_sk_s, &msg_s); + should_verify |= secp256k1_scalar_eq(&s_times_k_s, &msg_plus_r_times_sk_s); + } + } + /* nb we have a "high s" rule */ + should_verify &= !secp256k1_scalar_is_high(&s_s); + + /* Verify by calling verify */ + secp256k1_ecdsa_signature_save(&sig, &r_s, &s_s); + memcpy(&nonconst_ge, &group[sk_s], sizeof(nonconst_ge)); + secp256k1_pubkey_save(&pk, &nonconst_ge); + secp256k1_scalar_get_b32(msg32, &msg_s); + CHECK(should_verify == + secp256k1_ecdsa_verify(ctx, &sig, msg32, &pk)); + } + } + } + } +} + +void test_exhaustive_sign(const secp256k1_context *ctx, const secp256k1_ge *group, int order) { + int i, j, k; + + /* Loop */ + for (i = 1; i < order; i++) { /* message */ + for (j = 1; j < order; j++) { /* key */ + for (k = 1; k < order; k++) { /* nonce */ + const int starting_k = k; + secp256k1_ecdsa_signature sig; + secp256k1_scalar sk, msg, r, s, expected_r; + unsigned char sk32[32], msg32[32]; + secp256k1_scalar_set_int(&msg, i); + secp256k1_scalar_set_int(&sk, j); + secp256k1_scalar_get_b32(sk32, &sk); + secp256k1_scalar_get_b32(msg32, &msg); + + secp256k1_ecdsa_sign(ctx, &sig, msg32, sk32, secp256k1_nonce_function_smallint, &k); + + secp256k1_ecdsa_signature_load(ctx, &r, &s, &sig); + /* Note that we compute expected_r *after* signing -- this is important + * because our nonce-computing function function might change k during + * signing. */ + r_from_k(&expected_r, group, k); + CHECK(r == expected_r); + CHECK((k * s) % order == (i + r * j) % order || + (k * (EXHAUSTIVE_TEST_ORDER - s)) % order == (i + r * j) % order); + + /* Overflow means we've tried every possible nonce */ + if (k < starting_k) { + break; + } + } + } + } + + /* We would like to verify zero-knowledge here by counting how often every + * possible (s, r) tuple appears, but because the group order is larger + * than the field order, when coercing the x-values to scalar values, some + * appear more often than others, so we are actually not zero-knowledge. + * (This effect also appears in the real code, but the difference is on the + * order of 1/2^128th the field order, so the deviation is not useful to a + * computationally bounded attacker.) + */ +} + +#ifdef ENABLE_MODULE_RECOVERY +void test_exhaustive_recovery_sign(const secp256k1_context *ctx, const secp256k1_ge *group, int order) { + int i, j, k; + + /* Loop */ + for (i = 1; i < order; i++) { /* message */ + for (j = 1; j < order; j++) { /* key */ + for (k = 1; k < order; k++) { /* nonce */ + const int starting_k = k; + secp256k1_fe r_dot_y_normalized; + secp256k1_ecdsa_recoverable_signature rsig; + secp256k1_ecdsa_signature sig; + secp256k1_scalar sk, msg, r, s, expected_r; + unsigned char sk32[32], msg32[32]; + int expected_recid; + int recid; + secp256k1_scalar_set_int(&msg, i); + secp256k1_scalar_set_int(&sk, j); + secp256k1_scalar_get_b32(sk32, &sk); + secp256k1_scalar_get_b32(msg32, &msg); + + secp256k1_ecdsa_sign_recoverable(ctx, &rsig, msg32, sk32, secp256k1_nonce_function_smallint, &k); + + /* Check directly */ + secp256k1_ecdsa_recoverable_signature_load(ctx, &r, &s, &recid, &rsig); + r_from_k(&expected_r, group, k); + CHECK(r == expected_r); + CHECK((k * s) % order == (i + r * j) % order || + (k * (EXHAUSTIVE_TEST_ORDER - s)) % order == (i + r * j) % order); + /* In computing the recid, there is an overflow condition that is disabled in + * scalar_low_impl.h `secp256k1_scalar_set_b32` because almost every r.y value + * will exceed the group order, and our signing code always holds out for r + * values that don't overflow, so with a proper overflow check the tests would + * loop indefinitely. */ + r_dot_y_normalized = group[k].y; + secp256k1_fe_normalize(&r_dot_y_normalized); + /* Also the recovery id is flipped depending if we hit the low-s branch */ + if ((k * s) % order == (i + r * j) % order) { + expected_recid = secp256k1_fe_is_odd(&r_dot_y_normalized) ? 1 : 0; + } else { + expected_recid = secp256k1_fe_is_odd(&r_dot_y_normalized) ? 0 : 1; + } + CHECK(recid == expected_recid); + + /* Convert to a standard sig then check */ + secp256k1_ecdsa_recoverable_signature_convert(ctx, &sig, &rsig); + secp256k1_ecdsa_signature_load(ctx, &r, &s, &sig); + /* Note that we compute expected_r *after* signing -- this is important + * because our nonce-computing function function might change k during + * signing. */ + r_from_k(&expected_r, group, k); + CHECK(r == expected_r); + CHECK((k * s) % order == (i + r * j) % order || + (k * (EXHAUSTIVE_TEST_ORDER - s)) % order == (i + r * j) % order); + + /* Overflow means we've tried every possible nonce */ + if (k < starting_k) { + break; + } + } + } + } +} + +void test_exhaustive_recovery_verify(const secp256k1_context *ctx, const secp256k1_ge *group, int order) { + /* This is essentially a copy of test_exhaustive_verify, with recovery added */ + int s, r, msg, key; + for (s = 1; s < order; s++) { + for (r = 1; r < order; r++) { + for (msg = 1; msg < order; msg++) { + for (key = 1; key < order; key++) { + secp256k1_ge nonconst_ge; + secp256k1_ecdsa_recoverable_signature rsig; + secp256k1_ecdsa_signature sig; + secp256k1_pubkey pk; + secp256k1_scalar sk_s, msg_s, r_s, s_s; + secp256k1_scalar s_times_k_s, msg_plus_r_times_sk_s; + int recid = 0; + int k, should_verify; + unsigned char msg32[32]; + + secp256k1_scalar_set_int(&s_s, s); + secp256k1_scalar_set_int(&r_s, r); + secp256k1_scalar_set_int(&msg_s, msg); + secp256k1_scalar_set_int(&sk_s, key); + secp256k1_scalar_get_b32(msg32, &msg_s); + + /* Verify by hand */ + /* Run through every k value that gives us this r and check that *one* works. + * Note there could be none, there could be multiple, ECDSA is weird. */ + should_verify = 0; + for (k = 0; k < order; k++) { + secp256k1_scalar check_x_s; + r_from_k(&check_x_s, group, k); + if (r_s == check_x_s) { + secp256k1_scalar_set_int(&s_times_k_s, k); + secp256k1_scalar_mul(&s_times_k_s, &s_times_k_s, &s_s); + secp256k1_scalar_mul(&msg_plus_r_times_sk_s, &r_s, &sk_s); + secp256k1_scalar_add(&msg_plus_r_times_sk_s, &msg_plus_r_times_sk_s, &msg_s); + should_verify |= secp256k1_scalar_eq(&s_times_k_s, &msg_plus_r_times_sk_s); + } + } + /* nb we have a "high s" rule */ + should_verify &= !secp256k1_scalar_is_high(&s_s); + + /* We would like to try recovering the pubkey and checking that it matches, + * but pubkey recovery is impossible in the exhaustive tests (the reason + * being that there are 12 nonzero r values, 12 nonzero points, and no + * overlap between the sets, so there are no valid signatures). */ + + /* Verify by converting to a standard signature and calling verify */ + secp256k1_ecdsa_recoverable_signature_save(&rsig, &r_s, &s_s, recid); + secp256k1_ecdsa_recoverable_signature_convert(ctx, &sig, &rsig); + memcpy(&nonconst_ge, &group[sk_s], sizeof(nonconst_ge)); + secp256k1_pubkey_save(&pk, &nonconst_ge); + CHECK(should_verify == + secp256k1_ecdsa_verify(ctx, &sig, msg32, &pk)); + } + } + } + } +} +#endif + +int main(void) { + int i; + secp256k1_gej groupj[EXHAUSTIVE_TEST_ORDER]; + secp256k1_ge group[EXHAUSTIVE_TEST_ORDER]; + + /* Build context */ + secp256k1_context *ctx = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY); + + /* TODO set z = 1, then do num_tests runs with random z values */ + + /* Generate the entire group */ + secp256k1_gej_set_infinity(&groupj[0]); + secp256k1_ge_set_gej(&group[0], &groupj[0]); + for (i = 1; i < EXHAUSTIVE_TEST_ORDER; i++) { + /* Set a different random z-value for each Jacobian point */ + secp256k1_fe z; + random_fe(&z); + + secp256k1_gej_add_ge(&groupj[i], &groupj[i - 1], &secp256k1_ge_const_g); + secp256k1_ge_set_gej(&group[i], &groupj[i]); + secp256k1_gej_rescale(&groupj[i], &z); + + /* Verify against ecmult_gen */ + { + secp256k1_scalar scalar_i; + secp256k1_gej generatedj; + secp256k1_ge generated; + + secp256k1_scalar_set_int(&scalar_i, i); + secp256k1_ecmult_gen(&ctx->ecmult_gen_ctx, &generatedj, &scalar_i); + secp256k1_ge_set_gej(&generated, &generatedj); + + CHECK(group[i].infinity == 0); + CHECK(generated.infinity == 0); + CHECK(secp256k1_fe_equal_var(&generated.x, &group[i].x)); + CHECK(secp256k1_fe_equal_var(&generated.y, &group[i].y)); + } + } + + /* Run the tests */ +#ifdef USE_ENDOMORPHISM + test_exhaustive_endomorphism(group, EXHAUSTIVE_TEST_ORDER); +#endif + test_exhaustive_addition(group, groupj, EXHAUSTIVE_TEST_ORDER); + test_exhaustive_ecmult(ctx, group, groupj, EXHAUSTIVE_TEST_ORDER); + test_exhaustive_ecmult_multi(ctx, group, EXHAUSTIVE_TEST_ORDER); + test_exhaustive_sign(ctx, group, EXHAUSTIVE_TEST_ORDER); + test_exhaustive_verify(ctx, group, EXHAUSTIVE_TEST_ORDER); + +#ifdef ENABLE_MODULE_RECOVERY + test_exhaustive_recovery_sign(ctx, group, EXHAUSTIVE_TEST_ORDER); + test_exhaustive_recovery_verify(ctx, group, EXHAUSTIVE_TEST_ORDER); +#endif + + secp256k1_context_destroy(ctx); + return 0; +} + diff --git a/node_modules/secp256k1/src/secp256k1/src/util.h b/node_modules/secp256k1/src/secp256k1/src/util.h new file mode 100644 index 0000000..9deb61b --- /dev/null +++ b/node_modules/secp256k1/src/secp256k1/src/util.h @@ -0,0 +1,162 @@ +/********************************************************************** + * Copyright (c) 2013, 2014 Pieter Wuille * + * Distributed under the MIT software license, see the accompanying * + * file COPYING or http://www.opensource.org/licenses/mit-license.php.* + **********************************************************************/ + +#ifndef SECP256K1_UTIL_H +#define SECP256K1_UTIL_H + +#if defined HAVE_CONFIG_H +#include "libsecp256k1-config.h" +#endif + +#include +#include +#include + +typedef struct { + void (*fn)(const char *text, void* data); + const void* data; +} secp256k1_callback; + +static SECP256K1_INLINE void secp256k1_callback_call(const secp256k1_callback * const cb, const char * const text) { + cb->fn(text, (void*)cb->data); +} + +#ifdef DETERMINISTIC +#define TEST_FAILURE(msg) do { \ + fprintf(stderr, "%s\n", msg); \ + abort(); \ +} while(0); +#else +#define TEST_FAILURE(msg) do { \ + fprintf(stderr, "%s:%d: %s\n", __FILE__, __LINE__, msg); \ + abort(); \ +} while(0) +#endif + +#if SECP256K1_GNUC_PREREQ(3, 0) +#define EXPECT(x,c) __builtin_expect((x),(c)) +#else +#define EXPECT(x,c) (x) +#endif + +#ifdef DETERMINISTIC +#define CHECK(cond) do { \ + if (EXPECT(!(cond), 0)) { \ + TEST_FAILURE("test condition failed"); \ + } \ +} while(0) +#else +#define CHECK(cond) do { \ + if (EXPECT(!(cond), 0)) { \ + TEST_FAILURE("test condition failed: " #cond); \ + } \ +} while(0) +#endif + +/* Like assert(), but when VERIFY is defined, and side-effect safe. */ +#if defined(COVERAGE) +#define VERIFY_CHECK(check) +#define VERIFY_SETUP(stmt) +#elif defined(VERIFY) +#define VERIFY_CHECK CHECK +#define VERIFY_SETUP(stmt) do { stmt; } while(0) +#else +#define VERIFY_CHECK(cond) do { (void)(cond); } while(0) +#define VERIFY_SETUP(stmt) +#endif + +static SECP256K1_INLINE void *checked_malloc(const secp256k1_callback* cb, size_t size) { + void *ret = malloc(size); + if (ret == NULL) { + secp256k1_callback_call(cb, "Out of memory"); + } + return ret; +} + +static SECP256K1_INLINE void *checked_realloc(const secp256k1_callback* cb, void *ptr, size_t size) { + void *ret = realloc(ptr, size); + if (ret == NULL) { + secp256k1_callback_call(cb, "Out of memory"); + } + return ret; +} + +#if defined(__BIGGEST_ALIGNMENT__) +#define ALIGNMENT __BIGGEST_ALIGNMENT__ +#else +/* Using 16 bytes alignment because common architectures never have alignment + * requirements above 8 for any of the types we care about. In addition we + * leave some room because currently we don't care about a few bytes. */ +#define ALIGNMENT 16 +#endif + +#define ROUND_TO_ALIGN(size) (((size + ALIGNMENT - 1) / ALIGNMENT) * ALIGNMENT) + +/* Assume there is a contiguous memory object with bounds [base, base + max_size) + * of which the memory range [base, *prealloc_ptr) is already allocated for usage, + * where *prealloc_ptr is an aligned pointer. In that setting, this functions + * reserves the subobject [*prealloc_ptr, *prealloc_ptr + alloc_size) of + * alloc_size bytes by increasing *prealloc_ptr accordingly, taking into account + * alignment requirements. + * + * The function returns an aligned pointer to the newly allocated subobject. + * + * This is useful for manual memory management: if we're simply given a block + * [base, base + max_size), the caller can use this function to allocate memory + * in this block and keep track of the current allocation state with *prealloc_ptr. + * + * It is VERIFY_CHECKed that there is enough space left in the memory object and + * *prealloc_ptr is aligned relative to base. + */ +static SECP256K1_INLINE void *manual_alloc(void** prealloc_ptr, size_t alloc_size, void* base, size_t max_size) { + size_t aligned_alloc_size = ROUND_TO_ALIGN(alloc_size); + void* ret; + VERIFY_CHECK(prealloc_ptr != NULL); + VERIFY_CHECK(*prealloc_ptr != NULL); + VERIFY_CHECK(base != NULL); + VERIFY_CHECK((unsigned char*)*prealloc_ptr >= (unsigned char*)base); + VERIFY_CHECK(((unsigned char*)*prealloc_ptr - (unsigned char*)base) % ALIGNMENT == 0); + VERIFY_CHECK((unsigned char*)*prealloc_ptr - (unsigned char*)base + aligned_alloc_size <= max_size); + ret = *prealloc_ptr; + *((unsigned char**)prealloc_ptr) += aligned_alloc_size; + return ret; +} + +/* Macro for restrict, when available and not in a VERIFY build. */ +#if defined(SECP256K1_BUILD) && defined(VERIFY) +# define SECP256K1_RESTRICT +#else +# if (!defined(__STDC_VERSION__) || (__STDC_VERSION__ < 199901L) ) +# if SECP256K1_GNUC_PREREQ(3,0) +# define SECP256K1_RESTRICT __restrict__ +# elif (defined(_MSC_VER) && _MSC_VER >= 1400) +# define SECP256K1_RESTRICT __restrict +# else +# define SECP256K1_RESTRICT +# endif +# else +# define SECP256K1_RESTRICT restrict +# endif +#endif + +#if defined(_WIN32) +# define I64FORMAT "I64d" +# define I64uFORMAT "I64u" +#else +# define I64FORMAT "lld" +# define I64uFORMAT "llu" +#endif + +#if defined(HAVE___INT128) +# if defined(__GNUC__) +# define SECP256K1_GNUC_EXT __extension__ +# else +# define SECP256K1_GNUC_EXT +# endif +SECP256K1_GNUC_EXT typedef unsigned __int128 uint128_t; +#endif + +#endif /* SECP256K1_UTIL_H */ diff --git a/node_modules/string_decoder/LICENSE b/node_modules/string_decoder/LICENSE new file mode 100644 index 0000000..778edb2 --- /dev/null +++ b/node_modules/string_decoder/LICENSE @@ -0,0 +1,48 @@ +Node.js is licensed for use as follows: + +""" +Copyright Node.js contributors. All rights reserved. + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to +deal in the Software without restriction, including without limitation the +rights to use, copy, modify, merge, publish, distribute, sublicense, and/or +sell copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS +IN THE SOFTWARE. +""" + +This license applies to parts of Node.js originating from the +https://github.com/joyent/node repository: + +""" +Copyright Joyent, Inc. and other Node contributors. All rights reserved. +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to +deal in the Software without restriction, including without limitation the +rights to use, copy, modify, merge, publish, distribute, sublicense, and/or +sell copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS +IN THE SOFTWARE. +""" + diff --git a/node_modules/string_decoder/README.md b/node_modules/string_decoder/README.md new file mode 100644 index 0000000..5fd5831 --- /dev/null +++ b/node_modules/string_decoder/README.md @@ -0,0 +1,47 @@ +# string_decoder + +***Node-core v8.9.4 string_decoder for userland*** + + +[![NPM](https://nodei.co/npm/string_decoder.png?downloads=true&downloadRank=true)](https://nodei.co/npm/string_decoder/) +[![NPM](https://nodei.co/npm-dl/string_decoder.png?&months=6&height=3)](https://nodei.co/npm/string_decoder/) + + +```bash +npm install --save string_decoder +``` + +***Node-core string_decoder for userland*** + +This package is a mirror of the string_decoder implementation in Node-core. + +Full documentation may be found on the [Node.js website](https://nodejs.org/dist/v8.9.4/docs/api/). + +As of version 1.0.0 **string_decoder** uses semantic versioning. + +## Previous versions + +Previous version numbers match the versions found in Node core, e.g. 0.10.24 matches Node 0.10.24, likewise 0.11.10 matches Node 0.11.10. + +## Update + +The *build/* directory contains a build script that will scrape the source from the [nodejs/node](https://github.com/nodejs/node) repo given a specific Node version. + +## Streams Working Group + +`string_decoder` is maintained by the Streams Working Group, which +oversees the development and maintenance of the Streams API within +Node.js. The responsibilities of the Streams Working Group include: + +* Addressing stream issues on the Node.js issue tracker. +* Authoring and editing stream documentation within the Node.js project. +* Reviewing changes to stream subclasses within the Node.js project. +* Redirecting changes to streams from the Node.js project to this + project. +* Assisting in the implementation of stream providers within Node.js. +* Recommending versions of `readable-stream` to be included in Node.js. +* Messaging about the future of streams to give the community advance + notice of changes. + +See [readable-stream](https://github.com/nodejs/readable-stream) for +more details. diff --git a/node_modules/string_decoder/lib/string_decoder.js b/node_modules/string_decoder/lib/string_decoder.js new file mode 100644 index 0000000..2e89e63 --- /dev/null +++ b/node_modules/string_decoder/lib/string_decoder.js @@ -0,0 +1,296 @@ +// Copyright Joyent, Inc. and other Node contributors. +// +// Permission is hereby granted, free of charge, to any person obtaining a +// copy of this software and associated documentation files (the +// "Software"), to deal in the Software without restriction, including +// without limitation the rights to use, copy, modify, merge, publish, +// distribute, sublicense, and/or sell copies of the Software, and to permit +// persons to whom the Software is furnished to do so, subject to the +// following conditions: +// +// The above copyright notice and this permission notice shall be included +// in all copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN +// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, +// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR +// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE +// USE OR OTHER DEALINGS IN THE SOFTWARE. + +'use strict'; + +/**/ + +var Buffer = require('safe-buffer').Buffer; +/**/ + +var isEncoding = Buffer.isEncoding || function (encoding) { + encoding = '' + encoding; + switch (encoding && encoding.toLowerCase()) { + case 'hex':case 'utf8':case 'utf-8':case 'ascii':case 'binary':case 'base64':case 'ucs2':case 'ucs-2':case 'utf16le':case 'utf-16le':case 'raw': + return true; + default: + return false; + } +}; + +function _normalizeEncoding(enc) { + if (!enc) return 'utf8'; + var retried; + while (true) { + switch (enc) { + case 'utf8': + case 'utf-8': + return 'utf8'; + case 'ucs2': + case 'ucs-2': + case 'utf16le': + case 'utf-16le': + return 'utf16le'; + case 'latin1': + case 'binary': + return 'latin1'; + case 'base64': + case 'ascii': + case 'hex': + return enc; + default: + if (retried) return; // undefined + enc = ('' + enc).toLowerCase(); + retried = true; + } + } +}; + +// Do not cache `Buffer.isEncoding` when checking encoding names as some +// modules monkey-patch it to support additional encodings +function normalizeEncoding(enc) { + var nenc = _normalizeEncoding(enc); + if (typeof nenc !== 'string' && (Buffer.isEncoding === isEncoding || !isEncoding(enc))) throw new Error('Unknown encoding: ' + enc); + return nenc || enc; +} + +// StringDecoder provides an interface for efficiently splitting a series of +// buffers into a series of JS strings without breaking apart multi-byte +// characters. +exports.StringDecoder = StringDecoder; +function StringDecoder(encoding) { + this.encoding = normalizeEncoding(encoding); + var nb; + switch (this.encoding) { + case 'utf16le': + this.text = utf16Text; + this.end = utf16End; + nb = 4; + break; + case 'utf8': + this.fillLast = utf8FillLast; + nb = 4; + break; + case 'base64': + this.text = base64Text; + this.end = base64End; + nb = 3; + break; + default: + this.write = simpleWrite; + this.end = simpleEnd; + return; + } + this.lastNeed = 0; + this.lastTotal = 0; + this.lastChar = Buffer.allocUnsafe(nb); +} + +StringDecoder.prototype.write = function (buf) { + if (buf.length === 0) return ''; + var r; + var i; + if (this.lastNeed) { + r = this.fillLast(buf); + if (r === undefined) return ''; + i = this.lastNeed; + this.lastNeed = 0; + } else { + i = 0; + } + if (i < buf.length) return r ? r + this.text(buf, i) : this.text(buf, i); + return r || ''; +}; + +StringDecoder.prototype.end = utf8End; + +// Returns only complete characters in a Buffer +StringDecoder.prototype.text = utf8Text; + +// Attempts to complete a partial non-UTF-8 character using bytes from a Buffer +StringDecoder.prototype.fillLast = function (buf) { + if (this.lastNeed <= buf.length) { + buf.copy(this.lastChar, this.lastTotal - this.lastNeed, 0, this.lastNeed); + return this.lastChar.toString(this.encoding, 0, this.lastTotal); + } + buf.copy(this.lastChar, this.lastTotal - this.lastNeed, 0, buf.length); + this.lastNeed -= buf.length; +}; + +// Checks the type of a UTF-8 byte, whether it's ASCII, a leading byte, or a +// continuation byte. If an invalid byte is detected, -2 is returned. +function utf8CheckByte(byte) { + if (byte <= 0x7F) return 0;else if (byte >> 5 === 0x06) return 2;else if (byte >> 4 === 0x0E) return 3;else if (byte >> 3 === 0x1E) return 4; + return byte >> 6 === 0x02 ? -1 : -2; +} + +// Checks at most 3 bytes at the end of a Buffer in order to detect an +// incomplete multi-byte UTF-8 character. The total number of bytes (2, 3, or 4) +// needed to complete the UTF-8 character (if applicable) are returned. +function utf8CheckIncomplete(self, buf, i) { + var j = buf.length - 1; + if (j < i) return 0; + var nb = utf8CheckByte(buf[j]); + if (nb >= 0) { + if (nb > 0) self.lastNeed = nb - 1; + return nb; + } + if (--j < i || nb === -2) return 0; + nb = utf8CheckByte(buf[j]); + if (nb >= 0) { + if (nb > 0) self.lastNeed = nb - 2; + return nb; + } + if (--j < i || nb === -2) return 0; + nb = utf8CheckByte(buf[j]); + if (nb >= 0) { + if (nb > 0) { + if (nb === 2) nb = 0;else self.lastNeed = nb - 3; + } + return nb; + } + return 0; +} + +// Validates as many continuation bytes for a multi-byte UTF-8 character as +// needed or are available. If we see a non-continuation byte where we expect +// one, we "replace" the validated continuation bytes we've seen so far with +// a single UTF-8 replacement character ('\ufffd'), to match v8's UTF-8 decoding +// behavior. The continuation byte check is included three times in the case +// where all of the continuation bytes for a character exist in the same buffer. +// It is also done this way as a slight performance increase instead of using a +// loop. +function utf8CheckExtraBytes(self, buf, p) { + if ((buf[0] & 0xC0) !== 0x80) { + self.lastNeed = 0; + return '\ufffd'; + } + if (self.lastNeed > 1 && buf.length > 1) { + if ((buf[1] & 0xC0) !== 0x80) { + self.lastNeed = 1; + return '\ufffd'; + } + if (self.lastNeed > 2 && buf.length > 2) { + if ((buf[2] & 0xC0) !== 0x80) { + self.lastNeed = 2; + return '\ufffd'; + } + } + } +} + +// Attempts to complete a multi-byte UTF-8 character using bytes from a Buffer. +function utf8FillLast(buf) { + var p = this.lastTotal - this.lastNeed; + var r = utf8CheckExtraBytes(this, buf, p); + if (r !== undefined) return r; + if (this.lastNeed <= buf.length) { + buf.copy(this.lastChar, p, 0, this.lastNeed); + return this.lastChar.toString(this.encoding, 0, this.lastTotal); + } + buf.copy(this.lastChar, p, 0, buf.length); + this.lastNeed -= buf.length; +} + +// Returns all complete UTF-8 characters in a Buffer. If the Buffer ended on a +// partial character, the character's bytes are buffered until the required +// number of bytes are available. +function utf8Text(buf, i) { + var total = utf8CheckIncomplete(this, buf, i); + if (!this.lastNeed) return buf.toString('utf8', i); + this.lastTotal = total; + var end = buf.length - (total - this.lastNeed); + buf.copy(this.lastChar, 0, end); + return buf.toString('utf8', i, end); +} + +// For UTF-8, a replacement character is added when ending on a partial +// character. +function utf8End(buf) { + var r = buf && buf.length ? this.write(buf) : ''; + if (this.lastNeed) return r + '\ufffd'; + return r; +} + +// UTF-16LE typically needs two bytes per character, but even if we have an even +// number of bytes available, we need to check if we end on a leading/high +// surrogate. In that case, we need to wait for the next two bytes in order to +// decode the last character properly. +function utf16Text(buf, i) { + if ((buf.length - i) % 2 === 0) { + var r = buf.toString('utf16le', i); + if (r) { + var c = r.charCodeAt(r.length - 1); + if (c >= 0xD800 && c <= 0xDBFF) { + this.lastNeed = 2; + this.lastTotal = 4; + this.lastChar[0] = buf[buf.length - 2]; + this.lastChar[1] = buf[buf.length - 1]; + return r.slice(0, -1); + } + } + return r; + } + this.lastNeed = 1; + this.lastTotal = 2; + this.lastChar[0] = buf[buf.length - 1]; + return buf.toString('utf16le', i, buf.length - 1); +} + +// For UTF-16LE we do not explicitly append special replacement characters if we +// end on a partial character, we simply let v8 handle that. +function utf16End(buf) { + var r = buf && buf.length ? this.write(buf) : ''; + if (this.lastNeed) { + var end = this.lastTotal - this.lastNeed; + return r + this.lastChar.toString('utf16le', 0, end); + } + return r; +} + +function base64Text(buf, i) { + var n = (buf.length - i) % 3; + if (n === 0) return buf.toString('base64', i); + this.lastNeed = 3 - n; + this.lastTotal = 3; + if (n === 1) { + this.lastChar[0] = buf[buf.length - 1]; + } else { + this.lastChar[0] = buf[buf.length - 2]; + this.lastChar[1] = buf[buf.length - 1]; + } + return buf.toString('base64', i, buf.length - n); +} + +function base64End(buf) { + var r = buf && buf.length ? this.write(buf) : ''; + if (this.lastNeed) return r + this.lastChar.toString('base64', 0, 3 - this.lastNeed); + return r; +} + +// Pass bytes on through for single-byte encodings (e.g. ascii, latin1, hex) +function simpleWrite(buf) { + return buf.toString(this.encoding); +} + +function simpleEnd(buf) { + return buf && buf.length ? this.write(buf) : ''; +} \ No newline at end of file diff --git a/node_modules/string_decoder/package.json b/node_modules/string_decoder/package.json new file mode 100644 index 0000000..b2bb141 --- /dev/null +++ b/node_modules/string_decoder/package.json @@ -0,0 +1,34 @@ +{ + "name": "string_decoder", + "version": "1.3.0", + "description": "The string_decoder module from Node core", + "main": "lib/string_decoder.js", + "files": [ + "lib" + ], + "dependencies": { + "safe-buffer": "~5.2.0" + }, + "devDependencies": { + "babel-polyfill": "^6.23.0", + "core-util-is": "^1.0.2", + "inherits": "^2.0.3", + "tap": "~0.4.8" + }, + "scripts": { + "test": "tap test/parallel/*.js && node test/verify-dependencies", + "ci": "tap test/parallel/*.js test/ours/*.js --tap | tee test.tap && node test/verify-dependencies.js" + }, + "repository": { + "type": "git", + "url": "git://github.com/nodejs/string_decoder.git" + }, + "homepage": "https://github.com/nodejs/string_decoder", + "keywords": [ + "string", + "decoder", + "browser", + "browserify" + ], + "license": "MIT" +} diff --git a/node_modules/util-deprecate/History.md b/node_modules/util-deprecate/History.md new file mode 100644 index 0000000..acc8675 --- /dev/null +++ b/node_modules/util-deprecate/History.md @@ -0,0 +1,16 @@ + +1.0.2 / 2015-10-07 +================== + + * use try/catch when checking `localStorage` (#3, @kumavis) + +1.0.1 / 2014-11-25 +================== + + * browser: use `console.warn()` for deprecation calls + * browser: more jsdocs + +1.0.0 / 2014-04-30 +================== + + * initial commit diff --git a/node_modules/util-deprecate/LICENSE b/node_modules/util-deprecate/LICENSE new file mode 100644 index 0000000..6a60e8c --- /dev/null +++ b/node_modules/util-deprecate/LICENSE @@ -0,0 +1,24 @@ +(The MIT License) + +Copyright (c) 2014 Nathan Rajlich + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/node_modules/util-deprecate/README.md b/node_modules/util-deprecate/README.md new file mode 100644 index 0000000..75622fa --- /dev/null +++ b/node_modules/util-deprecate/README.md @@ -0,0 +1,53 @@ +util-deprecate +============== +### The Node.js `util.deprecate()` function with browser support + +In Node.js, this module simply re-exports the `util.deprecate()` function. + +In the web browser (i.e. via browserify), a browser-specific implementation +of the `util.deprecate()` function is used. + + +## API + +A `deprecate()` function is the only thing exposed by this module. + +``` javascript +// setup: +exports.foo = deprecate(foo, 'foo() is deprecated, use bar() instead'); + + +// users see: +foo(); +// foo() is deprecated, use bar() instead +foo(); +foo(); +``` + + +## License + +(The MIT License) + +Copyright (c) 2014 Nathan Rajlich + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/node_modules/util-deprecate/browser.js b/node_modules/util-deprecate/browser.js new file mode 100644 index 0000000..549ae2f --- /dev/null +++ b/node_modules/util-deprecate/browser.js @@ -0,0 +1,67 @@ + +/** + * Module exports. + */ + +module.exports = deprecate; + +/** + * Mark that a method should not be used. + * Returns a modified function which warns once by default. + * + * If `localStorage.noDeprecation = true` is set, then it is a no-op. + * + * If `localStorage.throwDeprecation = true` is set, then deprecated functions + * will throw an Error when invoked. + * + * If `localStorage.traceDeprecation = true` is set, then deprecated functions + * will invoke `console.trace()` instead of `console.error()`. + * + * @param {Function} fn - the function to deprecate + * @param {String} msg - the string to print to the console when `fn` is invoked + * @returns {Function} a new "deprecated" version of `fn` + * @api public + */ + +function deprecate (fn, msg) { + if (config('noDeprecation')) { + return fn; + } + + var warned = false; + function deprecated() { + if (!warned) { + if (config('throwDeprecation')) { + throw new Error(msg); + } else if (config('traceDeprecation')) { + console.trace(msg); + } else { + console.warn(msg); + } + warned = true; + } + return fn.apply(this, arguments); + } + + return deprecated; +} + +/** + * Checks `localStorage` for boolean values for the given `name`. + * + * @param {String} name + * @returns {Boolean} + * @api private + */ + +function config (name) { + // accessing global.localStorage can trigger a DOMException in sandboxed iframes + try { + if (!global.localStorage) return false; + } catch (_) { + return false; + } + var val = global.localStorage[name]; + if (null == val) return false; + return String(val).toLowerCase() === 'true'; +} diff --git a/node_modules/util-deprecate/node.js b/node_modules/util-deprecate/node.js new file mode 100644 index 0000000..5e6fcff --- /dev/null +++ b/node_modules/util-deprecate/node.js @@ -0,0 +1,6 @@ + +/** + * For Node.js, simply re-export the core `util.deprecate` function. + */ + +module.exports = require('util').deprecate; diff --git a/node_modules/util-deprecate/package.json b/node_modules/util-deprecate/package.json new file mode 100644 index 0000000..2e79f89 --- /dev/null +++ b/node_modules/util-deprecate/package.json @@ -0,0 +1,27 @@ +{ + "name": "util-deprecate", + "version": "1.0.2", + "description": "The Node.js `util.deprecate()` function with browser support", + "main": "node.js", + "browser": "browser.js", + "scripts": { + "test": "echo \"Error: no test specified\" && exit 1" + }, + "repository": { + "type": "git", + "url": "git://github.com/TooTallNate/util-deprecate.git" + }, + "keywords": [ + "util", + "deprecate", + "browserify", + "browser", + "node" + ], + "author": "Nathan Rajlich (http://n8.io/)", + "license": "MIT", + "bugs": { + "url": "https://github.com/TooTallNate/util-deprecate/issues" + }, + "homepage": "https://github.com/TooTallNate/util-deprecate" +} diff --git a/output.txt b/output.txt new file mode 100644 index 0000000..8483110 --- /dev/null +++ b/output.txt @@ -0,0 +1,5293 @@ +040000000000000000000000000000000000000000000000000000000000000000000000432ba63157f627a9c0599ace84fd287cf68da59f1bfd1c4692071ef3ef210e67fd911d66ffff001f05000000 +010000000001010000000000000000000000000000000000000000000000000000000000000000ffffffff1802d4ce0c486579206974277320406D697450726173616420ffffffff02ef85b98b000000001976a914edf10a7fac6b32e24daa5305c723f3de58db1bc888ac0000000000000000266a24aa21a9ed5aa51ae5a09bd1f63f565c0ba4f30b994c9c09c8b6f82210400497b47f42f9de0120000000000000000000000000000000000000000000000000000000000000000000000000 +23f6ae5049268f63e473c5314b58a1f2d4a0d4fafd8c8ccf8efe06974ad4e077 +f41c60ea16b21f696611aab3a62552037fbe95012ac4a459a6c19c2c23cd3222 +6494cdb062c4a1bc3e74760f9fd389c69c9cd0dcba246c0c58ab37688671edbd +ede7be5d4753a25139d6b6ff033cbdf0e43b729733cb09a6a3b44472ad27bdab +29ce2496811d9ec9b5c1bac7664a08e63505e29f1e7a1ab72968296117b20957 +3c6a04c31a65fb69f92158f5a1d94d848130bf206cb2b2a4f98059d724b38c7b +9466e7a4264414d732858f94df6964af60e9300eb62e47ceed322dd1c7759fad +d3aecfcd7dbf9e68b86159ee3db909821d23170befc8405a9edf2af9933122e5 +b27bcbeae0d3682814c3f308d66d77da0ce24c2a41c130f39a5768d30b64f5b6 +44e4868052fc4693438774c202d9e05729f41201729590a25ffbffc81bf825c9 +6af837a8b73ea1b5422680c5f5db6517460cfffa2ad4e81718654d5f3e875144 +75633ad93688b75d05e60de48956a0385f67b8d0a276b16a121783b7284aa832 +7c4e6413d905a5d3db5da4adabbd17e826ea98c3c7c907d14668bdc9698e28bc +3e281890426350ed7ba18d327586d9d1e9060105453862b31c62c6e7aee1e42d +94c3a0c2aa97fe8a9ec6d542f02dd31849c1bdcc0b65aee644c7a69d8cda87ce +113a664a4e8c737a2faec3a3c8d7ca0c000b775217c2145555f589f23b0701c5 +b4c54f14ea1d6755ea40dbb2d9c6e1dfa151ce80c17d1350c74aa05bbd7d94e1 +96ccf99eda56ded7a74e6a121ac28a5474efd5967681229f9f2275621cda99d5 +fed864de018531a2a256ea3e70335887cdf910ea7e6adbf50d9e74546d900770 +565fb24034a0a15cfd1d7ab1ddf6621f34701bc3cf6d5a2a58a3ac852dd61b92 +ea57b6b06cf20acfe8dc444cb8a5684efb0a2699ce91f213fdf85784d8d9f9f8 +01dfaef60604db4eedc66e4414d5c9da0014319a2b2b7442dbd2e2f3a6f40dd1 +3d4308c35827e01e087d212fb538df11db0453a818ebe196c52ab15ee3cb4634 +713cfb7e01dfccd2075553b3f786ad19e6b5d3d13ba47f0d800b290749a526c9 +30848a4d556d513b19459c96244b26d947fc9e09d25c3719a3e40c2f22213aaa +054e776502b4aa4cb46bd049fda6776187ddd5222a9ef6ce30816af197e949b4 +5f2551bc1d21afccc5ca0c2e901dd5915ddfb39e0a10a51267d087a9bb5126ac +66ec6b2bc37c2129740c2e611ed61ff144919ff89b34b976166b1fd9bf1f92a9 +261bb9755e4299510e74b033ea9d2c4484dd200a49681e0fe12875cb9e0f654f +0f1d3efdd07f9feead7b2015b02d76ba6f0d61d66173507f76d986291b2543f6 +e8010d7db50bd68acbdb5abf3d2ea02319d8f81498b5a2c899125513cc223171 +cc6e51d15a89ef50623b7e8cd8503472b497468f86f146357c41b3dc731e85bd +32271a0e0ba74a715cef2511e927e9945ef1a2757d0655130e5114949c94b205 +e51611d99fc256b3159287178e24fe79655f60592c2cf7a0bb867235f77a3f22 +562020c589b13373527aa059a1f176bd54a0b2187d0aa7007b41d97f99facb96 +607e8ac9f62647ee9d2bff659c1e08542aa99eec29a2bfc031215d31f37208f5 +1aef558645f6c9de23619ed3bc55fa93238360e2ac3aae0f22937fdf0a07c576 +2b381f25a8fb3e6f4be79cd7777f0aee1bf4f89787d8622375c7686e962971d2 +f1e653cde750a5a6fcaee3543e97f2de221d3a66f51024fc7831afb3de635ead +2d80fdcc1105546c6abd4a1bf6ff9bbf7b6735a57c47026ac54ba576477ccdac +4928b35d4a4375407d4984352ac715787a026bd60133644e9a4a496e9e17a73e +d7109a6d7dc31c66645aab1f196f7f443c551ecf84e5d779bf7960013d2e001e +501d7d02be0a0e7d61c4a88f4fc592f4741ef764f248fba354bd5fa7c3715943 +da419f20fbca14290ec1354efe22493fac7be8b804e73f66ac8eada3dcccfa73 +fe976f5cc884a042ef595778d344eeb0480eb8519b712f12c169d992e7ef6b46 +6a00d75b34368c65711afd5a979bbc669577ac49f8885156d1f07e2c6bfa116b +a06bcf91beee19d07ec3e3a96279924b58951f25827514b2e9bbebf807149891 +4df509521a5f585245ac7b63f60be24ff91617c89bf6986111d447b563b34ff2 +1e2219a02d4ab33bf2278b5109e007a06049a6bf9301a5efc9d6655b2c89fc5b +fe28fa2fe28f15e55b93311c2bda8843e0ac6dee81afbf2eafef57b51601edd0 +4138d633105a5c05a683e44762de2b4c7228aa08fb44ea199b31a002816d7afa +c4da5fecb9ee797031869cfd18c85032de50c1927f03bc4e3a9effbb2c25627a +157ba593e460784fffab72c644eb11eddf851f7a1734c1f42258e2924f3cf5e6 +e6c4f35ce3bdf932519de9c9056b196fa84d1aa9860824b177e89a9010b04ea0 +d015e3f97284bc217eddee377e5ab47100a7fbd13a66fdd1fcb9acfc0454f35e +fd634363ae6eef2d18ac8b15d4e09e9cfe42d75bc484417a5910ad636cb36d0c +78699d0d5143ebd1d652501c5ccc2e92c1d839e827311c9c8843e33eabf5f499 +d4c64b4503e737de8d7a84156a3f6811cd40bd80f1173a2691193ff333039aca +00c4a55f63c44d2e8916cf786988cb22b7cc5630ef25cd571c8d094acd0f5d97 +46c0d97acc09e143d657af6e9b6f19ab32fc1d6b6c385a2e3c483710dbad0d17 +b5d3c7844be6ba6e3743404617074e8f36022d1cad3984a19f3f86bfa5f5873b +331c77387f77ef84eefe9cfb1364986812950262a63753bc38b56788914447dc +c02edd9072f7a2638e66a6e39aeb3989e0e458a8af62de4ce19199adaac7f6b4 +751daec3159100873c9ff23eb238b1b89b6ad090b87470be5c63a634a81326a1 +c94490de2b84dd75b512de2009f44ae1deb901f35590feb085f6a8c5ee526ec5 +1532f6a2029b5e8c21a0754e4040c3cfd3813057833345de20114743a50682c5 +a1abf44f506bafa649f89bbe9e43a5ce3c8a180a7512a111c0644793efb435e3 +3ac1100e7fd86cd4a642eb98501e46ec849fe33f6ad48d2e9e4f87ea66d99d0d +4f6478da3089556099aacc61f5e41cc7b73af9c15b25ae0ed797c1ec31f2412b +9c06b9eb982ea82bf7fcbb965f3d56a179e06de9eb4e3d009df21a0097bc369f +22ccdabcf558c260b2872c54aafceb274230d5dc3c60fa26413124807923b3ed +47a4ff75b8b3f006c2651d78d50cb28eb7086bc5ee042774873e595dcefdedb4 +55d856074c126902d70e8328be059b72424684c956c947b21d35d86119636212 +f8d9e70096b38b30dc608dd368bde84887f5ea03185d7fb97f03b903128c92e4 +f3e83c448489faab14a6d5b58de0feb4ab398863113e46234b8397ab8428bd39 +fd4187d4951850f60555688933f56f8e79e09fa8556707ffc604e7227cdd43fd +01e1bc3b0791e66f7b6f67fd0cb4572f33fbe9ba79dbc8261696d45976aa8698 +c4d1712173287e5a91496e2f349fe4c4eab0294e409f4a908ff3b07dbd96ffaf +b41ae160dedb211e207908fd64471220303110da81e6022c0219765b849523e5 +9d3ad5c70154474405171bf4f645225891fb70aaf0a01eb44ea1ee3c509da33f +1e4f0f5d85bd749542f3c355ee6bd0258a77f262680d8b0b6f415b3e410184e0 +5e9dad9498d8e9921a87d8ad814dd4e060da7e275a11d81acf0e396ca02aee5d +ec90cbd0bbb52f34035e7f026612f188faac34da94edaf546a13b8fc6355d39c +f02486d56b084ff316697f4e148b32e640e60b47eb747b3adc29d8ce0939ef61 +239bfa7a46ae6a52586498c3b9c587f4f91c1b152e36778bd3b3c4c1a0f88df6 +c17f5cef441960890f409260a598eb352485018e9ce1111eabadfec889f7aa5b +a4a5ae1072104778400c31469fe3dfcc9520e6e232b70bead411dc50f1b74141 +f1cf56620b7a13dc05b172695aefa999f520aad96512c49995aae18fd2b6e9d0 +3051d6ba761294e23bece4f6a364d8b9fd91a4f0635a1b909f8fc49c2ac8f140 +97d2d0596bfc23a3da80b241f3b9c9ccc5b6d0bbca760fd447c3ce286e260a56 +e2dbcac985d31594ffa1193638e6fa07c948ba8199158b9bcdc4ea2a39dadc50 +fb4a49b58b5fedf3dcf916c176775f64453061d319fba56f578ed6775b432e5a +92f8e1ef93ce843c37ad1cfa22b98b25f08b2139f5f127e1cc913d2051577bc3 +56689baa126382cdb7dcbc09d28addaf0622199a2c500ab60bd8661956386714 +0adb5b155f7420324d89de01c624f2e2d77fa826fb6ead3c3ab105f37c57759c +15e56695d9c7c64e636e6cbe2319a5d84b2d627d3982824e75d895eae30720e1 +7591aaee8947dba90c28c8376cb83227fe948cbd5c944e4e2c0dc76aa1e92864 +5e1c71de443e12f56b486c4f6962ca875c8378845cb80c5082314dc4319a25ba +e4631e59174556759bef2f3be57eb9c87a7bc6e994d44ab4e3cd94b6c2aecaaf +f830f31e0a2a69c205583041674cd291a26a135616b14c72d7885a333936f60e +4d8a075e69dece980a5e4c390d04fe5ae3f72883b2c0c4d2ecfc9800363d6cba +a6fe7f814a0af9cd0702e16a22aef8abbec5da283e4c077c6680a99e34ded045 +d8b664012ac22039449f91327b7ceb671fe008b57858e35cf350d6ae1f5d4dc0 +eec17afc7e1d454a29dfc9b1c33287fcceeaed8745df8947f3938031e357b95b +58824207c3c85453dd1f58a6fde5c9a8b97353c71cb328e61efd453e10ccfdc5 +25b74a6c21df40b1e90bc990835a3ddbd23191d556ccfbda601d987ddf91fdd7 +5756e6e7176df6d4c88be11e2ac21a3afc28541c21ac1b30a81f1ef9b8ad0ff7 +e9628ded332b0ff209a7fef10147fe7145d81ec72330838acdb166d571e6d377 +db3bb1122de8df3af6ae29991b6c3ebcea77cc0ba9029d9beb9fff3637b20238 +a5d41ca3a3014586573d8a1a88be949bfc65da057485978331062a191ae74435 +2e291753c898481c06bdce5d2b36ea41e1350091d960570fefee483cc00ce62e +9a28a69d96ae817e9fa054aac4432e4c4342b9b61a61ab3d091f2a831b4c5617 +ae55da97bbca0b686d416e57782dd68b15f70eae6c163e6775bd0dbb96d51b04 +4abe701d6bcc7d2d64e17767375a0e53204cec53a52bc1aaeab6d7049dd30b4e +71bd2d163bde2a4dfa155a3f2ffbaa4586650c0f18123d511897e98f5035d397 +b8c89f699fec55eecd4d7b1369c03ae0a051e045bba4bfbe92bbd8e836de739e +668f0640f48b01464fd93c70dce16c707524f5d92f8b46538066281179ed96d7 +354a82835d1c0264bdd4ae12c8ce33af04560d96dcce0472f412828841b8e1f7 +652517493b4632b68a5e95f926106265d5629b763d7700b04a039c0fe7c4f001 +36aedff968224e7218506864f75a342bbef3903ea09ba5168e546d62760a1e97 +5541b25d6cef8eb1a651b8949f0ecd1ba8b47efabac2bc415795df676edc725d +b8d37029a2700ee2dd9351e8019041a9ca1274952cbe012125bf56da9efd4264 +e16e335db96384270f7c7645f6a6ae17d6ce8114dd047edda9136f0fa508ecbd +f116ed2d43424fc738a964f84c7add650b695d47bca83d37069f4d26f18e6bcc +6792217da65d73af7d9e192714055221a9e7e27ba2387e7e3da554973d2747d5 +de59f372a007f5c83e40e5ce731bf7bdca3bbe60cfae906a1f55321e3e617a42 +947506106adc4c2e6fcb8b5f31d8251dcfa10626a0fff1d69c9b23d59118f6b8 +ad3eef482455fe1735435301a567edecff4f9ce39fad533eb4c7b6334c2b3bc2 +f0f93ce326eae78a22eaec613fb0f1c2db7e8128fd350df380187b3ee4909920 +389441c88d003100dd2b18ae814870500bc359bdcff9f26bd341c57947be3331 +d3af98f45a1a5f7cdc1d178e79b53c0caba4520ac60464a0d1f858034318c943 +f46e148f7c85213773d22623e4263f8694c2ca193620080d26dcd94a81ee6864 +c01e9437a83182bffd2b2f2fd7b66bba0b7410dc475987a3fffe0754cc217a26 +0b38505cefb64a5220e2b840cf6c1ea1f3f6ca166919635958d1c5ad859fc5da +2db51fe88dd9fd0da2fcffdfa68efa49e9fbf9828bfb40f65b23f4b155bbadf0 +df1f285382850e97b092101ed8ee5738420466171beb292f5c9cf728d37b8a3c +334b70e6b9a08046cabc7bae845a1abf759ef02ad0ae99fbcc17f92b5976b0c9 +474a1c56c3bc40c6bc82711a2e412e077a38a3d7ab3ff637e4d89dbe6de271fc +0932f8811d9a8dcee568ac01a594e26c0e4021c01842c0cc23e0382e6a1a05f6 +4acd515ca47d6ed0a3529f614334c26835a6d4e976c14ab910bcaaa7b36d0a23 +3c31ffdb17aff785e3db32e8d766a546498af9a259f0cb3a0daba84dad5bf6ff +f26f3778288357fc96f5060781ed968e95daa6137df3327496d7498d22f77fd4 +5883ca99ee4a01ffcb31c462fb1c72dc0bae3daecd46da50f8ec6402088fa7d8 +e0715c04f235f7a349ac4dbd0c95ede273d3300994c2f6ba7a390bf6605f32fd +5f2bd18e58496c3ef620400fce307688500dfe57fde35b8876b601e35b2148ed +8e6172636c39fd528b628d91284f964990b7be16d6187881377dcc44af62aa9b +435ea260de9801f2ae2d969b4e5a52304e20d89d8a19d0f796134e501bdb09d6 +00d2b22b6998a7cdb59b8a732ef128f42b0808f18cd59b79d71dd8ead279da8f +fb734ca210817bb37966ef62d61635f01e59b943838945708ce527906a28f7c8 +e6fc148621c47f1de74d21b96ba4387b5238235b63d975837471518b092ac0a9 +a4b6a8ae2ac22f06095649240152150f96fe7131d2b2805ef53b17c7c9eba0c6 +72f6651e049c801d3d443623ba08659bb4e7fc849ac8c31ed0b8be06f3d25caf +1e4927d1e0f67bb9229888755af9e9482fb9947bedf7ab9bdb6b66d6703c14cc +b668c8770a6ea415619a57e0cef5b8e04b72645ee93439e677327472f45116d4 +eb050bfe81c9b0b8c22134fb8baff5ee3430150c91bda2752ab58e4700e8fe8a +01e64a1fb61c2f34e25130fd07d57cd60d26608563dc8c0a88c36f1ed163708e +2ed7b335eeeb9a73283da91089f03c5179957b3630a0146b4675aaf1aa316f68 +81fda39873c6ebb85ff1a277c9acf4b0651b1911d0fd5e7a960a28da58d88e24 +18e590ce44db65e64311499b0dd262ccf31daf510fce71a5cfcabc0ea887e982 +10c67b064dd30f411bc15ac3ddda5db2f91f697caa3dde48dfe853e4af660938 +6fbc732832a93a76e87947f53095813b2081b0696c08a62e97dee5c76956d15f +a27cebdf5ea140d5a01e70713d8f483ec273482721f8ee7357a2b21aeb88b675 +91733a34f18065dcd59641011b45890114a9c2387d52e1bb7f89e3a7fc5b258e +d54f8f64da628450848655afa5bf68b874909434064a19b5e534cd4790e57ef5 +4fc00f3ada7ce14c0f5b8b147a8e31045409c0f1300b8a599038a5c8aa42da64 +3f95cffe8d3e623a5e1ef500f16c331a1999354267e030e0c7eb9a6f3a4d6e08 +f3d221c5086b88e12afffd8e1a3da0769afc6dbbdc4f376eefc485cee2a9ce34 +dc35b63e2ef77c76a3895cf076afb74167a3d951fad43921f4cbdd7c1e92c06d +36373533c2dcbd7da81ee367c1ca6cc9d20530c5153c3b870ea543156bef8e8a +028e81955334667857e75d9e1c5c0ecdbe8ded4b509ba60c19e525e0f9324529 +ef9db44d230aad1eb98e4c31e21e1c32c2cd2be0924c7713b433abbdd4ee726d +4408e4d4f52837e9504056c8e33b7e921a10c072b520ff810fc938ad1201b652 +0fb5f6a27d1c71ad4c058b0feaf60946523a6092f18f658efd3ccbeef6e9067b +06b43ff291b2c5e2c26041592097991a95905abdfbffcfa03a29a68dbe183ad1 +a329c9ca1ab7be7efc420347706a90658a6f8828ec5372fa7de0dfd0d9f40c9c +8336e695305d3d09e868595b7bb0cd2113cc4b5a8fa2a3fb82d0b773c29f9d7c +84df2c3d71d1dae9f957b06e3b14132eb8b02bf4fab6ff3b00d60295276c3d82 +5796b04d9f19876a295c3dbb80118418e61c23c51e8ab0aa9070fa6c3c156e9a +2fa6608ec030a3a1e729312fe747228fd3ccab29ce1b34f112d861625c7cc519 +6bd77a65a1d60164a3d27421e5d0c2b924b2e7a73fa7aca104f8090954094b79 +2bfa9fac19d0840ea40d716f1189e0ed3f025f3a234d857bb7e1f47bed99016a +d950ac5d6bfecdb72f83c130414ed6d370b4a4fd84af267d3573f71e0b5fc6ca +84e29d085579fcb08a73c3efe9d6b81cd9aa1e12a714c42a602d426af6aa8dff +278fe7a833141ed5e770d3f204cf837137243c59b083fa703486785d9be95cb5 +d9d3b508ad5c2bd3aaad9a141b7d3c7512b01b9f97a67109e8eae5e406535d9b +41fcda1f907a7b6faccd818e3a68ebf2d35cc13e3fbcd824c6562d6d3e20de70 +2e09cd097b1d3ecbd45ede4b4019d62480d27222aab708d3574b03503375bc51 +5a32fc52360d43ba7764ca33a5692945ddcb11b7f4e6b2092e439fdef23d7b15 +f75ec76214779056154f67d410521217631bd713a28637774c9f2b1abd2d5294 +1b3ed3f93d0291d0222c1bf9d7026882f9a122443dd9afa876c99a34d3aee79d +e6abc2e4606be093f7ec6ad756dd8b38a5e563b2b705f269fd562cce579a9a3c +e72d4ac01e830056a2e59decb441e983d99478ace0e976bcb6c4d129542b4b3e +a32865113460c88561d6f6a338ad1075c60a983107c8b00340df9c010b7a5561 +491be1e6c8f1104be90aafd8f096f5534e9ed876b66d602ed5985aba0536674e +9be75b47c00999968dbf0d0cc64cb2e11a222e65bb480e82c38dab3212dd4935 +bcd852857df67606afb7e0dded9a2dc7c2459a02378bbbb5067d93d827050711 +73fbe719bf90ec5abdb9282f2bcc2d3898ca69e76bc52efafc9ec15c6ecf7e6f +2f26ef45203ba477bb9ee38be0049d4b794ac169805608bae95ea71a7b1f0d71 +1d5e6d9f793d1121a075bab2c1b21a6f2e9b668a1047ca7a88d22eea803ebdc3 +54d7a3afabc8f9e46a4e650e93b167b67b262492680a0053023d5aa91d0402af +f4d129d2f34ef150b71c967d92b3fb4f69bb5b8ac02fbc6d7c86c0c5828b3fac +8664c750ee7d618f862ed441ab6bb3df65fb98f855a4ddb9bb95e1a0dba626b2 +aa5a705c3925310445303cbaaf57ad76ef3937cd871bf6e350603c4676f8fce6 +88d0efbe681e09bfbbd6bae2698e1e8b697f3080a627ad653d91902d61c4fe70 +79dceb83aa2eabbb13ff2481bddc5c454129605c5be4d1da181c71880821f1d0 +5ae95728013d96066aa5f5769a04f8b23bd69320ecf3372cbb61187dea723783 +1c8aa90e20d6cedde86efa10e4a8bfb57a9f9e5b3e63fa7b0fade6c5cdb3def4 +49c723037d542808c093351ec40e5469f1644b2765c8b03e9537f72b839ab44e +3ec180732ad5bb6bd4b5ab413b2597932883fb7575c3d3f9148dcd52f0a19993 +ec89952759c2db1bb7ad63f54e218cc20b5251f1d20784028c89f92cecbc7eee +8f5d087992f35b70a24caf747bdbc2e6469e0281533ce388742c5d2149a899f6 +0e6d2f414fcf9b377ab0fc670074ab79f8bd51bb9d384870db2bba18e1d367e7 +27ee36ebc959a3d5acfa705a204b22b0715b78f054a2c863711b9139066c94d7 +7986704f56406f40de9da0f75f7fadaae830c9340d893da8e853d4ed632d3613 +d33e3ce4399a6bbff64d50b0f3afd8eec0c2788de42d2899c10593937a2d1837 +ccc6253c11d89af11bdcf20bdf513112ff3c0bc6fe526045542e1edf9ac6ecd6 +9813c2bc0f34210dd68f79ef060d64b8d531190b3ba1072b56c61d2d0f4fc32a +fb00aa533156569b654fb3f8eaf9ff926c721ffd9bd78586dbad9323b6f50bb8 +7caf72c076992de5bb24db15004c97da0c135910eb85986ea106de9b9a750b05 +90b22ecd1aec05105687e856b863ff3fbf8720d8436c013ec0db3dcc478794b4 +4f7d9562c25467a64f0403269de64a906aa4dd0a0f0091a7f0d0e683872b4885 +11b26c7d7881bae8dfd0378c0a6144f6a9480f68eca67c9e9c1347a5baef105f +6941caa3f62f5462668fce06fbea53b2e078072da2e6fe832fc9fd17ac97d399 +3b0697b9bd4a23f68878dea54f7ea3f3553832c9f7f9f479ea63f4f108e2b0d9 +4eda2b12862c3aff56323d76a33f0739c655249305ad68a49d73afd8b4ee6a89 +4f30834329f7edad2f63824893b0970e12707b6b0e0526532ad86bcf96695ecd +1125d8d58f35881c0e2a67c451b2533a171c3d6dac83e3f8b8a2822fd2885add +ff58846e06d32ccbc92dc8055499a5700dbac3278322de87584a5641e14dd84e +c5a042f7ebb78342c6c19d32e1d86009161a409e338399ddf4ebdf04a3cdb2c4 +efd4e9aea95a1e3c3d909cf3ec1cc3fb3679b8c849eb0b5e30e86fbf85304b10 +e1f3f97522d91c07396852274dba827e3c4b82bc7543448e9576fa40f1beaea8 +e95ba3c7d65ba8f7f49f766161b390a4044e1c73b3e5aa3091b4b41829e8b9c0 +43030528e8a7e94f14e241024b70def32ab9c8e4b2e6e21d67c8794de81ef09e +88caea9fa731401104fede6d8c55a0a209f5b47d9b2e1faf7b93b1e40100b48a +e329ae0d700c9c77b8a6854e49d875c81fb144668c5743147a9e759945917bca +d99d1ed0b029244c1ae60e2bbe925d3b9f0a3455627cc65825103708cb2e54d6 +fd4da374c3fff7764b9e621fae33ca705a198487cf9e9286a555a5043af0202a +30f084abfa05053fa8669ebd1b88a75db76576408a8ec055416a481b92e5920e +80806828c4f751441dcb7e7092b5f234a3850da315414b4401b7e6cd9f491c13 +3c46e1f7a29a8cd6df966c1acfcbfb07725908dda4ae7c67daa3b79c4562946c +fa4592ebe6bcff70d8f0bd2365af32a3713dbb06d7006993c789348ce1992ae1 +2c59b81d1e9db328c613d0871257bf64bd0e4950d3fe0e4840d3ee8f86438707 +b3ff4998e48ae2cabd84461faf7503eecd6284017b5bc5c5c05fc44f78c6c091 +850d9841ec040916948e2fd554f5d2fa310efd0cea960679224797b640a2a31f +5d557a9146084fb040a3ba3821252516b177bc58cc92ac80e7453e2e48305d34 +e295121aa3a1f036c197a456a8de1aa3581aa1dd3521ed03c0d0b10e4d19a30c +645def9858a5f318b1a0c6b2dfe362e03ead69aec62d92dec7c4f64a1e0bdf58 +363b56dc509514c4c5a213a8662cce41e726275622d76797617281410ed56e19 +75030761cb3f55f39727acc17ca9eaad7ccb1f30fe8aac0299d8f24995f38bb7 +616a3c7621e815d686d7e10c4c09e34b1a6381de3698e8582abf7a1c8194bf24 +17514744d6d0dba05cf4f715306e217649fb09cd6ed960a4e492c7563cfd1779 +7ee39c538e97395d94b36396eba5f140911d1156f0ac03945cb920f299564261 +1a5b04ec5053064aee792ba712d6da7a97227bf38043f985b276ae416bc1b6f4 +c8a66f02bf82632ed4ad38854a15b99fe8d4b5ed5a6d877175f364cc43145546 +5078ccef8445b6d9106d30c5a1eef4ed506cd5e5ffe7eea95b32c784bd0649bd +5b5f60c3d8a4047a4094995ec01f92a4d7b7404e89b20b8dae0e6aff4f2f72ab +cdf38142283471fbcd28e43f58e18fdeb8c5f3f1f8e911e99f881f4c892f47f8 +7af08f934eb0c65f9b5eb703533b83728895c770bbc267bcc915a2de319074a7 +4c78a63ccc28ca63a9827416cfb79c1f8f48bb2d3f7a15796e22f003af2fe3f8 +992f5212409d827a7a642572c1c0d057361ec677e7f4fb0164f2ba3782322e66 +85f61a4287d5c5f960731c98e052ff9160ca7e97be885812d9ac30928375fadf +e2c8385effec5772de2ebde71c8905735dac1619c25d9ab5d78357798cd723bb +ed2279eef7c135efae66e5c9c3c27c3408e028035876cd3ec98fd2569db14b7a +e45596793a3cbfe2854718298b2873d713838e313ea6f4cc779119cee6b0d6df +61502ca4e5f47e0af5cffe463fba0b2f30ba66df91b46b67f1528fc594abdc4e +ff4cbe61d1509efea59ec719e1e373381e8506ba45f30143194db878d7f007be +fe4327e8a50f82c4d9c5db51fa034a3481f9081c70c5e044c6c3aa5a90629021 +d17c2beda41c5e4d75813937ba1562d7c3092855711b60a476ced4bea19daa1b +88877248b629da0df5a8e31fd0d1fbecafe48a3d1fc65eaff141ddd9bdb3361a +32c59754db276d52aa1614fcd9da10db3b94b4567abd9d2ac8013841c66dc857 +e883f54431fe2d695db4ddb8525d0259849c04fd2147894de4b19161fe6240c3 +6f50621749d7e3f0e39a4e12609ca276372582a0ddf26ad20a9db154b4a11b88 +2d747cce829789de7a7d63e28e2ca010f08ac41460fbd7cf7c2f401acd03a3d2 +6977bf58e30728ee7e728c5d6e1a54f77e961ecdb7f7878d9c6cb66ae3cd8185 +6d4e879ed12d13e28928117f68128fd5fdc34f7d83d8335a06199d0b3c116f03 +3a94eee9363ddfd07d62397f79fcea8f67eae42066f157e3391c330cfb2c190d +97a08bd37176e19d096aea29b4e8056f37a1044cbd0c360952eff0d2f1e66e14 +6ce439a73c43933b7fa5e1c4827289701331ba64c57c8c8144925a26414a0f20 +59e6e8185a2ec697cc254fe507d4381badfd130081ec27e382437fde1b96e7b2 +dcc38fad28369f8cc7f0dd965d8af1e254f58adf1aca79ac7ec3c752d35095bb +309e6e0a8077d89384c6f0d5acda4003b0099240c13218f4aae87bfb1a2e6549 +aabf894a815ca072db22e11b6b296980c73efc1ff097e7a33a873b340e1e8cfe +d5c8db08359a29eae0a8832314eb860eab1ec9813264b708f228d0294eb46ce2 +bfdaf4f2f9a1f5f3ca04b5ec2eada866a0cd2eca7d596367c449fed1baa0df0f +826028c8ceb00cc8a72262b28ab4697c9ed5100076cc27dc76381c8491c78c10 +c915d5eaf3f70f36f1e095ec72266acf9c2e363e8a1dea27d6048818b625e0ba +4abc8c037943869a505a13b793fe4e09273528ad7d142627641a8bd931e7dcc8 +4acfc5c06029499275ac0c9f4a4126251911730032e54ed02a7b67652a88ca34 +66b926a712476c9e407169fcc89f7623898e836a115a7c2dc47322db62e048b9 +9b2ea67df5bb1a64b6f1a78681765758f4d88a776803428ef36fd5de0877aed9 +17d0b4baad6ac2f92e034501c9e240e84354a051edcbc736a9e747e4aa0f54dd +95a61bca0187840a674ee36689856c45e6a2ca151396e0b58608aa90008127d2 +e8fc54f536126d2ff772bb97456894abf57af592186a706130dca4bd733100e0 +a2c0d75488be52e1d8fb855e1669d18323ec7a2abc11357fdb76d652250a21cb +b73e65d50a91efc5f7c9c42145569ac42ff083cf5bf1ac68682a62fab5667338 +0a49d038b029e22177869a428fb54078592ae26714a044e7cd8bd396a0d3a0cc +3ce2cd3a62b9e4a55ba93f1ae35c8a1719fee86167e555bbcdb61846801cf5a6 +6a3aa44c335c90912fe8e00ca4c856342bedb5dc63433554db6afa989a0102c5 +f081702768fbfac76f1aabee5287f3e42bac3e6f45fba8fff894c2711e400e2a +5cf5e45f9a31cc25b43b7c090a16f9e2c6775fadbe1c7c8602fc35766bbdd694 +8ebf6fb79e91b4b5336dcc829225d877eb09052d011fb5f9c0f8a1394c9e4b14 +a6604e4d3e37b514e61158b50f7e72014fc3a16902e7dde50aa810205a601c46 +7cb92ecdb39820505d2d5fac724fa131437a4e94744f962c5b232e9701347cae +1fe9bf4d6be3f75394e0e9243ef224fb750caec44ad73e866328683301536e9b +8b64d85ae3550441fcd09a3965c80540750f8ad41619b9ec7c0cc3e5b2eeb6bd +a3b7fa922316a76ecb70c628db1af413f149e734f6e1784446b9bfed574c945a +a4c7349ee82054614b3f572ebc6407b404a141ac40e39f014d433c6e4546a54a +25c4f8f8cf15c2476d11508794f3399d685afa08302a4281e3659fb0059e056a +75014b0b913c56ddcccac98821d223a6e259d5284e05ce7c19cbf4ac7913f84a +495898ce409e3dc275697134cdb3e2384cb75d97762b4a2535f7e197c7b8407c +2cefda610288c15f4587fdc44e05b71fbd71e9e5eaf8a2380e3d96e62a75f6d2 +2b04dca720c5f72999836cc47ae147bb163f9aaf548b0c7b501ce3bccee1894f +cb0f3a524f1e8ec32ffc429c56c85823e1203538ddcc66f0a7c16dca82f193f7 +7bfdb3aa0787da35b2ec4cd4887da0c2ed737c8028bf763cb735faffec2ecab6 +df4926d13709bb88e23192eeca032937599ab4b8ef93984aa618eccf1f4cb06a +c5120cb73c4a74740d5d91597bf4ef88eff7af63332e38fff096dc6221676ffc +b0ba266cf29a520f0466ff7c12a85660f37f9716492b52084266147e26a85dbe +945f624a68af13879d803fb82fd3be8e3993e23c4c638909d2fa11c7aebffa67 +c343eaa9447bc1c2e9a67ba0d54c84eb8c3fa2f6b92e3712381edc9d4ee29934 +51a24a134668cffc7c27d9ced39689cc4e11d98438e624ac74447bb11c5aaadc +19fce64044d9a2506755184188b15a7a486eb95d4c569c88b130695a0de64add +42a50f8504dd68bb04c121a3db55eb2fd26db0bbff514cbc4e6f01fc3ffd6a69 +e62129c818eac960cb7629680dc576d6896b01a4ea407609e9b7254a2abadda4 +2e1d2933de238ffb0690fd7a0124c6964a28e082c786b9fd6c072844fe8a10b7 +0d5bfe9dc1d6db18e245e8e576aa0f9c3ce7429f0f2d75249c25dee7f08ab090 +2f2f35b41f1f8824144dffafa8dcfea47eb627c82cd8b4a425d29e3e93468336 +89d88aa4701fc8afe67b6871c3cae0070c79900a4c0ca2eeb0aa11d6b453dd0f +6ccbdb2d47dfeb2b644d180ce58c9f450ac4fab5ffe007daca501b859b4b51b6 +2b49ce93d8c37d389416ec96fe8cbfaa5f39fb99d77d8c51b6f795a29a14b3d4 +06a4a70caddaf2ee025abb13d41dadafe419da511f4c86f0d0f2b63fc439466d +980530a47282c1929419aa05e8838fc981be073c1a1e6d26815dec542e8586b1 +c593af3879b48528e2cfeffa6c84ded459cb0bce5710cda5295881a1da285cc6 +8c84a56cc66296a3599f325dbbdd268669b07a29ee29cecf1b7351be6cb050a9 +4d225e2584fc6763f02e46ec15fdb6cdc44be16df5a0c9e87836288b822d5155 +a02920445705cbc5bf6c6aebc4eebb029cee19b84a0443d3fd3bc34d3da35340 +6e3591997e165ec7d5bcff87ab35d5df2fb09ec6c5f902524564ad16d81fd1a7 +e6a8e262ddca89616873e66bcb0b7d8357975f613bc30f580b9bbd59eb203748 +4e66e1201785bf8a51572163778b5811a1d2fae8edd03dc7201c6e645078554a +65026ddc0389973e0c1a0972b27af7847fd26356464c13fb1107c0643b1ac57f +f9d628737b27e0df9c982486c3d404fa76d46348545825ae42b2b34a2bb88239 +84201fc071fe4e6cb27e0b23e03592206caa6d7ed04d92110d33b25ecbdb2135 +8ae8e6a0fa895f0bd1c8f131ea1b4f37c9cb7bb2f4a8851b83ddfdc5bb056af4 +8971146d485d0f4d74624e51b16aa8adf0ae3d1a5ab999e76b8fa1e573d6de8e +ee1d255237524e3cfe94ebfc3c231528c9a015a422c2fe1a5758ecc4b63a6adf +e054880c270db5c535f00dbe8c19a7a3303e1a5e4f2e16a00f90bb46b6d5c80a +c8965070a1b123043ad0ec9b91a6f8973965354933ca5552d503c37c463ea75f +25280625fb975446ee1bccf778933b3036e146ef59991fbe79d9613b0c9d5839 +3dc4a2e8b72094538eb0c76004b52fd0932326ce0adfa20685a213788b5af607 +4187f49b0c11730fa866349fe81ff468460f786f4493c43b42a64e02799f45a5 +65057a08d2fe86abeb4f708358307e242197ac864c0502bff550d2b8724869de +cf35e6b0f34356d9865c01a1bc17e7b8e0d74ee7025053ee96a226e3f3a75708 +57d7b09ddf00c95b78ebeb7d5ec1242180379e225da9848835a2df213901b148 +ff8209519db4c18f015cdd52200e7dbe27a0d8091df4c6e711b6192549f14e66 +77e325d836081143a5915a55b92897c44f220ab07fe72cffd46b0c07a2d1629a +567ee35327374472acaa8b4223ec39d19d5b97e4bd168cc6ba25951c7b84af8b +769381ab213bcbe6778f4a54e9fe7cb7f713ee9e79c7df9d413f89d1db0a45e4 +4ae07b4c93e54b18b220885e7e17de9759ff7dffe684c739d2e6969403baff7d +22592e2d1463064769e39ae6d68f5f23d6327c91aab1fde108b7f4a3e381b51d +77154f3f83739e811e7e30f017fe046a3b448a37a06563dfbd438583a6f92467 +ef3385ffde53e9e7c510e24623cbc39037e96b6c29d2db0b4eba836411cd7562 +ae4283a8eac969535e1beaee23a1ffef309b08df04c3eac5f24f3385958a67c8 +7e898c156c1553b17ce61bcccfed53064f5b4d25eaa55bacb1880fe18e3856d3 +2f9d0ef3524be3779edede8fa49490facf3063714c7fdd37d9d35e9bf152f225 +22ea615c9f50f226d81a21ce7655083124a20de4df222fb2335884637e616692 +fa928cdadefddf0aeb7027e05d97be567352fb973a357e6054bc06454a88c917 +084a6f432e44691e53abc81c93d9076c621f7a0a897662c623cfa8ced383512c +0e10d74b77aca5d329c73ac35e86d3df933f178851b7b9f7885ae4a703f0d99f +046e76facdd7058a348c93743db1b893a8fa5335975ec840c6f1d8112d387368 +76e3c979f1f7d6551bcca4bb79ea4e4c7d5d61da58cba3ba8d47fd7b10e35806 +aeaa1c3944e465fbc1ee7018797af9fdf568f2a16446d8b9fb6cd6dab0510f1e +4cd076a59bdf4fd1b24f04f17710f8777ccc93c0f5bb4d5076a68697bfb506d6 +a4c788c0ba40d168e1d7769d56a59f4bdbb4fdaa02cac655c764be9f037e0dd9 +d6f20b6b07d64bec6b7c681da2f74160246a9552b3aa9852e48ca2153880a7a4 +a301e110c14e1f6a4e721f9b65259b429230fdfd24218f447107d0a26980af6c +148d7c0ae548a910ab82eacc32043d423cfc4e63fed29f613fb3a76001b83a45 +8a26a0852b5240947925b864d91db59995cbf1158ad0189e108452c5e96bb477 +4a50886835d6d59d07f28d08d650b3a691ecd74fba087c182011bb969e01d01c +308a547d7401df528e54ab2b8e26ebabc47299170931fd94577ba746a4113458 +d87cd14baeebd2e7939c0a019fa3ee49122422b900447e5f1246c77d9803e4c9 +bef93e4811c3f7133825033ace72adb52d34c3e1447ee6c589a813f4c1fd0d4b +4243eafaa10312ab96393bbb430b524cfebbc09c25fe0b6e0012d532d7191aff +bcd3a68b65802f798620403e0fe914ffe055ce7bb9c429aeb6790042479043bc +ea9daad58c1604d34363e41da7b8a54ae3d7f174c3e94fe07486212a23111d8f +a92f169cc1222944fcd05277b80b51cb3d940657d7e7fb91244fcc920601d5da +ff172907aaae2527caaa32041be1b2362dd3794f364285655441265109aa2a7e +50d96c43c78ee35c96954ecb5e72be8d0142f6810c928a3c57e7590458f5b16e +e7c6c5e694911d7ca9ceb5fd0787b66dc276e085cbacc781a04bf2dc638eec2b +23cbb4c14577555a88da040cd36c94d7b15896227d79f605fefca77bfb39bb9e +8ffa91f2fe706360ef61524a7070376f76336ecedd9cdda18757db84fd09c3cd +9af161f38a18c09a8c50cf61b3c9099c52242b4c4e0aec881bb64a198df24571 +63b0ed06b2d3a6eb78e176591abc3548a027a0e7f261d1398516925f5944807d +77b639918e7ff45cef74e7ccddde43a1dd1ca35e2aafbba0f7c673dbd5e77359 +1cc42019721917c8a767a7a067c1ba526e892ab20446b86d1d85502a05f113d2 +f8a07aed132481f0938ad9712373c71134fe8f334e26ad08f4546b0560624258 +173a85542a577f34d1731310d94a58e0ecbb7538c84ba021c18aa780d651c7f0 +3069bf4ce43f0c00c049bc0ff2e33a17a4825c1a2c16b770e5b0819d89afe260 +c9eb1a33a769b0886fab721218380c0600dfc4931cbd15f993767e19382b755c +1b416c76516b3e7d4fce69d49992a298596d9c9c8b33925272a3c643b5b7cdff +3a54dd2f150e2851172b93672c7129ef911a7d0c5290bac06e45e22b1dcd99aa +994fbb34a1f726e620fc60ee59118b9f4ea6bd0d726d3ebb1c65315d2ee84d02 +1197360c30877fe7eac48939bf12b7df8d5a4b2a630c335b498ba6f5881909eb +985268f865d3d9794459eac85a5007eee037e0a9c0f105429b01c836cc2a9bfd +ec1bf337ed22eeb9fba1abdad6468b2b07895d5d22d4072955f0d2d197b98803 +a7520e9925bce7700b42c7773aed3dfae56f8b6de5bdf63ae03408234499a54b +424ddf9a4ecf90863639ea953b32a44fed84b68d92bf1beaa46e3c3564e09c1c +776595c53a1a39c135c9078b917b4824b7c4627ffbd54aefc2102c5d27db5b30 +78e4320513ccb7ce87d96a37bcada9c52fe402e96165afdbbe5c40ab2517b246 +1b6a7b3129a41d483a97023d40aab110c4c15e5adef7a7e63dd8d7bff12b3f0d +7452570f63ee32f760fffc766d2594a6421bc3ca12c021fef9a18f01360c9de0 +7d817d4066a14b7bcfbf6d826c30ef103dc9af9a8c07b79fed38e47cd258411d +537c5fb3a61cf78d62951194a1c039402c146eb34eeee1a08771bf178fc8bff5 +37fa08f9934ddfcb4cb6995ced98cb4c6862986b673bd9c031947a9655e2d6de +0039a6f53f95f41589a354498db454a3f9db24ba9dbe6132556c2de830a40772 +f0a9b97a66cca15db4f67b3fc02cc9a30d346adc0182a1a25cebb6a60d04a7c0 +a643772e10c7370a9e2c20d5416908b6c281fb4e83e15af6068db64db2892197 +70367ddcf0dc8453cdf3e94ee436a6bcbb85460d1281ecc1da3e41276c139082 +6b9beb2916c456bdba6c4d260197ca497838e40c83551e87215793214e9e9461 +550d1c7eac671fb95ce42cf31e42b9fbcf2cdfadc3556096511f2e06e946cff7 +885b538b492494e20bdb1cf146226229d29aae3c94942faaa8fe7fe84295fd30 +bf5c65cee6fa18358d378acbd848197b59caace9fac024a19b5bffd61b82d022 +96c4dafb543012b8d4a1cbd9d5cce2ca609ccb914fe8c3182f932e066c977595 +4c640a933f9a715b795b2b252e9a1504164057207a4ddb6874b020a7b04d64c6 +a249cb965c75d2ef7bea59f563efbbea2e04236fb153f35c35d00185ecf3ed90 +557c7a173653c6f9d029a9e3ed3ae35e7693582da45d2156f429a41f717267b9 +a4efc4a341c5935037a660df5048ae26dabde87b91bfb2f4aeff2bd95ca1d3ee +4fac49f0ef846f5f3365b6e8e7aaad05e0d62d92d29229c751a7789703a559e5 +4a4d567a38a88f7b25b52debab462888b42527b9f2c780bd73b9caf9971b3648 +293e6998f29681dd306c7226d837bee850c75c12eed2b0a95416433b3e4eaace +f23d4a1e8a337443d921b51c34c9de6fe12c20c7b064f3ac11cbef74e9534d73 +674a0b96abee633e9de9e0115f1684d9d29193f97070f8cb495f5eb3efa0cd5a +89e1a2dc4491c787b59afc4fcce6683038bea98fb0a7e1db598c5eb775c20085 +0ae92afdd505bf969d05a70acab21640e31983c6d33bb0ca34e2831d7d5163a9 +995873a15b6533d7dbd913f7594e5e07a70b0b3e048cad24d37594982468af77 +a9f07d8a8559503751c051bb8ccd8f9a8c73637b6d01a18eb71dc0e997cd1f19 +024b79a0c3de513732c0735048333440cde978628fd6e7eb55f0beecea1b3db1 +13daac68642862161fe904098001c323a4029aec7a560c808b1b37951ab4f891 +18c52d7d20358f70be2f4261e541cb645a3071bd069201ee7122f12fa42baf95 +8deefe00c6109d149f44e14ec9bf524d0f56e39e7555f486dbffb711962f5a88 +78e0c64fcacf9f33f0c72ebaf333b71a0c8dbe4bc6e956f8b58db190d33c387e +483122a32eac4d7db5e051792243ef0baf140b960d863bdad66a0b13697a3583 +a4ff53c76b2299cd6b45f6e83e76b720ed688054936568774480b11ac3de77bf +2ff342a5e7367c65aa623357649905e04cb48aa40aadc450cde961ce8b99065e +94fbfef25f116b423f810f60b8d4c6f494aba8d8b7c90ea150346cc88818e9fa +b36a8ae301bfa2709ffa100035206c2b67dea71640bfd734bd4be52f3875999d +3abb182d9732a3e46c5786ad8ebe9a744f32787c8b2cfdc1158e017b245aded3 +27a4da1cbcfd3f3b61e4cd464170362c36ed48ac6bb687268c347a3e0558f929 +043d26be9a82e267d723037d50a799b98bce4ea638c4282b7ba31dcb3b1da376 +87205f407511c09aa210c7a394129ba1213a78fa99ab66e9a0c79de3664b96a3 +96038a4907310d8f239d2746cab6b1c2f90e9d8f76c1d708d9a2f4c48bdc1b96 +8c0f50c0302a6ccd153d7dc1dc3d258a6b998d459e21147029b96a65fab4d733 +4c46dcf5b152eaff56b3056aa6df53470330ce4f051bcbba72f1be0452d537b6 +d24aacf92286652b3fb412fe4b4269fde537d732fb8ee6dd5149a67876ecb85d +54d1be5d4f90f64ed1834801b11f09847ac7a4cc70d8b8ff14efb81d722aef13 +006cee82da34170cdc93bd8de6fb40598b0bf3915afd84cd1f5a7563ec9f9e0b +4ebcd032c8b58c130daa931c7cf27bd8c39be6ebad22092d0b51e8fc861e3bb4 +09c39fb7c4d8751329c8b30c17f7f0e22864ec401e68b37656757b1f4ada0414 +1329f77bd96898ef4ebbaa0031f94a8f79d856bba8d09b42443b31ca3721ea7a +a1f5d621cf61d8900bff0aa7111eaa1f55a3718e6a5e9673babff08565ce5eaf +59e6c30bd72a3104d733fb2d197dc911f020b6e096d461b0da9af0a32c0f33f1 +42118bb7af6b7eb753bc2f7ca3374728247a5f628075e8dece2509022015b6f0 +4f7326149cece1d12653988308f4e6d57f499e3da26e005d203b755b7366dfbb +39b85e23bbe56f50a0e18baed104ecae91ec0b52b83571f930b050d3fc096fea +d553059b2d5b758df206c298562b6f635914f0e56e023fef23036638908aae93 +0b63afcbca7fbfae9c7684de73e10266e929e50883a1017429400b6c54694b3c +b70a6f0bb2144e125fd304f8438712b9c899e78a66f54c5061c8c373294ca128 +9ff8df743e393c3a8d629aace03e49396b04d811f6a645f08b8c216a8a6231f1 +25c50c4a4938207dedfb73c603539c602c1391581fc7c4554bfd7cb186fd3b25 +e4cc146d80927fb85e975684236915db7c7e71d5ad0e05b595d03527909f8b15 +0435c719db432496667c9aa78a0ad6b05a95f31557cc9977799b73c4a4f486e9 +c2968c8a276978bcac607701f1415f85b526d0bea59d85e6c875f33bfbfa478f +e5bbf27d6e7b58931fbd0b218d23784a499b7aa7fa4511f958ee9dede4016457 +db6fbfaee3dd87a4e8b7e363c3a9f8d0f96e956af9f26178168da19577f246f2 +dba4c1ac290b4e99f72abdaaa626b79af1060ae47d3437730d15b24dd949cd0d +892f24e55ef317cd538150fb8ae35db296f4a7547159022f239ebaf5b94111fe +5f93716704a9b26815597f15ec07886b31219dc669140875e366b03eafb2204f +779dba5cbe7df2a30aa2c5fb6ea01e4d138ae5f0daaba524566bd3591b9503d7 +f67c58c8bc3a61f8d86f023850d99185af84c2b42e8d30ade472e10893ec09ac +b84be88981bdbdca09c94d8b037be0aaa59861965a097e2ecde9c4e23ac154c9 +b610ac251c75fc94046058b1ee287bfa2476585f06a285b7c5b5e21041be8d2e +ab366ed7b3ce603f5e72887afe64146b949d22c6c94d4239214a79ca1e40bad0 +243fb0da5503ec568349a05b2d531f27a90479a5fd2028552750758f7194be8f +73f3b8544acb210f385835c189e63d9c71f9c79623341aa12320f23a88c40013 +da0d7b63bea7141f6bda10bf9392df5656633c422868db496f26edf35d3af889 +10bb3342035a4fb5f253460b243bae73af1829d071faa9f03310ff31d06f2586 +0925c6e0b5656de44458df7dbe81592d39a99fe46c15f4b218bf2473d536f890 +5c77277f4d6ad20b03a69d2ab042ecc9d56da536ff00ea5a9d49829b4a817a89 +ee80a3b899c5019df6aec80214bca901edf7d6baa83e4e74c7d254bde9f41297 +e5489b5f9c859fb0173eda877a3f539a33b0fdecb066bbf4b5c7df9cf6622ad8 +18d4ac91dd3ce5830735cb15ce770e3de84d2de3a5412dd094338bddefb85f48 +90577ee4cd8e4e94b2c0e63d8006e2fe343bb6ad78e6f071f9e90f9e54c7406a +51841ebe84bbcd25ac7b7c9e42a60834f1af0be57294e26d52a5daec7232b9b3 +103f5a7e6832f4f50637116936a0e93157835929ab284d31011a62494139d33b +70bae46b64c9e7a79ae27da721b58938b729475c5da25203a0f89861faca0326 +07494dbec82039c765bdb51243cdf279247136711c34762c5bfb938919cd66c2 +b3f42fb4e188f1d4cc59338a21760f538cea557c3be1f0535589d451a1db49f6 +422d2ad4bfecc31958abf7bc443f67d4d9cea0cc98a2caa9e1a008011ac8d0ad +904d039ce2f3825ae1fee72b75c39c5cd26247bf887a3f5378f57a49c13d8875 +c0e786b6e06e8f9efd9ca36398a96d5a11e972dd3015a418f6042a2977f8ccea +2e9a28d72f9cb208297d12289ca3596063329e9e1090394db4240f189862c330 +fbbc71204ca35eadb4212ecdc4c880adaebf3de31e489612483431d436bec3ff +d384e942183a61829d0c99d84c87475eefcdf6277b72989655007493e4bf021c +f4e2406276e8efb0f3040909b6d77214783576050ae3ff5dc0bfea80151c9b50 +0148a6661ca58b79ecac28596049ac006b7f915f28f317fc7ceb1414b932f66d +eaef93245209bc0bbbb806d6131fa32007e4f027c71ad6299ad0130e595883ff +81b3b214a11d78d72f44384710db6c0669b0389b8a3ff57483c9ae0bfb48a0cd +f8667dc1ca86a5306b3decda77deba1cce1197235f46be15876b8c34f954de6f +8b67051186fc1bb6d3a879f9696621d0352af7abfa6278b1d2c70b6c5f042d3d +4484e62b918faab6e22f75d3bbb623da98e9179fb38c25b7a2bdcdcef80bef5b +7ac9a1a7a96e92108c1c3f755f089211b504c56b25ed59340cf57827f397b9e2 +765bba9b3f1c0475c296307849ecac8c2b82a6c84c1f8571d167845b453bf7cb +b6a5d62a4ba3c02a520e3ebd7ab715dd111b3da917894bd807f3ed134f9f5e30 +039735c85575d6eb6eec70c1ad85966e7eef8319d17e6e8e3b1043d62ca59e47 +8ffde53f5b878f9d8d6da369013adbdc2a290ad237b5fe6d2870f27b033622ef +88aa4846f29a1034655fb2aba40de2d9471248755ddce97cb3cf90c937488295 +6356aa747802299c61f0674fb2deea7674303669df3fd051ca2e2cd36ef977cd +1ae1d54c53aa2010e7e172cd736b28cfba30346c605e74bbfc58114eb1f9f1aa +d030eb85299af43118e703a1d61a6915432289734d164f56d7c08d44351b05b2 +392334c51eb432b855b42312f22809650738d052817c386b2d977c3dff73349c +ed555082565d8a21d2f50371dd58abac5f0ad5c234a86166e58a23e8fb5b2e11 +fe1e001716914fec7c17357fab233efb81cc67772a559dfad42174174522af25 +4c5092439b4c117a91a7c7ca5d6dd79ce365f664f4b61060b99315947da64484 +6a2cb03353b8b417fe11f7fae8d01fc51d456ae99383ce76980981b6ee8a7f60 +7dc8fe0ef386d661da6b0d23859c2499cd48fe272e6a278d855d1cd10dccf1fe +3ca8440bb6377ebc8b0951930accd582ce631a72dc10f7a83e1b0ad6e2bef840 +5734b051fd53812970098d5a73ea301409d287d40d7100414e59a1e0aea9fade +bd5aa955fae94eb5dab26422f9c1d4faff1d4511b8814bd3b1ec4fd64797964b +86e67d2d97850bce7595ed9f6468623152e8af46953208ecdff8bc0296a87acb +e94fe8611b35caa09bc22bd45a9c11b60639ae91098d58df3c4fe637846f94c9 +fd5dbb0cc1793fde82bf2549c8c71b4977957e1d42aa03b9f4f61f487d8ee790 +899e497185cc1803ffd86cb066253c1a02a2f2cb1d1638bf11bda2931719fcf9 +3444e08bc2d09b93b262075de09ff0c8777251e2d4526c979e205bb039a8a047 +70c203260f3eaa2e318f8567d127a538f1a0e30a1f36cf0f37446b8f0f666d71 +299ffb8495dbd031e723d71e7a9e41e3aa64ca16ad470740efefb023c32525f5 +782091cc73972341edfdd20da4bb9935148a790149022768397066167efd0e81 +c006120f52952ec6857cd2605b1f0eb06f2b5f2c8d70ed2792f2995a00ea03d4 +c9890df20d3227e6f78dc9fe43b440a1111fd2ab46e11cb935f2e9a634614cdb +20269504b5b1295b68a04bbb152356c2ab5232e9c9f6f89f7a0b979ff7a692c0 +d647901bbe0cfcd4cef8fc87b6925ae88e9f7e501441077a47c7006a93bae1a7 +4a86991d9eb0d8533a92cf5e135b59f0d64bbe4e66ae7382a16c74714d3ee33b +e8e6d2499c8afba322ed633e63f475e5c9a1ab53fdf55e7ed008c8a1bf769114 +4a576c2352cd2283e44e2ff8946f60c47974d3f7116908b2f3af89328d0b0497 +3ceaf3c79dec614ea2d5268980f8512e14e2f9fa6b641b2b6cabb1b483cc3a3f +86c5aac0592d3316df8ac7d37652a2ff9e95ca3cf670bfde9f26ca47d9b17498 +52cc6b33e70cdba0cd2868d8f90e1d8e15c203f5a536945b9e13997af7ed0d21 +13b95d31a11056e9cab30f3dfdad49c12762a8c095c94b59bb688a79041a28b1 +a6a0d82d9cb5d8e8954090692cdaf9a225bf3000155fcf645c94af63a5969cf1 +927c98c0b02b2fce276b058cff33de28ea737685992458fd9329259d5c410df8 +409bed8556bcc8916ca855412fb85f302955a96b2d6400fc0498e1878faccec9 +bc67ef729905f32b6ccae01625a43757b88e5ccca2c6a9d02a553808977aa165 +9da69e47cdc1615199ec10ef9057c1debabc678ad446228268af721ab7ca6b6b +0be531f90b7f03eacfde082f494050039b1e81a651fa114b3cea053b3ce7b3aa +276a064030948a10086df5354a0dccf4c731a60ba801872512b25dc58667019b +a5f39ae2623d4ca3c8c2625b755315d35570bbc8d9a49bc0331d3bfc0a4487f8 +004947e806c5afa74ea4b64de0bfe63bb7488c2c3e4e5d4d5d6c8403d16de46a +d4d22264ff277176440adeef08313a40ccbef0a6bd63e7e7f13e78ac21a8d21b +95026ba2ff9ce39bb5267bdc0a1e19cfaacdae54c0375199448b843c6701209e +5741da1da00b6932d0a2639459b3e1ddceb69578036bfc9eae9866f30ff956d1 +2ebec5ad12c46e226578c88154097977cdd7384df4b47c98ae1b5a73009b8d0c +792257445f2f59fcb33e459e1c83d970601272deee376401ef21b21edbe5d4a0 +05488e148ac6bae282b40e219778317eeb59e30e320f730e6dd3f65d60402d58 +eba5693b48342a6fcd7be9393c343d2ae2841ca22d0bf131fac48eb44aa2e32d +a29ce83ebb869439cca17e8ab0a2c2c46ed4ddef68af71b09f2da49d45ce34f7 +5cbceece9337efe76e3636b3559fa676662b5c6ccde4c9467b32f6856725b35d +382f52c2606d0b2c9889aab51983513faaa3028f905b09f144cf8d4cabd922d5 +b6c72dacafe9d577985e4a39acb93ed53cbc3d9292d772628856a2c26e7c8fed +502bb6f07a4f9c10c4bd7d6a17336feb8f0274c0672240925e64b76830ef88ed +e56188a1ac89937ae99571ddefcdaf0bd0f05fe5d83c6c44da735a1e8520de25 +9a127620b355b8175170213042a6f9b4df3aa1bef4046f6be73e81f6f2694006 +44b7955116fe11a335928193273b9477568cddc61bbb8ceb2bed56a26df5465c +c05181923a92a8f493a58d3bfdcc3a7dc5fb3f4a3d743ed8aa71506c4e771ca9 +57dd36f69b16e1f54b77755753c486ec5423497412b2f65a09865a4bb356430b +2c3249d410841a41599d42d5bb9016b4d9dc94fec46e2c2398f9e13df8cc2eb2 +f39163564331cec9c05688788e25096cd95ab495038c6bb203244a2175dbe08a +1f25052d80d02257a4bf3a41627f392603f978112ae090247477d053d2538a60 +11276d9cff6e97e4dc70ed5e45f8361ebb240d5f208a8aad9d807da3094a0d65 +e0d9757d4d88ae3770d9670b0122aabf58dfbbeb2efee48f4e952056a8281604 +f65d26516ce7873375160d2a022f43b8f4f9c5e743ae742193175837b1dd05c8 +ce5182f11629e431c827bf6a72ba0974d6510ed12cde60901ce079f4f666ca27 +c22aeff2e48460b609c1d0623af2e8b484de0fb8beaba8ad374e235ccc70c8af +9daa68967cf3ecf5dd41c4218131afa3af4a46b3e40ac3088b98623d6dc31dac +5be9539feb6c45a0d5ae93da7090bf56ecfdc0e5a9073b52f8c2a3d0d4c63b6b +08bfd9b83964375d2b206144f38ea9a5e2965ca00425630ab795c9196b3b758e +b249d5c3930ad24a6c65cb9b2c6674ba6dd98dc78325e582cbded46de9211dba +9b5ede6660da2f6ea9e1aaf5887ecf6b5b0aed9b7852531d2f83f07b5f914842 +9e9769201a31a3b2f4b6c856cd49c34a97b61185d487fe64c81bb08a384b82d4 +5b4a436b40643465f6ac328f9b3f3e750b4b3907dc86f63298bd0062091e65a5 +9312a0568fe96de2762c9222832c91870b45fd07f0399281355fcd32e3cd562d +65109c0b4e4e843d8b0e27b429a78d1277ac999af71ef7c31952e45b0f747b2d +9652168767a9183578b8ef33391e87a3a73f5a711eaae60e4892501c7a9604f7 +5400a84b9c5b7fefbba9d651c9447e4517024a6d724befea78c37d775bfaef2b +b72b35e22ef707071ac43ae4e7eafcf027eef76dd351bcf84ee0f749e8bc0bad +85b968dadb55c3130efb9ae5da3a0e4561af5e888b18acce732587546477ff80 +0312433c75e12e46f408b1d2cfc99f044491520546b323876637195204b89fad +90ff83f08b27f364e5e046ca664c562525b0a2f2cade2992a3c8d1430280d248 +fe1ce018c06b424b7941c02c4740ec944c049c239d9c2418d626931d692af0bb +176a88ba2a00fc943be41ca69c1d1346cc780f550bdab1af72f740a1ad7eb676 +372e95302e54cdbbc6221aa066d3f7bfc08cde388b6e4cf4a448ee180322518b +f932a6f921ea2c1e7d920be5667f963c1dda94b976f1dc0ccaaad5af56750330 +3af4e655bbaabcbdff639791935101eb7c6ed0ce3e0141d09896d01ac9a440ed +3e451bff44a2bd3aa3e583daf6da2f8047563741ce48128178c3ca989453a025 +80526d0d743bf15b5e34ff6cd0e53a8c2579a726a576b0c24033e2408bdd9a61 +778568f790486a29c250a45d601120f30a83302e596e402e991c2fd97f196e60 +fe804ade979ba0e485efd3e3b965b8d9bb283bbefa6be9acd0f779a23e605fb1 +6eb31cfe7278617d86f08905bbe4b2d59bcb078776d59c71cae092d9e5334b66 +9fe84c0675db00fb56ec271b1756ffe1e28295aee61b69f85ef51606241a43a1 +3de6c1e1fed587f97bbb1f9800091b6b18c130a0dc2a50eba638ee4e27dd4df9 +55da5aeda4081085ebff1c0a100afc845234cd25691458601af4ebfed0ef6d10 +b17f07860b0c14e2bc6d34e195627d0dc4adece75d134042e84603cd96939578 +87a018d99d9c8f83fdb33c408a5d68c532eb9c550878994942cd11a885f6b636 +b8ee77979c6256e943f9587b883791acbb0f58b61f8b538b84332c1508cb4cf3 +1c9c35b8e8fe7027321497991a3922626ffe6b452c86c7e49af00aa59718acb8 +cd208b1e5569693c3491bc25684eda7a799d22511ffadbeb8bb493fbd9ced118 +bc5aa0be6822604bdc51681f79f79979c96707959745cc7656c0755315220a31 +b9d8869bea54606df91b216179ee99888556691e0874effa3858fb408c7d1b9a +1207676a97827ab6bd33503c3488a5478fa3b12263c1289c0718abc6fadb6353 +eeb2f90835be6a78b974e5604c1bbbdfafac2975ccf860dc6e29096a92a49629 +95405acbea414ddc6a105054c7278074fb1adec8be296b81a3ff8c7130c49153 +7cdf9102b615c7af4e32446d39d7ce00fa28a7be7223c801a9d13ca426d98451 +3a5c864f998db222264b6341022f8241ac719abc3ab2ce6b55d82e6ed7bdd6d1 +7af1720162fb4836acb1b664ee3d1e5a2e62dc4f176ae0787bad6bf10b4ad177 +a6e80cb0a5d7df4f4a92459a75c69e9f728d722d5263c8ed786565475064610b +0a16f8fd00064bf6987be8bf352ba6e7f7dd9fafc5cb3211e98895d637fefff9 +6c1cbf22ee54a5561da18b7602a2f48581bcf231f23b674a73c4b705f3d4d6bf +ed178aebd26b682900e990bf47e155bba9d9da9c7b8102748a5251b85113cf8a +41a2ee43af09ca218404987ee75ee5af9078fdbf25727b153c6b2f3f5dff99f6 +10c5dd0d943008f61df52ba020e877c3d50edc5a13249ba3cdef75cefa7b8099 +6cc0f3a4e4f769742030dcf6c863fb2055058778504b4a0162e0f56fca95dcbe +54049d94ca55f9629a1f77684d4223ccf573694d996abac505f3ce8f30a99a2f +835d922ab34992eb523ae79468800af340042e5ca31e3c374397f9a9f2dd78db +3eacddd6205d57c2d2ab08091cc2400d531cb3a68db09c288bf04dabfd242c98 +32e2cb839595bef0aefd3b878ebf3932e7aab36d7240551ec3337de3ec02f92b +3e30527e6999ac26cdc19d03e89804653f1b9223c96a5298885b17e396f9725f +141655b0e4239480afbdc12874e642ffeb94da72536ee4b8bace750820b26a06 +90950c341b189be5715a1b6865dc1d345c3df78f5a6f4d170c3ddf687bd50ada +2f0ada73e7579b6430f45bb5d709225d7693ed037777eadb615845bf42dbc487 +d656c247d39629aabb160f0f7ef27137c9a38deadd6fa546b4138280ec5a2a6a +104536329a17c0fc991d49ca92f70a3a783594759e644fc53d68b3e963439ddf +34323ee50be733dbc8b6c941e309146e9eb80ef19f150ba691d79f43596f45df +58bc766d3c2b53b2ba809ea324a118f3462fc9b31d6b0c84b59a0ffa57b95229 +9b987bef1945cda4966dad88b436d6c2018128362221aafc8db5b68ee41f053d +3839bb72ce2c1a0541749bf66432f8caf06163b2b1323b95dc11260c29075da4 +5c28d6c7f9473bdd97b15fb6338af8ab4563e96006f8986162945d26101e27bc +05c6c3956d29471641a64c742bc20b0df2f0433967a93551b3b44be45bb1940d +81dc454d2a424048b4c525a04c22f2eaca189e1ac8babc9aa4c016657f3736ad +30be3e9123203d31cca49bcf7be527507afea03be6b5405357d0be5e6d0218bc +d1e1bce3634ccf7d4cdf1253338818a413904af68d0d118e24f691bb3aea912d +88259a5412270ef7d7ad5e193bd1fb80116876572be64e5ed4ee0655a90663ea +da9f11369022a13bd75605a6565894a2797e2add93e4946996abbafb69173ba2 +f7ae3a49a02a6ed20ac1285727df775c89cb5a173a97101707659f8ec4b862b9 +5acc8d0d01a087bccf76d8b00e2e718cb89a851b9751334a4570a0e7a78c43a1 +cc7dc394c5f7690fc13d6d8108d3267d1f8307be6647aa013b7c5ddc79bf3bf6 +1e3853af76de2f01c704981d8471d960ec786e09e1dd4450af152a618d30bfcb +620cbe4ef4208fef238358325d26f8402f4ce0258bf8e8aeffa59dd1a97d7739 +9e6312c96f4a73fd2ffbe01c1396afc6f03ea79a1f72f6e84816d2b260311c7e +307147f0bea68f85bc5e75feb9a44364b29d9e3bda55f2c7d64cd691d2d957d7 +b017b7eb681c42459118baba989f0ec85658a5a1d849d7b906005632eb436b8e +0dc58ee231b90c98191e4eb12d0b713100393239c4e1423f56e5a9fbe9a47a46 +f67a518f790b12caea7eb0f80d85798c9032fe177396289c8f5ee068b40df854 +4bb76255eeb125d07cb4e0bacb58cd892ac500a35b7ccbfba464445c906e6b88 +bffc60c5e5833f5296c97e64ecd889c8246fdcd9e3febb09ae2642ae7e582e3e +6d1642a12cffc513a6c1354f841e7476430d45cdc238b554411ebf5068baadb2 +c1ec95bde4686e2705aab12440f1bbb594ed37e7fb3467f280ab6d4851009334 +1997cacb8111f42e2759639249a6cc5cdbf8be60446156f1075f4a3c07ba318b +7563c40550811f0c61f1a4d3efc4d6bbaba8336b0c32866bcbde4356187972e9 +7d2ec1dae96bab0f1d9e7c1f757c7e99b2ae358623a5dd38d1942f48ecef6c1a +a40bff04419e0198de71be67293d6098119f76bdc9111f56f45fd7774239a8c7 +d884a04e05aa9d2c80274e32876234b7a512b9aa19d8cc38d31d5b8914b4294c +0c7694679a8a9c9e01797fe6d50b205f78bfa08c103a6a071f254f315202a472 +fb010de922a325f102668f4857a8122af2f1d76d6017b242e6bade87919f897b +b6160f7709397675345b68a1086e69fed6101b5e88e1cfbf8dda98f9e5f47c2b +eccc616bccda53868b317cc2574238e06f132e1d2e8b7545ea36fc7fb6968506 +0021ac86c40196fc165cd50a666c97c533e6597abb8fc58028df700a9f1b77c3 +935c8162fb14a2e82ab4e3e2ba0d136506da3dabf939f6003fecbc34376945e7 +b876f3b2fef63386a8e1dc0e303fb1af91d471fa80ec379cd065b27d2d79b691 +d1e78f0babf79268778d6cf15aa5d366487bf62287ff001003a4ca8efb704d87 +aae418af49fd87b79697c821b360f5468775f31e57f77f7d9ef65f3f734d1196 +97677e4f0f3a121c7d76ca49534b4dfb4f5312da4509b1e4a0a569381c25d17c +85c71ef80a4a32009062a8448513560d9371308895026f04c1c2dd041277385b +80ca6ea4c6d68d687bfcdec1842981e3c40dd39f9a3586b80c6ac59af1949b69 +90ce1468ac3a3da1147287f5e5b4b50af79c8c91d82c95d4e513b69c37ed3b50 +5e92e851331bed72c2f071622f935b13bc546052ab6bf1d93c4bedb1874e4ef2 +06df0c9ccb11887a2407f093d81f9d37ebb1dc147d982c99173e318c4c5b9733 +aff9722a9867bda2a1820bdcbd0a4ba406024881df9c688d536924d4aad821d9 +d876e87acc413203c999db930024566b868fea8d543410edfbfcdaf0a4752ec2 +0121370b10dc8703b4e37714652e19613433d9ff559d2f8af0f12177ebb6e9e3 +b732d6cc5e24f272d5a0edc54e3b631c09305156bee4fbf22d93985a93c4e9a9 +d80ec2e12215c4de6bbee244173391c7af4a91ec9d2ef352ce9cfe3a565f9be5 +a35f7e003d937932767c798c71602c5178ce5d0151fcfec2043787ab43ead2d9 +06414dd9c015a9e9e6cd2cc81c7067b2ba81617cdd559e8b5b72697736e7693e +bbc3dba8d1aa63c9da777b6bfe06444bbc8df6411c60424fc30245439a432524 +99563e2c59348f1cda5070499bee85ab8a72635aefcdda13c8454b83ebe010cf +c7c0cc6d879fac3a5774bca00cac0e8a56c52c930e51d16c8bb45ac3f8fba3e8 +f3ba337c544ab0113acf18280f611f798883fdf11456104f648f364b74baa09e +56b9d1938f7155691e95d7af717e2d313b57c5f06382726f3690dc5e26508945 +6350ef6541697b2e097886431cec8c2613dbc28b2536301edf7c734ab26bbe8a +4ea0b5fca7a36e4ed43aca9028da1e83b5cfa2fefee99590134972f51891e6db +76d69630d85ea660c6ef19c7e38e60b3d0478ec116f53a3f985746bdf054a931 +d0251a830a057750356e4e64ece9fdd57e02ea8f059b4843798da841883c3767 +9e8f6ebb1be6b4b917f9cb735c9c15b7afa94731d557cea26f5069bb612edab2 +9fbdf4acc5e5d03551a3a959313a10f2a2c6bce51f890344138db1e9a654fea7 +42b7511e313cdd893200a92d9a49a91a41321705737a6f0c2ec69e945244890c +b6f569eedce326d450d2c33045aa38049b62b21d6643cb1947d8bfa4a4e74d2d +d31ced8fc1e88c2bb412a77b38a5f9a7e9a373c3c951e72f8c50b0ce0f5fd9f2 +fb2338e8d3f4199d6a7c9e09120cdfe3ef0d63d3d61e9efd019f59b224f176b0 +be65e899c13d0f49ef5d4e6d5de7213078a5dcfb28b096bca4844047bf8f2b56 +0d50a2c4ea199f506b763db1722e23e3a87d865be9200b8c1ba204c765a1bff5 +a37221191dc8591155fed00abd44b436b3a666c7054314a11d16cea32650eecf +03108ea503040bfdaaa0a78e79ad8dc090013f1297a1731ac1095c0793166c66 +1e93897df26f965d16de3666a7e5cea7c5d6a826e4a4465bec67c934ef0df98c +efe132b854f1deca46e3358d44aa81087c7f7d3a0e19790a5fe514b445e7951c +d2f2b08faa39f33ac1a17fcedbbd8ab38a11f9cf9669b3739381bfc1513f75f9 +a8ac062730a37a6794828720d19ab603a81e10f3cbbc74ec2d224f0f84c86cce +3c1030220b6705236a917ebe0d4cac1e517d65f35f6a98a852b435475593b6d2 +3e70de0b0cab2d104a2c2d5a68b277f03f6b94e1362aab6f3d9575d3361142b3 +68d9901c3e1b60e588b5a162449195b9307a153766f2258f3cfa189ca4b5d618 +2f4ef9708215278a83b4afc481739d72fb4a32e0ab1e1de798218940864811ca +bb2b7e13abe856b2738ae9fe86d4209ae723cbcacc3d1cab86a65e8badcfd59b +d1156e5752876584e17b45f887a826b8c3b9a72fd33e70805c9d8f3f03bf81bd +dfe2e943a84d5464481343783a02923e7d6da31ac7a4d6fb26ca5c3fafa55e57 +27bb364c98a7970af301b85bfc1d924b71797d293c82aea7b85300b6d0a90fb6 +bbe7ecadc4b7e7d30742a567a4ede550851055ac68f85df78ce665bb90c058d0 +a57b44f9ab66a9cafd4c33c1a3a5419202e8eff3d357eae002039871e225b57a +91d10da010124b30d3e1994aa0bfa8fe270d8d1d127c3cdc7e33965e4b6667c4 +952254b6117063f7d1621155751d0a28e459b73d8151ff3718f74cb4fb4ebfb0 +da3e3409b821829d8f7ea7a3732129d3e25c0ce7285cebab3daf4d18b2e376ae +e442b167157b5a028dee9b2fb427411f6d5fb185e956d16644f90a64f42d14a5 +46af7ed7dada9334b223d9d70e3eb6290979bc85f78e8617411e9503716b6a7b +e70e7a182eb3d7ca9fc62a2bad33611f93c221720a04d432821dd0b3da5469e0 +5ae5be014ba43d0054e9e5a8028cef55ef765733fc42c58d91985c4a1a95c980 +06531bf411cb31b669e4b6ce0354cfa2c2bff30d55c33553c6bf8a4da946e20d +91bba3a17be20c5687347a4614318c60880e9604b95cc37566dd1e05523e60e6 +c97f8c6cf9105f50e403e596f45a15bff7101935345e9f77e9619607cf6a9d8f +b2a2e3361a1bcdb30cd08d9f9a3c735c613ab0213240c535356136f00e314cb8 +61ac4a854c3f42511a623b93c460545229637798ec259ff4aa4e4cf339234282 +3845d1bb27a66a678f219c001276cf8a34421dcdcc04329affffb1f487bdd6b6 +0eea2721c97b3a58caa2487ba8b09fe90ab3ab33c1bcae0bf6bc6a7da76da830 +daf4112917442679e670fb713dd7ce4447ce3a68d5b9494e40d274e03486e3ab +90c32b1af80079f25601acfa9357e8f5a4593f85b2b13da93b7bf4b64ec98bbc +5470967c00b7873c37abb69e8c634a97067287465fda9efff73f56150a1eb81a +e18f48c949baf9e2c257586a75e001b3e26e5e166146b87081310fa425e41bf7 +b66b4e672529cf7011463cf77fb536dd379cc72ee6eb89884f9981cfd1ad87c9 +4caa9ff4f81dd8ccf860e7cc2918e537160512bde802930d1380db5cc1d75da3 +989ff46f81ac81a3b97b05f91c92806ab0fbc0cb6558e6ce69b38d6863d5fd12 +44f395080a8268041b384496130cbce63e701d09c23e81e1aba24c45a5a5252c +6be7a5e610ad76a763eb69fd4eeecc546b1a5bdb539226a73d857a2469d5f0b9 +6eef97c80d9e975c678a26044da564a51e5850b88630fbafb73204a5f7e34449 +87a36b486397703f39846e521f2e1a54879a8b13d7cd70d4f8b416da6da2e500 +b121ddd4c5dc762bc6c56935919db200c0624f8ba9ca6522a2ed14e9109cd5de +30969f10e6206a789f1ec6b7a122b534eb85e23087d29273ee1d9370ebb0a5a5 +36474fde7d01b0014bd2de348e00862701254557e02b6b5bbaaf3b16f9b6208a +6d20e7c653c63cce9f26ff5c0b53a94da8cd427448a623d3d5302c830c49f165 +76dc1a816583f375612b2c018f923b5d305f88ddeab7901f1861e65b934e7114 +c26aacfab1d28b5285be0ed573ce035ea8f93a8f90d00d24b5271e8499354ed5 +a9c865c893bdeb8c1c5968e0e475157aba5cfeb1814c020c70eb7846cb00f8ec +fdaf8b0286c66577d0f8636e37f341e166eb53cab6909f844d2921e75b1f1102 +641b5ce7a7989b175c0bbae12d446a66bf955fe43c159a882fc041af9c88b8d0 +46ca3110109a70f29e6cc81a11a9a9df55340fd99c22cd2073a745b1f8951e81 +e00e397f1f27d1cf78a68252e1c4c8e491cfcb9c6d31dc40ab15a0aacaa7b12b +861227c9ba2fdd40974af9aa559d538e108defef7d8613cecdca65b10b6827af +46ae20730ec4b6064a25f5f23a35a230f73edbeb359e82f5c01a3eda8db11083 +2aae23125a349be545cf1b1220b3f04ad5ac648de82b9bc62ca678a1fec7d5cb +a5254f0e7f9171b5bc979917ca87f6bc8419f71fdc3fb512f37965df3b9420a9 +76dc5c0790e7273d3862741ab570682e42cfec9b101636b7d5df76b504bdabea +3160fe1c2f780ba53f2fa2b44a09de609936eed52435946b0139621bce5a19b2 +461570b1d4894e0eaef5db795ec75726a0305808104be67e3022ac55eac6159c +e1fa9b3aeea63b028d47072444875f1efab1d845cb3e1b9c605c7ec430c0208e +72fe6219b3799d8ea6ebe5f35b171e189e750c27d17f722503c7840b71eea69f +d74ca0d0ec7978746d50c18364e958d7fae3c43f007f1ffbb7cf94d56036548e +005be5e0e9dbb7f0f5d792fa412ca86efe6c7d6c6e014b94d8ab8840db096178 +dd54ab1ac66e9559f1a0bac4e763d37586f4f8d7c14aa0cb22486b6a76a1ad79 +67542bd9593de095f47c9ce1a81614bf972b50de08b190b9f65f32c2433a16cc +b73474d1689027e4e65dd1bd0c6b0ac279101fe5f8a1f03820e11081c0bf2728 +1491212fb92deb287f3e1804cec650828da91d5c86133d8ca7539cdc761ae0e9 +5328b5afc54ecbc36d720317b4d27bf5ccd383f35074592ab3493c8d76ccda62 +836b1602ab6e6c798cbba19e82bc22ea5887c64e01a317607f809e9c37ebf984 +f2584170829e025e4682e9225cd63ae8b81674dd234f0dfed03f709f0dbad9b8 +4ec75bc934da27bd34f45b6ba301404ba0d686381e37f46e37b305c37a10be35 +38c79d5d70a78e7076a6a99c4650ec04dd72df1666ebde0f81b377bc2ae5bf3e +0ebd5b7a3cbce0720ebfb86c41832c91e947549aec4fe3e9bcffcde2d83f63a3 +dd86ceffdcca7a745e868592cdc6c11266564f319ddc28d3614fdaee8e9550c5 +e353c3a1cf90d8302d94052e3d902592a105928e4299c6b940d707020600f5fb +8aa951de6e9f5c1b616c392308e5d933c1c08f9c72c7fbe3f5dca11bce472154 +ec033859d8eefd57e5cde2422d604c50bd33b07973d899f20a228ecfa6f62725 +3b3743cbfbfd3a7fe8e8360c3be2c2ac30a0caa6f19ab6f4ad7488749bcaf3d7 +098c2489bdbb5d728fd74327edf0303f0a2185bf5871b636ab9322cdfe7e73d5 +2f01956bfdde1fcba2fd926ab53f910fd7c53690000ef871dd333c6aa38d4c01 +5e1c18001fe4193833f8ed3cc40c82eda65b6d601252ae78150d559f806e39f4 +32ea519c8a31c99b41507e8830a1007b477fb277aeac7c0e6414d4446d550bc0 +23bae6ce16593f84a6822eb4204f7e7628d506d147daf7379b6ae122286f4dce +8b3873b02afbeb5df225e1fbc5efe2c41b38013ebc4225ea3f32f50ed36cbb2d +290079b62dbdfcfd0ae51082dc6fc7cfe4c22603dabc72083e9fa8b9ca54845f +212924687a58942fd23c30e2e42a802cb00837dc043956bfa64b7ba456235cd9 +6313c4ea814c91ec72c3e2930dd9370726a5fa41615f8f12aebd34897dcdd4a6 +08d65b4ff1c2336f80ba3780a5578e129e6805379a340489aba9fdff0c4a6e13 +492c8a34b81f3d21ccabad349befc1f7c81dc54b68e3c42e60830d29f1d069e1 +6d28b966b27d9e93e6239a9765d980d1be5d56d7443a4bae10e132fe8e66b7f8 +378952e80a8b59a6defd4f38d9269c0cf89833c886ad0945f715839a1633d27f +2f33026021780535d8ddded4d31cbdce2b986d88fb11585e39bbbf0f563aac75 +f948a03b285b8fecc125c42ee230e5ac22228263189583c2d0a572e35496fca9 +bc3881b6b812f725e4a08c48b075a7a7487eb8c6ac04d345125467e695455553 +ff3fafcb58d546a36fed36931daf9b88784770d5aed6a507d78bf08d545eabd2 +a7978b6d09ffca907a1f69dd3eab1511e8363a2d3b3dc9c5da64c57d3c72da74 +bd6c0982cdf3673d593e51429b6e00aed22caaf5d542663b47c168d7b95cbf8d +0ce80f60e096ccaefcb787ef0ee3cf7bfbf7164649a3c7e63c4705684d373045 +2bcdd7448e349009433ba4a6501b9feffabb8db16c4ec9329aa6f6ac45cc57fe +b0f323ffd5563099f1d9b08ac7a8a18817fc4cface7d7493ed040b37e2d96eda +ab539e7a9f45cc673ec8639ae1a8394f002cd126ac7a51151a7a5ce6e8ac36a5 +d9920d28aa499f164621b167ffbe20687fed0f29e36ea1fbe254067913dc1ca9 +54b3cf54dc48f939bf797609a3bb9aaabf8ce2cbfa5e1247f5b74d24521803e5 +6da0c04a0f446599a0c8979dcd6285eb7f9ccc0ea3f9fed5892a14371a31793b +ad3ab415a409ea6ebf4ffac39fc079ffcb68168064d39953054c62b631cc75f2 +600ebd36e6c63f625603a24b986038550352581f0d23f5b259f167d7af0fb95b +34c7ccfa4e8e20644563b2651eccb64d7bc31d3e8a681bda769e7c17b9bdc881 +24a1e103bc83dd7bf1ca8d9515553717896972804638445b3ca9e41a671f001c +7c8d5b4f0d75bb55babf4cd2259210cd05a62e9a839ddd78c9d6745686ae53c3 +7494ca987f52215a8c4b7fa34c5b7f055ec0977d95753abf105a4a89e36398ad +b0593790d7ea4ed74ffff34905913da78345a4237a449839c6f8eced37fab135 +a4010714cb1ef94d5a42d0800c5ad314f937617c5a7c384399b875c650246db4 +8759a2ec09c960f7cef83d72ccdcff7cc33007dbb0b694f50cdbeb611c04b3e0 +6953f23701a306b0893a2d0761544407759511c80e057c113b2dd260046c7e91 +03aed9b68c1b79deb9672fe6682e6ce3e12041cdf480235dc923c6fcf86fe03a +e580c56a0d3a5b2596037a667c7d17aa6ffbffdf80c26a9fa30433217bc01487 +67d250ac876136ca23990b87f37ccee6390ff7ddc73cfb8dea8133b0d4d9783f +0031da6252125a1199f740e3c90c45359d27c695c6849e537e4173dccbc9e1e4 +485cd6b0eb17c59ea62f4a9feb82e0c9ceb4d395e711056b4814c4ff5a99e176 +fd8a839ebe70c444c15274a6c6baa10acdc71712e47d99f1e7b0264b100ccb28 +9e48f83e5ce4cd704858f5185bf4eb98ab9966ae0f5fdb777362cfd8c1cccedc +f868ab94dc48f4dac8ef3651ccf09e0405e6efd7c7f804d726d675875db6c9d8 +750a27adc83d719ce863303f4c8d7f74a4ea83c95fa092ed4c2faf26a56f858c +d469180799a6cbbdf54001fc041b1b2fdc4e24bcf1f19d1a60bd20a87b9537f3 +1904b89f195d8d617f8a416b80dd7abc2c1ac500be432170bc509b237c5d1283 +65775d2f215148ed213fcc5587d35e09a9561563e7de65a08867e7da49245229 +63aafbdc6470b5288f70b3ac6063a59acb0a88492f2560023a4112c2a77f9795 +a62a1dd31befa1df1d86b3b5a4d99f5d175367fb4a3784af64bccd2ac3937e3e +04a491f51552d679e9821226548aeedacf049c36257137cc14c66412fe755c61 +e5576849d5981f76d3b1d19184110c8991132115dbf1e780e7df0a7241df5a8d +d026b3a4457ff5a99ad28ba834db8952c35e8509efb0bba5cb45eee116df70ef +a73182747492d62c4adfbc9ead9b8ce7a03d4882094fb227633909c389781fd8 +ee0c6b15a655f3f62dab4ac16498bd0fae4974cc4cf7422337c2d0e02b10e275 +fc3f5e03c024eec3b0963d31af92572dbfc5d0ded79314dee78717b37d4f8cfb +e125398fafabe2e5da03048b910ce12de3671f532055a59e2ac45bc1481b1153 +99f10c8fe5f1aba938d343e4388c5d3cd0aa06aae35d1b3784f43537466ddb0c +73d35230bf345028709c15a49977813b34bf6a299a1f8d4bdbfce7a69fd4de85 +8ad482e9fabf469deaddcb124ad27ba2a41df82fc811c03a1f23bc11df9df1b7 +d0fd211e1582829c4199038a3b80f81bb312a5592ae58ee9e815dc608c9bb3d1 +e4132f4fbe1a42d0aca1c5979f4326b4112003b818bc6965e8e4f01c6b6de5da +07b26af043c8ff5803af941fccb8380826f0b131c77ad3ebe13b7f1cc7227a4f +98f38fee065b764b0d626f9051e958efa95cccd2ea77707c2fa2f50df7c8db75 +bf0a8a673a9963cfc7e02a016c30f4e642de3e2ff775b15c7e02cb07d0a96ee4 +73416c4fb267f1ed00bc3d699a0227239ad690d12ba6c3f62e31a6b0ef7625c8 +ec8850d6175fd6abe7b3f06f32ecae3295d5d54b9276c0162b2e81f54cccefc3 +5c8007abd4add002d727d02cec6231adfbde7541eba9f4a3c2307fefcc02297d +0f74c58e39cef7493b87318a2e6edbc8cf34ab3a0e83db5f43f40d7cc7af0d26 +a4ea8fed43ca5d0d4f09daf5898b70e435ac5aa0469ad3ccbdd6c2e6a2f61bd0 +e8df671e19136a1f1b82ebb8980589e3c0fe4b97d13623ad45faf06f3169ce0a +9e7f81e22653a88204f7336615893eff2ee519b9c26f29090ad1acf271fab8ce +c6d9518519ff651a87388616ae16d528bec73775b92c8749a77110ff8a93ce87 +c1eb971113b9fac237f9f1d9736ce083dc203bf35274c1b8b7493b66612a39f1 +d7c6f9fb13f626b2bcf626537cbd0087d52bff5f299bebc525773c6ca3f23b0b +ec89c40bf72cc34ecdc6f387b431f9ed0a8f1a5e9b831c908dd7be9e93254733 +aedc3cc5b83ad3e2f6ff5729e7ccd3917f74917e85ec4f01a07a5082b2da5982 +f2bb39ef4c826a60505ec938470b9af73d0836ae5128ef1f7f086d226cb81238 +3ee3271720cabb318cc8b2eebcee9baa5863854bdca5aefa3d0f9b0f90b702ab +c8df8671f9668ac130fc7765ce316ebbf578b00a1c4d7bcd26f63cd8332ea6e5 +b8114415290445dde4ca10bbdbb8aa73aabc15119b30fc70ebd176c563bf3326 +a7370d812757a2e8e637cbdf33647cc46204baba770627116bca31a1271c6bcc +8eb1eaf6e5b17328b174bb6c2597f5e2ec3d1a9b24e68c467cd7f74f5b83b37d +6d591ee06e1d68699f42802480d586fbd988ab6abd7b911cff06b307a0aa7e69 +d3d6e988886bbf08e81e25ecaa89e1e93554973cc1a5a3f72de6b6b6b9fa5aab +e7e3c6160851e8b7bbac99ca699cea7858093ad4643a33d75ae8faf91f835bd7 +e87429659187d346236bf1e487866e6a1a0765195aba3ba6e9d9b5fe4fd7ecdb +3a104d3ade01ab3b7f3e29915c8e032354aa811d0476f4a16322aafcccea4e56 +9de08224dd9c48619a6573fcdc089303e1dbcb1c51e5f48aa2b83a7710b5b172 +39959f0279207ca7414d864aca5233d998d4ebc950f15a4556f4dfd3b1932b92 +60fa4a4157f7731241fb8710edb8af7291abb5d8ae5654905e93e6345da5e8cb +2c472f7d38dfeaf8bb8fad365daefe593f144f30b6913bb18fc4e0cef8feec46 +0697ec1fc33104b8fc13fcfb19e9392fd18f49ea8566c6dfbcf8180b2a26f911 +ae55af382d8a9dcb958f1414582092347166bb4b7bb1a754396e0bccd56014bd +20aa244555a00f1cf0f73d4eef596d9118db4aff90e348959795917617647dbf +e5145332793533c93b0d20d5eec0e9003c78458c25eaab02bcd3fdc66ab92ec9 +edf6637fa0a2d95568c8e67527e4cf15be07080ecda9bf51cfb9beb759c74f13 +71531db280f61dab2c5d70bd100850b5b348a2fbaae8d488cec04339d1262b45 +147652be2b51a411e060a0bf72a4bee43d4ae1ba1aad56ab0304c6f85a637372 +a80f68d30b4ed14f6f24461cf9ef99799d651e8af9c9bb218f343449e555ca74 +89a53ae8789a005d9e17c3206fedc25582a87977600f55da02f842e0cc1f4980 +ac4fadfbba8ea96d4371ac8acb30d3559c8f7c8de1abd372ec0bb64419cbb3f7 +6548415e0f89a6f7975be174d890df4bae8ff4e8d552d6bac3659227402dd033 +cb6956e264a5fd11c99a08b537f9513a13a2e3412b031dec3c01c45d1f7ed667 +7622523495aaf2e35c50bedfb81e0f246c72d5d82452d3bbd8b70fa7bffec7da +e49c3bd519d0b13d795f511b63b2ce0e4b57f5a9b14f3872e40dcd8dcb65e638 +c7c4244bea1bd64f84c8aa9acd7f3483afab2b3b58bbbde549b9d5dec124d3f4 +eb20dd7c5433acebd6e1cd3d8193feae63e19312c0b08558151f20e6cb1868f7 +8dda430bad837cf356ebfe7271a00f92d4802ba85a2aa4f28f5fda02eff98338 +5eabe6b4d9ecabbeb050b2e646cbdac4e265e6b40fec3f35eb23834d92505bae +3bef08d2589d3f05b995800795f75e3b81d3690d36fd2a381573d6c5d4f13bb3 +e7afaa3577d559b05eaf5db73ab7925f9ec68b72efc59d4a85456d006d2727ac +3e403252010e3f1bac62e325a318919253cc3d47d0c408aa4fd2ee7d81bdac5b +4f5078f08730d54999066d3991efe2ae575e5451d98ed0856091cfc1a6fc6da9 +c6fcaf5c357c28461cacf9da67723aa6704136a19b0615642fb7409c5c17e38d +620f69a5df706798affc8735c789242c5c10b5d0f44ae3de687f9178ef8d69e9 +6f7288ced692f20b9eea4c638c5368846a279ddf857992b41e70ec9c1357f8da +f12fcc9d8b0f8544e8e45094100c678d32d842fe91c29a9f0f6e187ef7192a1f +f690cded71dc52db6f9a6540a6883c75cba2da5de472861beb8cacccfd6d6090 +2cc9123974ff8558289f0d5e4b50fc8e1ebc3e5f33e836fcc074834bf84b45f3 +e0c9233c257a3dc6acfc04118472f9c55623cba384acf1d19362946a040ded8c +dea52eb1777e41b812657b91b2e078b8b2cc39b3948b0366bca3a45821902481 +0a403d28dff954a0ff41f056b1c1f95315e7cf337caef05a9e2e7f18e62026b9 +ee967636de2477ea0410a16c7526509f2902d52ed1cce379f64256f0f3549d50 +ae2f3352a6262c1dde90c3a67b7401c724c00b4e25fb8fe6407508d879ff8fa5 +0fdfdce7a90db27147ee168103b36985f61ad43065fe6664951c5861d361c5aa +2b5e9926cf8abaac9192a405131406b0abcaaf27c9ea8a91770c29d92ec5e43e +7077c693d8b4af463781a7b7df92ecb3ae8ac9d1cf016244327d1c75e4d60123 +0edf9bed57bb4d663c690e17eb20692f0339f337f3c8f881d9834650a38c4b62 +6327f69509bb03c4e065cf7776ecc6ab025aad72e34a6c3ea96dfe66e0250028 +e09e03d2f8c380440752d1241742ceb50d1987b8fd07722787307c5087f59ee8 +b99dfdf05957d075cb84953d0211da38c1e8901cff792f39e25f63962ed45392 +33a7db843f094fd2dbd59c60df2d592091d4d0fb8ce152eac0c9b3c42bef46b5 +da0be3db59c76e336e9957bf859b76cf86fa9cefff4963f3703d82b55661a161 +e4db2d84306a2ca488ebe1d8099bc0f5d355571e0643c02eb7030b4afa2aa971 +4a560e22c006f2a9155795ad0f04cfc271405c0656050b73b07d4dc5aa7cd1de +6965423c14c817a42affbf88f9ae2464a678aee1a9194327b49368e5b5bca8c0 +d10c32a5d1137338367c32454990e4b98f6f1a09d82340327d6910eab6b75e85 +6fd1af0ad43bbcf728184c6bcb6a48f1eedc2bae0066a9bc42fa9cddb1de028d +4e0b8bf2fc44fd3fff3662d5544d73cb23358ffc981e675d04eaa7a31e9ade9a +e29374f5cffb03c1b8eb54e183fc932226ff0037870fa562cb804832a1fbeb57 +f561dd555e9726ed474a1f5cb1dbcaceb70496264f485d77372771d9504c2e77 +81958fe6ee48c9989416c1b433ef4f66640567d1a2ebbe04180724b603065dc5 +d5df196a1ed41e50d0c8ec443d102d1f38ea5b39f057e9c43af6081fea8ea482 +6bfc92ab1ae8323b9d3e8b7527264a9222a03364ca2595c0007c95c5754ebf83 +6df284f006191e823c1a496076a7b45e8562f840d53e152b4c4d335acbb9b42b +1afa88e19b848d95de486606f833ce6a45bc9ac684f2f8b762e136e96f2699aa +180ac14dcdbf3759430aafcac0a4b28201914a6b2ed15bb4b470cbc9b7244997 +a7df823f41af9f4b16d14076776680f0a0b4992672e5dd48b55751ceb91687e1 +4912b4b39370659b75386d6a974b67e4c7e5434f1604912c6b742b76c5f80401 +99b139a1bfa3b50973c882be74108261f201cbc20696f1aca4dda3547fb83628 +139f31c4fed4bc91c3f2ddb59ef81af7feaba999693771a91d2d88578c7dfa4b +87c9c83a2974fb47023abe3827da17428a4028192e38b1c89bd65c1e196d3e42 +3cb3e87475c541029686f32ebffbf7a2e512e86738a3cba79f7ca56c811dc06d +9acbca60437a52735c17a2f6642bbab45a5fc2248d19dcb7e3a61ad061e246ef +46c24abd65f4aa55862c73106794efe647641219d217018ed66df8b5f718a971 +216e83ee1fe7abc8d89c1b0816ad2447ee9f76acece04aeb004f59b4b21468d1 +ef415ae3b56e816fe2ea5dfa42ad7d4f0a5ea4518ae256aba3ce3dfbf61a54d8 +818d3ac49eb25ee39f830dc820ac905920ed58657f0bef5f19defe7b6ce62925 +34fbc123ef45b39e393919cafe4d75138a117d8eb96bba48506f4f0fe50592c6 +d263c02f82edbe2d14b9044902eb3b9abf5a2c0fe52b7e7abe5d83efac857733 +23c1f8ca022dff9e586aaf43743916b5cb5b45d505ee995b55fa14560bd22516 +e3e3efe45babf86d2a23dc93592c1e707963c604615c3eb5b5e9cb7b6f13cca9 +15509b01e5ee29fcf75399ecacf7ec7036b6178d64dec1b5c930512a116a692b +7ef71ada196a4fa0287d09f60852b7b108ac663cc92085f6768989f1be64bca6 +aed28953a220d0320bc33e45b1c7d89b26670f6fc630f00eb86f579cf03802ca +172224957b79af8a907b2a50b69a55d4b3dc77ed44ab7ac91506e45579a5d0d7 +9f381711c7687d63b033bb5923520b32d71d08c2acb7e6cf8744511b0802af50 +7f24e59d0e7ce3697888ace50aafcb92f5f55e8cb63ad9301a090408fcc4f6be +e6c629cbbf8a73e8603b914d5b2fb62c74890f8fbea75d20e08c3dfaae937ad3 +9a7b6bfde4985e82f671e75262aa0e20512e083ddcaa54e1cdffe7b49519a3be +fe19763d5b5e2eb1b38088afbb6fe6bbd55bc613823f37311c9034fbe2ced354 +1e05deb3e9b1a05c5a29b164b1593ba4a0672b186772bc6524462a81328cbb0e +4783725c77e2d0e90d4ae4656c69c1b54a264685a8d0180d8b412397962f2aed +8e3d980fd5f51c7c473ba6a41fadea679691fc87956940a74037a34511869fe9 +99c4347dc561ed023c1d48e7826e3f6a0d903c95001e6e2e1466a3a280591786 +aa7638781f16b52f40d2e691550bf4682ad10f323584ea56566b007d1580444a +2164dcfb6fb3fac50f08cb48e33d1aecaef232e0706e7986ded1ef3bb12bfadf +958e82458f1384fe0fccabf06ea3399e8522be4b5ef95c882ecf932f33b06b68 +e9a4487734fb9e8c064504cf71e0ffe53020d97a3cf8f19b9678e8a3e0cebae6 +97143a98ed954698e0294fe1bc4a72887e66440e273c48e860e50c6bcb1bef90 +1f68a97c6267a26c608e64d37fdb3be6844c75ba8d94c860beac5011fed3227b +dca16444e7da02aacf3931ebf9a4ebc11a3ba8949cb89f8649db794863a18053 +c4fca7284be57bdf4caa683c351f214654c137bacaf7bd0a6cfa966f605a7cee +3c0a15188d149c3856cf0cfbe5f8f64e52db9608c4b3f66b8d421688390bd9cf +852cf9b19c5b25d74374b5d3ee968a899c30d3ac33be1673cff532f252009399 +dda05acd2c6261bec3a864570b65b485283c16ca76e81ea309c445d64dee529f +e6d6d088eb9a5ff76f5a0a6f3e6193608c3e59a69ab9b734e3554428cddd0eae +3744de57025ea32d934798ea141316675594a7f8e96ec2ef62e5b531efe9f249 +8d8d1a54b84c50b98f2b7cc7629711a9e8545e31325a6a1cb88321d3faa25b02 +42a310ba16765c51c20de19e1fce4709bff69ac5bd51dc3a03011fa21136d333 +585c16b4debabcc57f6a55ae79b6a366e3b609b50775e2fa7aaa29fb308b5c8c +2367f2a6655220c19cb85753029cecb06c4bc284bacd88dff4d8e91ecbe0907e +c2432559e7a6d98f5973ada70913faddd976f702da2c051284bd9f7b5deb595a +c1942e58a1e33e8c339fe8192e475de3edbde70ad7d615c0c77c39af8960a6cb +7cae6e6ea152db0e106164f91883b2f181c636286e6aab59b6ab5e46ff692820 +a1501250daf41bab535c9fe97146137d4fbef4d5e19cf9716c6a12087c37647a +83921e142baf96bdcc921e2c91ebcf3e6b673e0bb95ebdd625ecdc469adb6920 +a1eddf5ff3f1c5a760ac1dab80b3220b8048d4009a07275dee306a3350e905a6 +4147526b3e1020ce01bff062c83eee439e4926aed7b3e9889d17484f67edce5a +d9a514de486ddc2e049224b8a121aa8d54009c6468b45ccebb94846aec6c6b40 +c2ca2ac4023ebf6b1949654423c03b7b7f75f8dd7d08e080d9be60a5f7852d7d +2a86d46e4403c4c63ae49cdee6e9e28b0eacebf2f1f3b6008d076e12b40fdfde +5b2b3f1ed7857040d575b55a58d32b0322a1741e67e6d2c6072ce178f26c8f46 +fcb26c7d17dd0a981209893db9283fe19f2d4ce3ea35d6eccefd8db3c67000a2 +2e5dd132f996c9eeca24d4663f937e5e5935b896b1bb696930977cb0ed722e84 +1401025700fd9a9e477e39b78282b1e9929920395d9f4cb77afb671af0f740a4 +9d43a58596639ed577a208d86eb3ff599972c0db25bd8f37701a5fba55105100 +7a41df7255c7d308e663b53c8df7ee8da507079f20c599274142e59721ea21d6 +825217c06a33d7e84618c185fe02ab5d03330098d3e3374503f5e5be4a9408e4 +c82ae7b8c2f03f7928111c36447b75900acd3874a26cc7503370902dd986064b +b58d6ce4b4d10e1cb7dd03d0b305c9e02f723760d916505d4182a1362813a1d1 +7e06d11c611a0cb3fe9934165cd12b254637ba624ddbfc25439e714cf6bafe0f +242f8d05ed142a2050ffb131e55ab9f85eab3460a6647176804334a69cfa48d7 +8364022cbc702107c5f9f980961757187a20eed12e7ff1969ecf13d5035a239e +a6f80ce4c354a4b16b33e35ab36e6362f6f419d018ab16dd17b5964cfebfcbc3 +bc9be08b0b1dcb0720615fcbca3326e14fbec3bc280f80dff17797d98c51b761 +4745a9c3ae1c1b4a2ed9ae0c2e9d476d65565e6a44454f58a33c82485c99b18b +626a3586c13085feb31c4ef0d19049f34e79f34e811aa66691642f40d98c7a1a +a41a28f7e42bdaf72848ad408e8c6d2555275019d3a6391f24f54335b202c350 +9320514680fa08c400c18ffe3de8a7378add2ac7c5a282fb8e8a5c0d2605fce1 +bfa0ca24afb007e9a169c627c8c15aaf28379c091bae5a5adae7504d7d405cfa +b446a4e6f04aa5782dada0957b6367a6222960364b8b627df23cd60e8f74f3ff +05b0ab9e186a891133458510c8dddf7912bbbaf930637f0467247c0229808fa0 +de26b797a9361ca0bdfa68844dfb5834696a566719d1b648a2c3c54e4cb507db +3859b8c2730121b0c104826755e5d8be17a45f0df54068a946b6fd2767a0a4ff +ed8dddbae3333dcf0faf8bf676f9d65ab3573231b18d28922e089003618da116 +f46414f0f299f022eba28d7a4317563a12b220532a419b80b99ee1ed28f6d998 +a8a98b1813fb488d4561f0e0d6261cf613b26b676df3b58bec889a65f7f1fb3a +0683ac705a28e52cf290a07d9572bcb75be656e8a83f45666d2dea9b7241ab76 +b6109caa4fe3961c82f2f1ce0209e7ff15fe272f240bc7348314232d91e4d447 +5d59da2f8dfe6e6bcf5bea27674891a1f70d1f37bedf5f87dbdd6e87b88728f3 +dd6bf5c6dd325941405718734adc3bb5a2563233a6dba612f56d9146691f73ff +afa9a2de1ca70b25c26b63a8a6db8fe3a8b6cea76367af4797ad9dc27365382c +af539611fc84be3fb44acf5852f01f1410ca1f599ec29b01ad5236e39280d277 +4f35f9988802e1907758014c1acb76b332fff721b547056a9aef9ec74c968557 +c98fbd90c7ee9de9231503bcbd9724519b5aa20fd5539e13d83199d89d86eac8 +0a735872c8a99e84003ed853e8819233ae7c91ed6bd23942d379e2918be246ab +8d644475a04da3daeb324a5d3c6a3ba2c4c1ebb36a7b0985581bc249bf480089 +ad90a93589b4afe272455de68769433643249e5fd6fdd0142c00fc5baae53953 +dc7b72e68da13906f0ed524aab9f068b035fa74019d7153753f63e44225195ea +0c86c33183010597b4d25870b8407f139982987d6f5a93862d81f904aabd6ddf +c7388e1b8f441cbb44676413900602fbd58ac3771c3734e819f42aba2b31937b +fa966edcd3ca6c3fecd2e1e520a2e1480ac7b6f41b27a195a128846f3116760f +7a37ed6919d5bc567b82cf68eefa9d0103dab2702a60878f7f9fd108fbedce58 +0e366ae2120a3cfb879e8ca1c9b1e698a52138148b185bb0ffe5e3af2651a7bb +7d416d43783c830dc6d3766168f03633a482b7ae3c0c8a64c5e130f7f260ec54 +e28e9ac150fe846011deb5a39ee2d71afb08aa79cd5970b37a282bfe129ff0ff +eb9c004efa23db44c0e5bb2f82e72473d1d61bac19490d16a94d2cbd5cffad99 +357225d6fc5a5a8aad8916d35c58c05b4c00af05e3f5e9069a6905ca7f820d65 +b354cb67cab5fa86dcaf2b175de39e500b9150dbebc6b65fe97e836722448701 +b4baac3dd349a1b2adcae0e9c1250ae43476d3405b9834172879c6aa7101e8b0 +18aa3fd906ad29ddd2154bc2e09781ee50078d2b905214805e4d48aaf9e366b7 +85954b8070762e3a89333300bf650f3be61c3e4fc636ebd7d0741e1046b5c9c1 +55473324a576e966c7b1249ef44b7b53323cdecffceeb6f7790ae51549d59da5 +925c9e11ff464190f5b5b8199785683fb613f6432bddd20f435536762f97cb44 +319ab99f5daa2548652043db0a2e53198ebc29b80aae4b0b88e1ebbdf4cdab7c +4ea7ee648c9d55fbbd37b8ecdf6368ca095e3b8d23439fd0de17b23029ccc470 +7d73dbd506c024176c76d51bec6e89ae2d504f628412b53952df4298fb115081 +c13175c938e3f403e540909a5e11b4d78388a98b8d7394fc098b466d8cd41b41 +287a71724b0a834ffba9c00ea8ba62155b00629a1e0407290d51e7b6f5eea0f9 +656e9d8f71e4bce077aeb33889cb6bad99a99e8296d97e81ea18cc2c7cf7cd00 +997c11c9d0e3a5f6f3ff9c65b2d4457458862692215a343833120699271befb4 +aba137d9fa97658b3c2857b146e05927af961acfffa2a9394940f78df2ad2956 +b10fcf88e9ea770c7a170ff150188850041e14d2c5f30cb80320d2fc13152cfe +a61dad5e7db4e0524c9f6fb351296d1af9c1118fd1e6bdc257b3dc7c48fec71f +4e1bc2a7fc422ee2ee6a686cc58503f397cf898df143d1759df70ee5c5a8d411 +243f0f0459853a3a54bc75bd3d3ef4d775e72397a49aa8760b3086f6e7a72184 +495dbc4d8fe0dc22d8e811d041117b8ca2797f8f11c2c198ad11b54f42bcdc2c +f8c8f8db9c402362dd6b5332e81debdd0e4ce83ae4224238496d9fdab730b245 +1033cb6409b4da991a641fb2474977f2f24496d84483c73c259a6e292b01a1d0 +22f1bcb84b2607883f23508a4b014ffe61aba9b5b44396899840207047c630b0 +04f329b2c220448826b3b58d03192014a2bd6f8c987e5a55dd58609b258d3432 +d5d476b26e47c3037ea25a558d36e54cdf8740d868bc06da5befb28868c9ca52 +e13e223dd787f78182a6259b8408e59062a5dedcdea20028d600d111de5b6390 +8564b8d6a3a16b00c0e986c0fa030b5d14876d5d4a4ea696885e524b5bae5bbc +da8966bc61cdadb5845c8dc43b288e92c0bc742755969b349fd721d6a38af544 +68c7dbf8ce6a1dc995d91a9849bf61541d9eea392eb426f758368145e8b84c3c +bd61373a85e59b45da0eb15f413bea3c8bba8e903abd72fc0ca10f6adcf29429 +13c644c4dbacee307d30d6719879ffa546756aaaa3b29d1125382ecb46474b1a +613c14fc3ac3504318b01f9dce309190e989fff19e5f10658bdf7364515d0e95 +71c5e0578f18c9a558627b38e660f590c3eb3f6655e144e7508e0a79ef1db3f1 +ca57407f3a2cfc4fb8def48fb2bf92dc9963f890971d196d1221e0cc67650fa6 +05a5366a6fd57e3a3508eb6938aa4633dca934c79444adf8dbc599feb8213e8e +35f36a6bfeef16c447c212b94a9fd039b1d1ff638597e9aa17330c373387cef7 +42fb4252f03aef5d460299d694eaa45095482fb59282600c4095a9fe3aebd2b8 +ab980ed336134e2b8e130e44b59433421e5ad9a0af0287b58f71ccc5d450aef2 +98f4a1691f83beb110f3bc2b726e3b4542a3eba1116d57820de95393f93cbb21 +6499ff42493740187b1098a7ce2f732c150fb65918542c3c2e3989f074f72d33 +01269a7ee8dccd5fb374251a7a299a23c227ed142564503dd5c10942ce0065af +aa437628638cb98163f14fc91b2029b24aa19303fde63e9c33a509c1a9a7b62a +15750e413196f28d589f4ba75c0d21a15c465d2b68a1ab068423b378015196ca +a03904ea8531b30af884144712bfd7c7ec1d7ffd092aa81efe653fd9feebd99a +2fd700bef22948f7754792ee9418c582444135a685e3b9f7d721c633035228ef +c587bf02524cd4a5dd529f855ee41192501255fa80f1073e044d678ec8865ed4 +6be90bf1e61600d90e8eabaf79ede8c079c751b4c63601fa53d4c937253bd722 +26e8b71a8c7c043ca18cf83956ae00f293d397059c3a028e18d49821ef93a10b +67fa4b87d1031e4f68dc757dccb5ad5b9a9faa1e6025b428b49cc63df813ec41 +2383ce5d0e87d094815e0b232550347873cbf7ccb679581206dbff8f98c57d7e +14cc9d14aeaa57110bfcdfdd298c91b03ecd52e6d4c20f55cd2c6b393a3a8a97 +54ad84f9eafea39b70af34512d1efbc4cc44121a66ea2f916912ae226006c0b9 +d8a9360a4ce4c01ec2bd1f7e14f26d7ac70eb423e54acfa1b00d8116c0e6e974 +6da086fb73167e54767d266fe80d9860b10a80065e970dd85873395e013bfcda +8de5f153fe213f18c4cb691414a970928ae26e9ee682106234b62915e1db1025 +cbf62c4d4c652c932ef0ece58c8fbdd616efe3c138bbc768871e62ceceb40e11 +a3d3ac697f33ada49392469e66a14204de4464b23ddef2e6b4cdc212dc43bcab +8f946ecbd72c33a04d5c0a0796bb2878c90d3d93018e13422b8e37fd6f63f0d7 +13a28d790145f1c4b9d5fb7d17add3e2487b2296efe2d12b528186ebf71b873e +9d34c9b4cf094d63b2f332725b20376d9a2f36b1da15e9ed094102d5d7fbb267 +d484a803815e472112341e6f11633800609fee9bdb42f52ece0d59ed89ea0bba +c1f4c4e98cc4c1a1b5fd6e17953ff36525807731a03bcb3af3ef5e6c2a0c35b6 +167feaf0674f7d6e99943920438688edfbb2c7f031e0aee4b03f2843c369222a +3415272952cd2403d3472d2796ceede3a698257173ebb32c88e7a8cbf07044cd +6e0aa90945fbbfde4c43bfef3caadfda10cffcccf715e1158a15c6761857f938 +bd988dca987a913a0264e481bc2e8961668a3bb29f51c42cdef4b74c99b561dd +5c088b8837d4b3414a4604358ee7ad7ea9c26abf50e4c733de1777a18c1f744d +301c86e6b4295861d2e93071d7d7955fa68110befdb4cd626d68cd1bcc2d7b9f +981208ef73bce2a891efaeefd3dc97c8b374db0714bd0b0fe2348b5fa507865a +b4a11bd6b1642cdf20391b1a0a23ba5aa438198f9d00406f4f0df0a2cfb4f53c +14dca7836407782411d2f9d7f9b62d770e0dcbc43d30d417dde6745ac00de412 +855995aa6cfad55562207d5fc8d8c644f80e61b4699d9810cddb8025ef40b0fe +4c6d13fcf37ad1246690ca798f7f9ed574f3b33ca634bbe2489929a14a41fd06 +e19ec8ef252fb24a7f74466d93d7544cc5bd017ba084a7c6671a14f174f8ce66 +de275dc5a9ad045a295ae98b8b7b5f113e79e778590a8a1925b597bf0407409e +e603f52844e2096a8ba02de86ff4e868756079887c1de037b5ffc7eb04c8a1c1 +dd074456bacaec2242dbe390063ac3a60db054b39ed92c479eb8123cb2b0e256 +1c0226de2a17981aeb5345fea922e10feb5432f04b9a9533524b25d2567213c2 +4f1df95bcdbb6e0e161608439d10e34b4e5ed9863d2eb2ea8c07fc6137555cc9 +ac1a4a40be1a14543d2579c2eb0cde8c447a41c7f42c63743d7847a8fd6a5361 +150b2939f4707c283545fd4021fe3cb933742236b58c63ce3ca9aea7aaf38bce +237da897b97bd740690f192c8b644db259d12edab83fcbb4acfcbe02444ed831 +33a43c58b5a21847e28e57eadb2778468b03888431c90c9bd12219244cf347d4 +2e642f5b8ffd0aefc455ed55ec83e8956cf660de757e040ccf9a31039807beb3 +5aa9b34078c144d472bfc662b9697de939f0eeb6525c78c5f16bb8b1963e9eaa +ced856f750aabeefbab75dbc00f528490732bbec8bdb0f7ee4c429428c489d03 +41c96df8b72b3c9693ded032102eef5c7d354772d577e949d061e9b631f9a2c3 +647d82a668e9c2bbef88a7395bc0f28932b1d6e488093110219b948716191dcb +b1e6d8d88676c87b6f06877c12ee1a1463e14c5c9839611ba7de87d752f080f5 +996e281a807314354c977a4fda773babcfbe9f39ae515dd0491e1f988c84e1bf +889bb6a7d582e0a3845a6c0062e5012cb3506e70987cf359398a9d3a08ff8c22 +3fdda48a52e50717c1ba37d4063481b3cffb83979a306452a658203f67c128ed +2a30b6547075d68259ccab0933b6aa0cdcb38e3a5a6fdc98a4ef43720be99137 +9b224ad62ab1c70f1660fc3d051083fae13108e907a385db265d22aecfabd65a +b6dc30e17f48612e805bf459c1163b93c8c797160dd719f351153bccfde4d033 +4bb5795219999b07321fec1f12a65b2ae2c4ef5b3aab44573d7aa766c8adab27 +0d11b55375880cf6a0b77037ceeb53f81c56baae3de5cacaf4223f4b1a248ecf +bfef5590ab7c6294e078de66521406eaf3d885da1fd8b62057dd0d0b4a4bc885 +9280293c579316ab2bcbde6d71e54ae58a682804d999fda218953dbc8b4c5f62 +284f8efc379a6ab3d58afea7f693aa845032b06248bbc4db4cb375deb921d9b1 +a1cf0b2ed6d800e22a226956e01abed8f8934d3d5fdf659fdf029a08a690164e +c748f8cd6098337267b98f4207419a6c5603b37624698d9f2ebb78981e3d3d91 +d44f7d749d4687c8082f49eb148a114e7f5443615c43353c74c459b1673dd2c3 +af283b8c1fe2e2a43dcd6992664eb1f1272aba683c9f75aece89bebf5a6a8ba2 +e58eadcd82d81dbc545fade9104d6925a89f599a4366f9403b773115d88cca07 +a032cb84e79b6b4705edeee77a1ff57fe4e42327025447987c1236ab26c9bcc1 +e135d13dd7e9e195c0d1a9d396e49576a6f832a1a93c176ee4c0d1572258a828 +55a9caa385e76be158c22b617d6ed92a789d16c3736c8340668f4c9413896f5e +d564e61b494621148344a091deeccd997e43793d3f4613b7e06fa660a7be4902 +9ee1ded0a1f8906efe0702ea6fb4af7d3f027516758cd2734e7b734c13297940 +9a73611a2ead901874359d3110e525af13b59fcf2cba9256bdb6c1080b2ee34c +ed1a896d53c1f83c97cf26993d4342bfa7c692aaadeaa0b181c6edbcb0dc9d7c +6a53100cfeed1647a6b5c3534fa199c67c52b71606382b50d4f386525a77dc88 +df13e0972e9e5446ea13d886b20e17a404eaf6f5e4ed42d838ecb067cae8971d +a11fa065f1f61f868706a799eba2407d19e2c0a9143327150df96c76fbfd2a7a +087742207b0e7f0d171f0ce4999bb1003ec9e7da29dfcb67d2b2b3590b462679 +8bdbe4439df7f63259203666efaebe82944e614a4de380d1ae2ab79631774ce9 +61a6ba7e8760fbbc770c799794e7489910da75e8d569f52cb8bcfa3910e107d0 +b34ae492d8c8a85679ab659a08e5becf213ee8ecb576bb787450bf0b1803d300 +a57adc7478e44365ac3d9c9d4241c77d55678c5fb579a1cce4e202e44a588113 +e09c0dba5b94c9c4325e622879410e0dc0696944bbcba052a1639c548f4ef96c +88cfb56f223d4a7c46abdca119d528fb2acba9d5409fcf263ce4e347234698f1 +c3bdce9d2a6a9a5bec080feeb96babd80644d1a53e39e0340e695d50bae395fa +779c3da046d1366ec072b2094427e7f3a7809a16a5f3ceddc5acc2527931d2b9 +9feeb4fd5faf78b95d767460d8efb78188f512090eaec0771ec9e65f88d57b4d +2cf09409b56681b093a73e79d3569717e2d2500655f6644fc098ffd3fe229b87 +be5c1e12a978f3bfea562fa6e4708ebbb7c45bc112b23c33887a937cd51627a6 +e96afee7b926ba56e1be566c6225945f88a0a2cb968b6700e79763385fb89f10 +7eac82262c8f68c068c882b747e02bf1be690fd499dfc52e83ff74e95fd90fca +2c17890130218b6495366a50f67c8d77c00e9732a11196ef9ec89b66beea9771 +1da8788d59a43aec654d688970b044151453076ec26d940f2545a3ceded9e2ef +69217171c28c4bd2a677622822c4c73458b6004656e5eadaec56bf2b6a5a3d42 +9ac193c94cedabb91b2c12f8fc078846f527822acd162a3603e04d8d76a9e3e4 +084ad8c8f5e3ac702326bb1705d31be52bc45e42166abd456e37de350ec87f9d +271ea297da421ab506d26597c863598fd7b897d26b7514d7904ee828137400a9 +08cd857277ca8b5a1ad1470ba26bcf716c31146bc3a73636728a1ca9ba1666a6 +042103b178a9217bbe7dc81f8d24dbece68e72fd7e201e89af835f27928bffcf +63100fb102b1c720155ab64c96583316f4efe2c38b19ee7df75ced9004f4d1fc +d1953a8a32722b02ffa34a2be7f6a3cbf6f8993c30f25b04d27cc69710910084 +d11e998cb62ed02941783a49681ff6873677b8e1d1642bcb7214bbee80f1f4fa +4f07a9fe7e51928ba285b8d007988598e0e4bddf16bedb4e030aa1006a9d0c40 +92459b3f14f695553abb5d42713a348e02efe9034e236e135c181984d9307459 +6bc4a0f5bc77e1ebb244b53445b219d3bad378683bc645000bc8a760e5c93550 +0fa17676ce16a01098aaf5e36acb05fc1e5c79e2d5cd6b2f5f1bd7bc5e119834 +60a68f2b83632dcc4117fb186a2186cda35b5121fb2ca110a7474369528ba8da +30ecd2af479a70391c8f5af4006217b2c5de10b71805b6270d71780535c852f0 +5ee990e6350e6f9eb03c019000e263956f991925034883009d859fb1cca5be50 +df1bb89e84a1c981d67b3b174d38120c790e8429150a9f9f536d0a12b9bf2637 +03ebac2aaf222d7332789dbb1423e3869edb30bc3760f6f3669ccfd3cd787677 +67599d7ef119a94ca9d8e04874a7d9dddcfe22abbdec174535471b685c1d1170 +ff1b464da89547767fffcbf91899faf761a711b0f06f6c6833a0f516c7a2923d +5dc2a2e28431c6d64e1522bd945269bd9f839f98825e8affe6671b7388399d7a +516f6fb6c6ac4d044b901c164dbc933a2b28ccd54ad2aed8e707e34c001095de +240edac60a22ef8b8fca52340bffc12ee15bfa81b74fb386e62878253ae30c75 +f21102e542fd69d236eeef717817e9b3adc914bee13004e9772077d25b28ade5 +f0fdbfc57e4bbba5b108025953edae824830e0ae489328d103c2096f0165561a +e902c0f9a45ea0a219ecce5cbd02a21eb8bacb23e28cc178061105bb46349b5c +02410a1210c73736556ea279f0cb06f9096bcfe681124592de4434709356cc78 +8d8f9ce2baec7a5db57aa559f20d80e2bcc18647d85086c7a702b5117045ee0e +99a941793e680f6b2c5979ca640c1c207b4b65eaf03c6b49878b3598de5ded97 +c06486c0da11c75248a6a946aeb2f60af3240c0d3012167b42e9fae014f5dc81 +a92898793b636e3ab616742ffa1e48215e19b1104a9c3794a0dffc54e43d3121 +98d5db4b5cf8a6586e1d0ccc9609c93a456a0acf37f2f2d790134383de93af99 +9a5ced0d22af618cc0d9c7caf039f4f0d3b70c9f414bd30e90222f65b32ff0b0 +2c5e732a3265796cfee9d92ab1e728f401186cf967194a2784a3982dea570678 +e6bed07a1073eb3a92062c8233e83810f54ad0c54a1dc924b12322fb6eadb57b +a0b84f1dab142ded79326cde049b80aecdf1d0041f2ac442874549917b4cae35 +bd5d76d22b2eabb02f7cd0d726c31bb1694b37d6a54bcb55c11610bc1b004642 +d52043fd2914f0790500c162539b43a86543b4cdba0178b15926da07bb0ffb62 +f0be0be33490fdcb1cf9087acdb7ff9f7054741cab5c49c67637a5541142d9d4 +e7901fea9d74c2754e870f75ea0f387a0d4715065f09ca404cd5c0cbaf7308bf +71a4b2970df0d368cd3a74acf967e743f6c6a2ef22b1032c4290bbd04021b6cb +0f1b4b0f3c6a68b2b86c98808f5ad6534337edda484f3289bcc303754ec0b6d6 +b2ba410dff075a4f91aee02773677a8b4c833c934ccde6e74d34ace4b7148580 +a99b3015a73390ddefac4d3af2d2208b011277f2008cbe40f962a551e01b13e3 +bd56d8914979e7ef7b000663c977d74323ef51f110a7720f94aa8bd4bab901a3 +b6844a6f003d20f3748ab1243f7a8fe10aa2638da7b690c705a9a44eb716ac93 +b138fd81fa6c0123dbcd13e1aee0fbdea7ba5bea4d9e4edff84a64a79efac5c0 +e96255df0710ff341f22447357f154fa7c133c6166c02a93a583e8655ed71f59 +2520d13affa7d61399397b2eab77d23d1299874797b820d845bd01a17f380ea2 +0457094775ea1a3ea4ebbb39ab8625646475a5c0ff3445bd16a381cac854de41 +9aeb1dc93cec556f94fe470eedd2c5ee9216c39ddfcd8a8834bd37d207c9e9b4 +ad4281028ff2d41d0b422540ff3b671d70787a6696172d282d8c5590e689c564 +c306b7e72f151427d82ebfef22e8b96e74a3881e5a216b859d061e07013f6330 +0ce90994acbe4ce246bc371815f50c06ce5b1cebe6a42f165073ad64736905d8 +073539c868485c35b80370c656348bd86047e6c6c93d8bed9024d51204797e83 +38c528d4b1bd8880a26d015f7be69d4fbd200611f2c914916c08e1d26e0dce66 +7a1845f7e5b71035754a273d448641e521a69fdf38b50a90ca69c65cdef4afa9 +7a2f51ab16dac88b311af9fb5a40bf43cd003e2a65b8187d97dc89d9b1a830b5 +c85c0199ac0089d5c701b32e0924a34fc2b9c1bd07ea4d905b9219aaa7adff61 +5c37b68c8f8023cca53be56c331f26e8e3a15c1f71b03e90ce8a10b5e3868d7b +89d16d94844468705d365d8b252aa7c02267eb321a24f57ee2ab1f9c0b693d9b +22a37a0a183b02dec4d15a04dcaa6547a7bad9435380b01b0108fb842e704996 +7a7aab65e0b372b210c6828e5a7294b48bc77930c7aade90662900b599562f0f +22ecce78d190c103f9179d917f7a6a2f347fdac0371a955402d9c5e98bedbd37 +b480bcbf8c02857ea59a95e6d7529bc65de9de53167f5cbf185ca0b3299154e8 +82fffefd42468d52cd400ac2361894b4224478e9ca351f2186e82e96196e0988 +8b6847d7cc36e6369432813aa3e5b5e162e0f84d5dc0d5e29ab7fe37f350b77d +780d298936dd67f594bbb737af993fcc87384be16b0240a4f591aa02d87d660c +a6e77b508d7c226be1f06dadaaf8539e8955b9b9cddfb698ef499b235eac0730 +264dda46242a930297449d5fb74da1c6f7ce7e0a4592a5e30a357fe04eb9622d +72232b27f144f59f2a99174c12a678b833c9630c338992c94b987b2c8b8cb019 +0fccee4e52882c1745b99f85ad916ff61c37180f0b583a32e857245778cc6f56 +85c91c2edb6aea350889523c736ab0d3cd5ff4ae6a94e9516784c9968a1e653b +2400f8e114afafdf863cc5b885577e7eb5f4bb665d59ad4c3c3d02336f6a88c1 +280aa849df5decfb645429b5cf0032c2b35718cb6fc1dcec7736266f2d775002 +69ae782a6a82b8532a92d042f5c6fb31729f3a664ef35f4b1fc5f295e8a53367 +031bf0ea8333540667cf4dbfc3c42eee999c567f4de6e1e73e3ce4edd540fdca +03d9176063dfc3aefb4745646bee694ea9f2f8a020e65297137c4840362e986d +eeb2c0c2b94144cfdacb3cb7ec7d072542eeef13052258f2f4642e2c2008f6d3 +c760f441017ddae392a8dcf1d696e5ebcfd94d564c03675cede5663b285af16d +a0f788702b89b1d271a1abe2bfc1942dec066072d2d3f730c01a839d000b9929 +64cd6ea7c4b938ab5cf922be0a0186f4e4a8625bd638fd181a18beada1a39c74 +9d5f44d5a330bc23dd8eb97ddc21efaa3c750bc46d9abc13777d80b8d8060604 +7e219d62b739c9bf4e626ddc9e8c497b8a6d24d9052ad7d9176d69d7138edc24 +4ce0e30a7def90afb5d886e0f569b9b02100a5ad34ae157badaeac2c4504c1d6 +61259c5895440ea711034f7ebdc8adfd0fe4bb00cfed4c886654fc81d5c1a84f +488ad886f5bb5bb953b8ddf552a8fd8ca44413244a4680a84c886f0ffc507eeb +61aa79fe9fc48911ba8b398081cb155bcf14e6e8e83d1b841a439d957414851a +515cabb2025b42711646fc81a651031dcfc2f97c70f9f4ed3266a20de4de2d51 +b205142a83648a3b042b1ede7da30ba12746a23b11e8d075349b17d6bbcb867d +a0a3c22f800e7cd63b1685d6f670ea9904e761b66378e373f4ef6367c6f7fa75 +eb7092dac401b4574fe51762dda03960e86697abed02544ac5b6e64eb1b4bd2a +1fd99ad634b9eb3dc774907af4162e7314d56e1ed734d05ca8e608b60a9d1597 +2f3158ceb743c15f97a59e3a8a6ff689aa8b50cba6039825012117ad1607e01c +c2a49fa150934affe9c501080df700b8dc073ed57ba87febfac8743e46d3138d +e9630738257e18af8aae0e5de9a0fd93a01d3018377b72a2ba72443c987570a5 +7882a18362a3794d4d89e8a10d00e35fc539511e6babea172c9885dbae840aa9 +31e67840a9a42ceddd3be18a42e891f8fbbcff35b14681a8c6c795a63e8811ae +de339fd5b37d5dad46fd9270a7e7a9e99d0c75bb1b0525dd9d266d2fc6c19cd7 +1e4d2b827c7174693046a2c0b31410622acffa56d154d5f510af52f0b851bf1c +e24c041ee6a1f4fffc2bafb03164531ac8eba631bf2525b7c607a4ff9ddee854 +4527eb021765205ee1a8326b86ffe2e243d0da4bd328118a1921bd080ca3b40f +0bd2dabab83ce32e087466c1e5b07a6c21e4e84028d057af6413e70f3f0133a9 +af481e2d6b741da62f760c7324607ba467279ed2cc26a24846c9604d42092f8e +98fe2e5e4373e35bcdfd43cd98b146fd2ccf2673415f87201ba750c2c7d3b6fc +8d4d9c39678b266d5b9d167b4d76806d4807934881ad4a8c9db4b3b183ba9f37 +ac6520d561c1d1e1341e1311869cc1e8a12bb3884b267d12c65b42cabd35afdb +9c173249c3684c30b78b7e229c6b46f9c06a752cbf7e3c69881e5a2229fd5cd1 +fc94b65160000d3af4a0ce1e73af86ee39d5e1c2d6f543d1c8a7b5b754332d16 +3a1ab7b6ba5b162440465af79fdfef30e36ec154a5cda9e0294628734a9de89b +23be7ee555e2a1f9c9b944c865da89a457f653485749cf2e13bdf8c8f0ff9e48 +e6fff6c382176d23bab447d71ce87e7542303551de9f22d3e3566cfad438e9ca +b151009ce2a6f580c271af4c8229ec417bb388fc8634c8bb77b6eaf9baa90699 +9cdd6931082f6f78f962159043e90cf8d3b717b97573cc81a7995ed5c7f2afd3 +9a4b7a9e63a874cda9c82adfb592bd91be77baaf988b6a22bca45c8d24ff1c8f +0699d3d7e30aedde9c7c928fec7b42c3bc9ee074229ed14df55be8816be05def +41db86a3df4dec84ba4b5bc81b3c71b36710366972ec95aae67136445ad23cd0 +0f443c93552e2cb76e9ce698c46a66b05be7d23e9a7098d8ae2c1720f57cb1a3 +8f03c857b872f3c5abb2450489c71d906d3d96b2e6a9f3a466a7a53cf3287050 +b4b9d8bff7d67683bc8479715d9ac68422324f3ee2313daab0b8fd78825d3d97 +06abb57ed26e5ba5ea10c548e5acf2f4608d8b21b2a296d37eff0320c51b54c3 +2a913432682d6dec351e66034b7f9cc2f64385974684e06b57dcd5e491730b0a +f5d965e2991b96606060352ee6aded1d24f2526c43c749809f25ac4a45141cb4 +fd0744eddcea5bbae28b65daa28373241ccc009e56854382c16b9d17cf2017c3 +20ad12bc366328da65d5aec6608704acccc58eb40fa2b67fd313dab2c6925ab1 +08b75653a13105a350552565a8a6ac0dcb6393e8c06681ed08a852f56ab22ed1 +97287725ff79c89566b653ba298f5695ce2e546f181552a9c0b689f63596dace +8438ad4b63db524e5c95ef084e5460b1649f834d2f0897c99ea20230cf01bba0 +045bcc862bb05ad4d930553262026b858821f3a8777e9a0c19f614f8adc763db +2f197ba672c12fe3b6c16b61197e6cf67ddbd2178f6411638cfcdd53e80c736e +3cd813f9df72ffc561df4e1bbf1ece86f5365bd9b48d7f7169c7b20dcb2372f0 +e56cb6f41c6991c63da50b9b5138ff31fc96ef5b23b16e69791abb9b66dcfc50 +5aee7d5c6afac0c13a80ac03a559ae47592d5e4c104c12cabf1267fc8aca0e02 +8fa7787a69a1602a4d323a5256e7198e6bec18916878979bad48baad4a46a2b3 +48666e5b5b86483df943c40872d74a5f5dd3fa63b09f143217c1fd041a579534 +48bc4ac3a0a3fbfaaa42450371553779f9554094c085c162235f42aaacc3e0b3 +c1f4b1453ee08b1fa1ce45e5b3167a1e0853493041fa1837de255ef4810acd2c +bfbaee3d514c35bbcb02769a3912d98b9a873cb0a78255809184354cc429cfc7 +c8fe90c3b3dbcfb7a7d40925a8f55021a394d1d17007fb753659fd6689f01b9b +300923f05fcad251ead00fc55773b4016f841e26ed6bbc36bb3a93ab5feaade5 +eb19d975da3e1a7b398a98f6bc0255b11e5a44c7305bc48ddd6f30afc357582c +e042f94c7e39181965d64d12f45c978e190d27f715e241e66a639b3e841d82a2 +bb3cb5bce05a454e72544adde97db2042459f58031b0f0c1d29c992777e0fb73 +a2ed88b9159ff2d9b19bc2bae6694c238d6b80ceb8a993b46170c97857e59483 +0e02e5472d03e5fbf4b2d4b818a778d04af8ce00be730c8bd7efa30bb430c8c7 +3f2ecf3a1bc64fd431358b34af41e60b45da42af585bcda364f8894488813f14 +5518611b2b9cf3985941d8d9a077967282b719331947da7099f2eb2d43d5892e +219f3b922ac3fa2735f437d6542a26b7d744ac8899b53cb50e7f059ebe1c2960 +c7375cb79f52c69882c6c1ad66179bb1cedf83c7fe50362bd27ce4322ab89a5f +f8b56567d82df2a24a0fe656e59a653c9ff9fc332aa35b6890268ba95bf0471f +3823b06f22f403535f1ed1e4c0bdcba397c7362f9c66df370c63501f16b7535b +6aed957c964c83361e50f1047a34468e95a810d961ff08036e5c1727fa233879 +cab865910ed8ade413af57a20af437d6cde5581b3d15d78bcf52ae5586ae1d3f +63ca106b2c0225549304c2c69186e2750c78d1a0daca533560ea04e7220ee275 +f4f842eb867148c875473020426bad302cb6d53a2a511cfb98548aa0de351994 +4bba76fba3e34ef57702888467eb2ec58968d3ad533ec5464d534eaaf3198644 +954fc6d2f8581e4af6d1ac4935b84696b977349404cc339611324fd76c737c25 +09c12608e99cb664901294cd6653928418e8c33d334fdcf83eaf5e829d8217d8 +5e451cdfe1614ecd48f73d93f767d651686985c7a6bbd033a4325b76c5a9750e +f25f01f2117268e17ae1d9adf475424c872c49d09723aea986bf5709960cfc94 +680dca1fb419aba7bde2a7bed26726addcbfd0f49f27cd1304be177e63083f7e +665de573c6e33a076b43c9910c8aa0c30cc36e0f93046e457a178faf6fe34db9 +6de8b59377f776db437cb004cfb451f7a3b34803153d67876065fe0d8fac0079 +1b1aad7d15d1f4c8a7117e093730e91d3c6056cdb933f4e9eb2fd05be7ca9f40 +c3e6bce70ed4c65de4a39d56a01a3565a11923fdf786c225880728d53ea2143d +35ca5398197a5c50bcdc84c97c6ac62debec478537d1ee76838b12d6d71dedac +b65547e285ca6ab1af1c46776fe5e36e20879b6a3d639e47e10d846a5b6f1dd2 +93ff5438847545f0cff55e7299f10e70e1445252bb2e188a6ff7186595588dd5 +91a2f03502d4fb54854bb4a85ddcd4724e4caf0545ec2dd7103bc9ee001ac49e +7cf3d609de6b4210d442f17c2b38bb0674fa5a0543cdef473568dd16a2b30513 +231be30df236bf79de2f886d8d9fbf98c412e563bcec75d6635c56e37f62d3ec +5c14b064500961a509bc9210fb85f67350f1618e10fd4ce40f5c11445e6b687e +304685258605cd9da92209731642c74b7830895aa5a5e949680b55cdef503c18 +717b106c2921561a6911337ba203d76114ef9853a00cb096f6b7333493b469a2 +68e2bcf484e504c0c0761139336942b41617a4d39688cc5719e30554271f165d +b68f347cb270120c52e9ff251581209da81425735e1ac520b884096e93bff3b0 +4f64719f4492f54e796551afc6ba0417868fff7094839ca7a26dc3382069cf94 +7d8398a134283859761dc63024cdad03ef281f53a3753b14ef6bf124c56ab007 +93be193605b4a6decb7cb4312e3d45cbfbebc5869e619a209539bf58eeaae8d6 +37b93829afe8369b53d9d5ed8df93b14aa4295879244d0b2d0fcaba3e5d38f57 +f4c5d0af3f0f0ad299d889a47df833fc9b3c86ea3a4145c18578a726fb3d7b42 +0e113ef2cbb645f8b806e417a52d32c166d40f0bfba2f2d569018aeb707f17b7 +a0940d6a65a300139490c760e3072104cafe4ddc5bdd601a8577cf34fe56e56b +32c6c63ee6b0921b7862a4b1ab901790d883389a34eee7c4aa83c3b8773c3b34 +67d8b3f272db8072b9f22013c6900409ee5b5db808f9b6698b72496e560eb9f2 +62446c242dfe108964b2347b1ac672f95dc6533d5a18a8588b3eb85c12988ecc +057ce720effaabfd7c1579f7f2ab28d57879dedbcd8726c6d0e592e384987e8a +7134a8ffb3eafbf1083c31cccd2b66853b5851af67919c4cfdbb93d3ae68b512 +5b72faf5c8b978a25a4f9e99a860caafc0a845fb6deebd79f88425ff933de029 +9b5aac822f0d91e0e020f193c0045e70f783f444aabe46ea87dd8ce338ec1349 +ec2a9dbd481b72ba4d8297ce9d739a1a640ffdeedf3d90c855b5951b25a941db +037abc468787fff73f8d3c8313e561c86d3949ead8858a50519a915ef54725df +4b2719d77688554c05e6ed5f685e3048a7c06ac2f2342e310046579d880ff394 +b783ac6f17b3d8e7aee6249a39db9cd455bdd573eef26bfd4c1f56392ed96de8 +52d29df5428ca3db30f9960dee0d90595f611638ef969202ced1e0990dfd9545 +9ec000036ea4b44d4bd6cc0f0dd1b9dcd6b2911b242e3fc451cde8914569cdd8 +ab7efa786f7ef4617edd4cf7d5dc029bfe6b3f808233bb47efd8aee7d646a746 +d175178c5df2ee708924be190724ec5f3bf7a34395a42c8a473fa98c67e5ab29 +f966e98098542440a54ddca84d6db81871fa2be1dfa8fde7f3a9b42a704e4f60 +f1ae9980b1f4eaa7e0cd8c0c1f0ccb112661d86bfdb39f0bbe997288f7b5d144 +d5aa0ec6c2031419c5326bbc523e56230a7cd97c0d30318b8ec07ef29c624866 +94004f1e307c3a0ed43e0270049c61dadb0c6900c7b7181b58c4d768b3475e34 +8785d3c76a07ad5c69f9326806299ce1f919035aed99b47b332d5cccdd9d4ded +7f0bb4fe2b411bfae5c97991e120f502bc1a3728e9ebabb9adfe4fa2634904bf +1cd6aba258d43b2528c54e4c9e9b87af87374d0c2f73fc29645282e06c1bdf4d +a06876e09cb6c2add17ac006b25b076131a359d28cc0672476db4b155a217cef +a664af53be46db53cbcd5ca312084976611fbc94737d25635fa04d6627324adf +17af26a0fc638c9153a57ca5357382d28d7e8f07f67ff02e9b5921431d3957a0 +258dd927855bc24cd89620d2799f84f4c4b7fd565ed78c0db486786d9ce769d3 +bedab1b1c93c0723a4260282c0fcd9cf77fdd57883abd3bc27fcc7ce895950cc +70a6e9ff6774eab796046213deaa9c54cf81773533a613bd3bcff0fdb4ed2d4b +4d71284351fdf86cbd880fa5f144643876b1043253d8c7c228dd830a16ddc0ff +046af3a0c38927f1d3259366d586dcc9309be00c720bd4fdf89e24f7e817e53b +1dab7aa7e13a55ab9031bbb7d95f3bc5a81b4fe1cdf664cd572150a3d5ddb87a +dc0a188b472a4af2c258fb79e237f5890f618a2b5b8b3db94531c64abe64b8cd +2879e934d1ca3c1955cddad5edd1b46b706583065180bb6a1d9fe674d1ddbd1c +61dd0e0f83b51ad52d26776b3ff5b60c362f5d69df6300da52f711240cc9c323 +6c0283d6cdb921712848299169e1907742429908dca0c4a10a9303e930c63675 +6c3718d84774f47846cb531b7034d5d6c512df69b6310a829a46734cabd4b07b +be2888ab0b2aff926b4d7d9fd51f91901cbddf9a9ad894ab102e73d6200d5d4b +f8438c7a53b0eb1d1b198d7acf540cec9db7aaee3a274f892d1f2bd522738b29 +8c4edb54071e56342fb8ee5af6337541964708bb757a339092866ebd42a07262 +8d423895ffd10527bcdbcd8bca0468716cda0c5a6fd8483e9dbf4e5bc6d5ebd4 +2b7babdeb9388f53be9286ba816062c4f79a86a7905aed0b4e2a1439c9ff5c3f +df73b5a24f178df4dc47c3f8c36b8f219b519e73e88cd883ac1a2e1c05cd0fd4 +8a55a4f179370e44c4f67dc31f88dcd25b9de90ed9dfcc2c26f73fa95c8e54e0 +2f479813aed08440c90f05c349fbe0c1378801e30c6b1daa05f82f426196007d +b85f8721ee147e0f1d0893b2f12999caed4ff163bb2056a19dbbd4413a4bcd4c +6646bc7333822ffa55fea388ce4cf1372645333f63231fa99fd87ffc370bfcc2 +fa545494c3b2f6a34bb2d7f1dae71979df159ef9b5204b5f0e4df265f4b2c949 +8f648d2f81702a89404c19100a71c7146e52782ed486f081b0d29f91be613e92 +9398548d6cf8f00c360e369701a26d8b1707f1fd1677c740a94c93dffd9e0533 +af72c7a077898218a4ee2ca75fb2bb3f48b686e1e10dfa7d9474503f554e7d14 +0acf2d678a416f19d7dcbca05c8ff935f0b66deb165e6bcd4adede640c803ab7 +00d705df7e4d32dca8633c6ac9920f6b38da49b33eb6aa4de0822f23d91844e6 +e2d646315b23795671ba4d8fe5e11ba2d87403907390adcd73a5a95992fc36ef +81e27781d8e21f909cc12bd273f13ce10082e24350115f49474d079de6bcb1eb +8ceceecc07521be40ee25297c5316d56756788b48e992d90d235ac49ca9854bf +53075780e0a466dadf948d8f5ce889e35fbe45e061ebbadfc0b6f4f1a8e5082b +c3cc975b73da6c110a1f2e85e263a1055b6863cc87ba577c24ceeb39dd8fc78c +ff3b31b767765ba551a4b150a76101060c2fe2c2e9bb68335b505313d4b4bf49 +bd787ca8dd69df814de515acbd2f2340ca10f2d4c4e8a9d3d09830e35c03c430 +9fd0509589d6f83538f5b7a2ea6a4d74a061ee1b4118ca833820683856aaa28a +219cb1f05a1ae32013f6f22bcda51bdfd651753206d28218e498ab2495ea6de2 +50b500f9aac80d9ef6566d50a84038c149bc6bd10c5251a5162d4251f984c7af +9661ab74ebe8ae6291d5185688374431ae544e908082743b9e7bcf4d9345abf5 +8e6ada5bc3323f297f3f3e8e244be2c711c354b5725f2d94a61adcf4ca4a41ae +5f3508474e41a1d6e0470ef9d881b7d4f94a9c06254b5e0c4b97487d232da456 +6a1d5f6b4f459ff9f26eea409a7f01a130017235be79396ab49204b1292549be +7df5143600a53f3e6ef3709d3833c89fc6d5beae7ecf06c1c676f47cc3fa7b71 +88e3c69af5ba0bb8f58ae6056ebd21e36f4fab123d375a3629ee92c402acf696 +106241e45d9da92da84ae874ca22f9a79bf29e928913c4c9cd52428a3da6a2f9 +e5fdcf01c5582c4b7537e57a0217676c062a67967a5679258f85fa990f1422ca +ddcf98e28823f484033ee08cfdd8ca05dbcb3e71e545a0744289e5e8780bcbf7 +6cd65a624cbf1049aa14ee271e4f1495dae5ac3b48ecbab4dbc741d93e59d0af +5161a52ff061a78ed0a0f782ae93ba4a5b70d50d280dac9bfdad4d7c65f2933c +a7660e7a6acb691b984af1a99b6aa6d738fe6ac9592ad2de791930b0f8960f0d +55c9e5afd6710bb7143be7977700eeef76d045f1bcb54cf8b19bf46ab4acda0e +cb72367ff7618ef29b63b18ae2c2117f4d4ab3f2daae686467510d1433edbec4 +0eee63cc7561e6909a0215fe338c2271ea5b09ec726abe605b2bf207266600fa +1c6c6f407aba94a67d6406b2b57dda4f46405b2092961a4c537205fbb9622718 +96bec80f916d29574f229eb966ededebcb3660d81794e0ad8cff6fe4e7a6ef31 +e535597cf230f8f59f386034232f56439ef518cacf3e564d98d3ad81c878756e +f8461b6f8a8b894fd4b7bdc0abcda42a0dd9849bd5f960b12485c1e284622597 +494e43313b121b97a49dfcb0c2c0aa74aad3956c02585f00ea334a73381b78c3 +e7deb5cfbae743d68bc230350e0075d5b309605343d3b16e0f8b84b52e98ea28 +4074c7ef2f047d54ba02f5760187df82f58f5a2fce59e2698fda66edb22b75d4 +788f0d52e2e9bfc0db0a8ee972fb2b4ad3d89a49a994c78184138adcc6e5c492 +b2580a917e9f935e0ba3fce840346efe552cdeed0dc4609d42e40adaf169d3e4 +3ff06318588ec7ec5524872e4f66e00298b840d464abfdcfc3e95848a58834cb +b5fb1967a921c1a0e23e67d998346afb92fcd7ff50a3133b919f8058ce058f96 +e472d24de8e1ba5d97f42adc6e6c0dd7dabf119c1ef99e36b52d8524efcba420 +00d60a0c0a8b1355c15931c3a729ee6d966bf62f7b18171746a75213a17d2c5b +a2d6ea0af7a5523da36883e03ee1d77a5d4b49eef097643f51c5cb4d3e066f41 +3dc3c85870fd489a9d97ac9a27c655d6c481478160eae3b4c021deb8e64294be +5274dc12379d81494bc0a907f4a49c1cbaca59912e2e6415b0438949a4759b4e +ed6420353160d9b78f16238993d7f308ce0b6ea69fb48458af39f77fab595de1 +bfac9e7574f5ba6ab95a7c41a55dd3730cf5778b10aa804c572bead9f9e7fbbe +063f377ca3a4572dcdb511e8086fa75c57a3d7d63b1cdfb9d84b2c57a1962487 +4e3cc5f1c2097d8c6800482c8d39147a403a9a5080b0b9d777e1f00c815685b6 +cdcf7bfbd085f9656a7a4e5562758849a97e245de07bebb369f752d426b341cd +a203f7aa44aa23c676ab2505362c59281b9ea8035817612b3bdf2672105bcde4 +e537711d6bac1ecef536176e2ca96f0e1f49d2d97e5c7e59598bf6dbceb394f3 +c42a5311c9455740f127beacd6486b7ea17808017ab26bae649049c2e1546389 +802ffcc10c75831684179c7958b901c9c9fe618cd7f27444526c24d6f1b32288 +eefc82010b4c7e42733d702c3869f9bc170fad49f47948787dbff27795f29562 +d7ae4d948821c3d013491a081c3fbee437a6beb88e2c37a6ba4cf9578b1f35cf +a3be5b6c843f8c9f87e38b02f182017072b3df459a1a7e3f227fbcdeb1373d1b +5f81d3b68daf550de47f4b2831c1cfd822f59c56a7589d21109fd5d2a61f5211 +85b6c84b4fbe49ccf6c91e5c9edfe7e0958c2734b4b96fde7a636599d1e529ea +9906c67e04c5ccb2921ffc06a6a101f598e5095cd5027a014865d58f65b435d7 +733c2fdafdf992a9a4448dd1766861b433b91fb40d4aea0f089beea6e7fec1c7 +801b1656646d2e028fb9fdb67245580b775efe29900c622c1f36d1907ffa43da +1f535f5d034b2ac0e568fef3494d6853f827ad1c59551f78a9894fe634233733 +d445ece16d21807ab1d287109a775938ea09418befc20224a918882f7c81a619 +396e5b31ee4c56a7080808e04f9c16deaedb9e757575657edd2236a426797f0e +1286458756197647e666f30b00951c4b8f9033c3f8708659fcacb9f87c99acbd +d8268bf2403e94ac9c58b0279c280d188a0fa17e72b733dbba67f3d034c7a7c7 +eaee1b3b289cb0c12fa12df3d081a861e85cd5d8b28dfb6c49c7fa12c6d3c06f +0225feb554acb116bf399e804c206761345e212195848f7d574f336f891380d2 +898c2dc8e72fac821b8ff1bda7103a5a7b32818b63f991b6a6626680d7635396 +f95ea0508b9f21d93191966fe39d3f24d1f912758e9a94f0bf756d1f5c69a529 +41939b9ce4b709b1c81c8c304e4238f140d06640848923dabf07d3e3028b39e4 +3bc7809e5e37b5f67db4d3a5f9dfc69e2c96aa6eaf68c5b03a596dab012d079f +ece104f0b952dae0a2eba0a757ac135eea1de09dcb1fcf7a41b0a575ef561659 +5954ddf026038976f185ac077935c2e57309f34fbf70bd0fa0e9ecb84ea78efa +2d8fa3fbd052969fa8259eabde5bd22738943c006a9e1218ed697896fb545488 +4093baadf53c44636768afa3f6bc120d6b17eabdcf7b9eca7891cea65a725329 +28f7a397b1ec84fc9e5f1f8071b824cc071210ca942618a1a95a0596a646d01e +5278197fcd1824ba790af2ddb72aa8ee16a0a313afaba3f7034344fc791c4c11 +60027c1bd20f9849027966cdc4043a3d4057286b5e9d8d5e9966118195382932 +eb0aa8996c12396c005ec1b90889e20cf7dfa05e1b4ed8838132b75066da7c11 +b5704a3d26b8ce38e4af17ca90881fe2b8f6cd59bab9e70d9a9610d388f6cf44 +4dc0081058331df702e3ce06f633eb99b6447f3d05d348ac7c5cb4e5b12b17c1 +dac915b37f8c9333398938473ffdaa1e05ff6d03d3be7bea384d503eb62310f4 +6bd4278c6a22a03127387fbe11a27dc7e926d506e72bddd67c9128db5ae79d23 +b96b0fe632d8dcaaea82f128e5dee2eb9f1fe8b6d6dc8322cf5c5efcad147e63 +7056db093774100e34a60c899481c7e9b998954699c6136540659ee985947e89 +8a40e64b80a6068544de68edce9fac41e278471c86e3cef12991cb295527ceda +2bab3f2cab81d5f9d52be82b1342fff93d14b810a6f0bd0465cb387aa51c0a2d +cab3cdf822d993f077f6a6812bcac1740dc36ef24a0852dc0c79da7530824f08 +6cc92eb7a9ea78d759f3fcf3675c31c71f7a1bf9f47483eb211d940bc1b2a2d2 +4698904a9d5e8e35d1875977a6c75243bc895d6efec907108beb7916aee6fe9c +cdd5369767b8e3b2db8f59fdc468fbe5cfb38ac61be7163736b14d861a724235 +da82ea51506e8abd2be0890a5010e6d727fc5a0bf6f401997832ffb93861fe50 +e46899cd78a3adfcb03bca7f51d588b55c08066f18e0a0a0d462d73afb58b0e9 +a02460ebff03b7cf95326527f13cf58eb0d99e61a8aa238927cb19300ae7dc74 +3da58ba9e381d208aa8964c1e25e09e1207463dcff70062382314661f4ee8dd6 +d6d4c2b7c224e34ced602e01574ee611003cad4ca3895584d720c9f999e6d487 +5e1dde02072846b5a444dfd708a6e134eaee79bf8d0af96691b7ded7073c3957 +961ace1aeae3f11523890212486fe4e7b7e74bae2d0dbbae6e69d0d26f4824a3 +1205d967cab0569406fb616e8376f2a05c878c24143697a72880a8a4ae80ae72 +beb17508e483c079270952af15ae6fefce3dbf176830a75b919a93fdff69cb32 +db7fe099ade891eb1eca805c745ef3816d77814c8250309f7e7c5670444718de +27eb31611d11b5a263f7aba36162ea771407b3c30f481e0231dd45a13c3af447 +bea2bcdc91abf90451f34dd569b90a65f394acda38f427d182eca52073be9822 +8440c504ef7bf0273f4476335f218b44e0f32c5b49c27e84fff3ca399635e1b8 +29ebc7fe9c2bd71e05ce4427430ab9ea9fc8f0f6302c6eb0f146bebf6787f6d6 +9eb4389e99169d3d93c4d7ebb9b1f1c79f8c5534ec75e61cba5b7c606e803478 +d3db1e77cd4bb36faa5f8b368e089f7d98d4b076bb8688f353ad35d42e99436c +17e444e6cf143ee9d631895cc56472806752e1967e2391dc3b43f670cafc4717 +f4fc1c3d765ed81d4c7a3926364d5c65f9ceae3c8665d01d4d66d99515ff94b2 +7e62866c76782d9e3685873a08caab233901150dc58696b800e26f2b51b83b1c +e1fd4d87ebf9be9b0ee16445495b6454688d621fd55cd0a48e51dac94a5b6328 +c6c2401e683f9ff1f7a842f047d8348b547328d99c6e3b7c020e28e2d76f802e +3d7e2379aa346e10642fab0572552350f8ac27d39488d887d8b5d02c5934da9f +376ce969c7dd8df7b0ccc5a99946dafab757dfe83ff1162b60e7747ade770cf7 +605f2b5c3c19c4fa6fbe2225b0ca6581cc91bb9791e5bf95e46449928d54369d +74a2d23794c84fa5c0fe24b2ac6d9ba6836edc6862d454076134f04d47226ae6 +a96b1e91e1d0553f3e19a3aab84a229816f06f9b9cd8270461d2b0cd18907850 +a7a0e98d9ffcdc98c901753f36310c8703295a89ac76f370dd7cf779fa300a40 +c5927770b7704c09100a47dfe7601886cfe749e2a0efb751c809fde69a7e6bad +b3d575178e077c1ae72a7997e6ab114da2d018f2a448aa4cc2d16f9f55439319 +84fdc1a1e915cae357bbed97393ad3ff808b8f32e9816770b2563f5553b69cad +6e14f95cbc3724c4970901139a1323e2ab33d995aab43d3589e0e04ffe095b50 +b9853a28283219ae597b67ac6b497ea561bccf155e737307b3610e1d5fc458a4 +6afdb674c5acc1dfaa30c879d9138ab16ca644337221c4f0d7bb0bfc6d8816d0 +acbe5685751f2b93601b3b398bbc3892a4984bdc732b574f4a8c7a29cebc9c8f +4ded6220c6942f58492b1153d7e0890753c782b2591dbbed35f53e7638281ca3 +0cca72a6471a124b710a974ebe899348c13aa2e8832a007ad2c31ef9d9f61f95 +1da646f93b785043907635ea60dfa3dfaf7c8004dec3e60bd6e322f9adbb3a8a +9e921e39aea0b0d3ea7c62a9e0e8e74fc8e8b8a35e892399f86c84f9d71155f2 +16d99335dd5705186137eb676101a21c79649df906bc3cc1407c884712d76456 +7a55c56e81b7208e67bb9c37d86bd5f97f948cf9841746f9fae1ba07ee153239 +cc40f50e13ca70396bd645d6c838527f613227587e90d78d477349e1f3e74fd9 +66d3624fe63ba68c0387083a7e10ada01433d7b3ecb6c2d430c31ffb09d23064 +fbbf422016f1e9ecc33873c5981fb28caa39aff1a1801a0affbcb5412ea1c0f7 +d97dd52b18670c246b5b389106b35b96b05bf678f2ff17dc6b0e97023d3950d3 +66503e5b733c8bd76efcc3bc7843e0a4e8eddcf53b82a252021c11262544badd +83d5b74d8457a63e906afd487bda8b1875aa147f8e179ee87ee9cc4ab0cf11e6 +1a61d2adcde32e3d57be92cd7df0ab58a4ff91d45c410bfaba8a4ad53df2403c +b26911c3b64a8b8f5fcf3c63aa19817999b98e8dd2ca7f89e7e5e5cc2aa92f92 +f643348c888b2e75476d306107bea5437b49a67195ffb58bed36ea7e4a73f849 +416c5f21f2c823cff8c48ca91e15e61a7e4d95237ff093fc3f17c67466a55c08 +17596be44371c32b667845aec2f6262d3e5cc977367b5666d5a85fca302664eb +3f0223b42a8ca21374ad8e9813a3772e48173610eebceef9b4a7df71f95b4061 +cd390b3b760c981be2b472cc7bca1af1efe641820075fb5c93e680a621e05146 +127ae7c479881992556b4db7dcd780b9a8ddf9e66ad6ba036d0fccc90f92ddfc +ae9c764976e53bd3d193e90d2188930feb85f75104aff65e0b699f7ea40363be +5ff0b645ccc18f67606b99d09bcf24d1177d111c9bbb2f57b0d215b5caf9bc85 +c960f579dde942f2e40b20bced7823b5ab210742a69bda0f0e698e1551eeb6d9 +a713a72d92d8ecd5ac03e693ef39a07a4098249b46f4a03105835154067ca999 +5f447c93685f77d4df9a5c83f7938493853f24824b8b123d00a246087d0053e4 +6495d86d2995256fd14b84d9e4f003892adc6a99a16739a6fc555621933d4214 +c892deb36dc8c66669cb068fb19f78708d862e96e143395af893ec18f3abad8a +55653cfdf72d99322aa90f0ec4897665773f87546db9b3e6d516c8af3419d004 +8a4ad9e18a6639b620a5518c182b966c97fdaa06f88d56e91c7ee99b1e43aa8d +2726b3b0f3bad42f2552a99e7bbf08d3d7dde44c820817790891c152146250aa +39cbb57ad363de140a7d1186480415dd9a3fb9c35d6888921be77cf58cd5c2e3 +3fb855efbcb3fec4f757e93365cc9fd859350e521b34c2bf9e7b64bcd0dea74a +001d79a968a5f679b7b17276bc2d45700b575c0bf40befb52ec7687bc00d4364 +346c1621eba9e080eb7d6fa8b2ece2abbd5e455f586abfb1121ade27bcebceec +4774b79f811437fa091a466403e405564086f879d2dc01f45a589726d8a6f752 +91f512a6372949ef37fee9f44b9ec329c6d5f65dec5ae6c6241e599caf118888 +3d2b535736849e72659b4b3d45cb13de6fac8e62e16967d22a9e76ea22b74ccd +6e2c001a8f5a48ffec63251a6bf775f4c0430b752f215b6c5c92e39de03287f1 +8d9d7def8190f30ffc53cb7f164dc5a0d09a41c34e1373257a4189a468c1df47 +18e4220eea2dd52a5f13854cba7ee28f7d37080b0e9060a83981c499da3ac59a +ad83d40410e9e11db4b266c11854f9e9185b5d90d67e9922941576a7b6a864fd +981b74ebc69354d8fdb6e521274027b9b281818372bcc77129be31a8ac13cf5d +27f4d324cf382295ec0f2e7b4a68a809091247f7818f82f704e9daed8fbd759c +09d215a9feba586688a6ca4fca243ac637f5aadda42d2b22c0baf9f970392a21 +fb7ad2d0e085b771138355547e82b771482f952957a14f38f8d5f5eadc3c0d89 +85948e46615dc2fd905d34ec40b9b6915ebab7e34682a0941c7ef83c3b28c4f6 +f4be74d783dd07a06a09c00a65ce237689ecbb9ce9c80c8c73ca75bcb2cbcfca +e49e348e0f948e807c7a8edb3618ec36ec1c6134bf88494f21d676d91c2b2f68 +b966aa04b58ae3be1e3561c0970015bc21f63b02c63dce11f919294f38eaba0e +c0d67d6cb71d5cd372060a6f508017d0e90167f111a5219a0008cfa5eb37b0fa +d3f7ffaba43fa508904ef332c8d136bcdaaf78b02f39bd764356f0043734e98f +762e993525994f47fe6c4856a6e085fc39a6d1958910096be529463ad466ccc4 +0e27250947c4752966c4d0d4d2ad7f0199872f530403b06bc8af5a2ba92f058c +9e72d7d6b746f56748150add0a102c889f4557aa6f560845d96dda2da90e9b49 +31f00827fa8c729d986a4f6d1e614010b7c89bd90ea7c5ca6815f76d828644e3 +098613fb6ecf597581eaf7d9e9d7a5c65979c89a6b58f687b514b52a707e2304 +9a3b446ab8432f17ec385828b4ddf857f601f484ce143c45ed584787465365b7 +57aaf6c681b513e4c89a2e49a7bad2048d6946169d75407c4ce96ac61d80d277 +4896cd2098056578af3292355516c06963776d8fcb7115c53da5e4b9e6a19a0b +d72ac7ce1612e1e0ee0b9f9af20c2c57d0cacf5b2c6f66e3df799ee8a1d20c86 +75300adb34e13214f0ef195fcec89295d26eab98b457091ca12d0ceb8aa00d32 +2c77ea91432117c5432597b7b5698abf07f9cc8a9f979cd5b7a7009c512d2e31 +922e60a25fb0b5f77824f19aba23052e8e4e4bd329aa3003a5372ac3382b7832 +51a98c63d91c29dd82f6819510bbf6a15d0d2e2eec78a59d7610937ff418a7f4 +91014d045d15da8fac001b4e677e8e6095aa5bf801bea8cef8f8e809d40f3998 +b02aa2d6a8158f5f6d4e1e669ec566ea4079b87096f2bbc19bcf196d51eaff7b +4ce2f41020168cbeea74e584a4fe46d792f1f94bcade7326b5e93cfbc6f9ea91 +1c89165bf0811a71d664613101469e4315137e42fab5c7c6286372f0ab3f038c +e8dbaedc2527f7976c781191f00b50d270b87938d70a0f83c005764d9e2db2c7 +ec7286cd86bd479efc98d8af58b5b3c13bd5b4733266485cfb72016797409355 +a9bea7cb3138b05d4b45c906256e47ff208d9fa61af82007dc805982ba5549dd +463f0da046415fda3a49d52738fdad88bca8c30d94d644027e3cc043c72e4eba +a7ca317aa81981d99356cf6bce3b19aaed7dbbd0282879a363381007a717c74a +3b1747ab686c199e1e3fc18bfb2d8c6455f252cce3a6103e1d133c7e49fdcf55 +913e0b1eb0ee7d90b7ac0e30573cd4c0e25e12f6fc87e924c6f267ff5e31e1be +d19932aa222ef2f0c261028dc60f5a2a150d725386425151b03ab9fbb2f10e4a +efb3bc5943651d444f2299345e01876e7948b976e0a3d7e60cc661e69c71315e +089aaf0d90529605e92de89f303b6daac6a8b3d2365ab1843a06763ff3a61b65 +615760ca663a9287a9f172e8945a093e9b4b102e9e43d0ebc61547df63c5d1ca +5b71faa9dfe326cfca20c3e57f1d8570365f88fce2f835e8f7a8ddbf5f6723ea +844180c0a4f8625627a583a4eecf22bca9fca008b9e6ec8e22fb01b677de87d8 +6c0a84a394234a3f4d575a43a7ace87c8cbf11320c5afb3b275f40c2399dfe9b +24dd36b911f4f44654ff6db007316ff4e5afa2b6e060ea81e125fc12a217cb03 +ae2d18472d5f07759848ef26651aff9c0a358401741e7764640b38fabf3469a4 +c15068be6e3b472f6d88cb6a6c5c63c79afb5ddc7e7f8f4292319d5f486770b1 +8100221a73178f3e89eb2f5ae2159a57217701bc70dcfcd2b41b1e0990361c44 +0e91714cc5efd828292f4245ef9875853fa464a5ac6400816652da269d36dda3 +836ba1041dd73136db41d8c4b587fab80b840bfbbe4723e1d6fa16c7c60a9b9f +d1be6b877e67fae5ce663aec6aeb9656b2f30089c8e67d9cd9413006a9b3d61f +681808447f0e7f91dc11cd37c26e3e8f3f42d03e133b12ac3a8b0f65576526ce +326d0359570720abcbfbd94c8b2420484b6422f597a4100bf247fdbadf6679ef +cf26971fb787c2fe5af1618dc6b8bd461c34513e59cee4b85ade0e56a074bf3f +bf12d1867d278b420660fdeec8fe5a611848e761735faccfb98a49f8a5ec1902 +4f47d16b0911139f7a9dc066c15101452871fcf9fea332b80adc078702f6dc75 +94c0376bf8cc96c8a728db4ea9abc5b1daf2efb7be05dab3c6bf7ad4f462c358 +cef7d3a0e0eab7cd1793003c1d5863bc165af3ec845a12c9c5e45774cca203c4 +52450d3f0d8cd3b5f9147ae2cbc73693d8c0f71d74e0408d7f8af9164ffbf3ca +45ca5ba483123c7f3ca5154785b90f32a7189f3286f1821e36ba1ac48f4f3178 +a97ed1e2a1279358e6456ece5bba564455f11d1ef0949fdeaa25168b377d7af1 +f3d3d75973a0646d8b93fd27f4dc58c0f27800502dc773186fb3a28aeb53ec64 +beaac4c5066b5def64a42807784e677c82daab1be5d6b2a8895e824712c5bf08 +192c2f3efa73ad809cc1e800b03b88cda056945c507fc6aab629cde9a61a446d +d774e878ef009be9b0e4b5f48602559fddea82ad860643969158582a9c418185 +e13327f3e617fcd80183997b742b35429e050839104c6a9762001d3728ad52a5 +abb37929d054efac2bfa3a27e161c3ea83bab536cc5fd8357b7609cf76e410a2 +a657f12435c3336e8def2c5f2f5c5a9fc2b7012bf68922aa94e0eebd488c4283 +3b0dff844c1a76dcec5a6dfaf096346111920c330bb6501637a22a58d854d215 +c461dc4e0e9b637fddebcb88b104cde0cebaf140298a4382ef94c6c9cdb0e63c +9ddeed510c3817eb7f1764f7cdddff50ea5d461af6c413cb982493bc346699f5 +aaec6b4702067e61ad0f2ce31d308b56db21d7bb5af4f1509be5ae65e3ff20d9 +82a8960f1d0601bc102de5035d7ab4dbd2558fd6348929fcfffa24b16cc54ab2 +827ae2d8af0ea350e9093cffdb2e21244e1bd645ea48f21023fc085a67577755 +0c09cb82b0d16302a4e67f80a98508cdd3e7d66e9c951a41241aea3ec3735901 +1b25259739a4cb7863103bc3f259c929bc8296406f69bb15d102dbdbd823a84d +a9c777de2eea92af61dee0cfb16531d8df5fea9d68757bb86f3ba261c245ef1c +278a9746c0b868c3411b826682283c3466eb7a93b863f52bad360521d60b7a02 +a43af7842b317c82776283341a9047073e802a12ce4cd65cf6c5a7759cedfe56 +bb8cf5260a7ce66ce410e4e87d3449954be5f478f27e32beb05fa23dbf4b6614 +a5cabbaefa3e764ded60f74344014d496bb07a56f535a6cc25dfbbfd9f6d110f +7f9bc724ff18d6838b0b1a9edf2f3c389039813dba2ea749970889a4777130b4 +3a124998570182c6550c5a9947b89bc1aefc3c21eaeea91d227670cd27d8375b +3ec6d85de8af1626d4c4f324181df7812bff18f2146c493c9bdc6f48d0927be2 +1746bc146bbb4a126fdf15448ef86191ef644fe9a48ce7a6648ce6e726a81817 +a7992e04276bb786a9222166c1ef100d6c011a4553d864c09f7b85dc33b4c175 +239c58758aa88e340d3c90bb354d2c6d7182904e95504d00d94de424c0f0d04c +f150324921e7f086290f507a9c15b4a119993b3b5c294bc43c40c7da94acecd4 +844e1ddbbde58cddd1f0f0cfb828941401a48627e38e80c1e08fdaba1da0217a +1c3444aa8059dfe473ccc589077f3c7af11cb91e68ad6e7f42dd8e949236b70d +4b4a5fd88c4bc90b87addec6f07175323721de086ada49070af69ac80d5e2e82 +9cc491797d5fa756094e953f824ef09d21f7db694e6ca95b44429e58ea9cc1e5 +f0f2a616d0bfb0b4950adeead946180b05dcc8e374553180402939365605d2cb +b37b579e277ca91ba474d3423838f5e249d20ece433138db1cc799ae53722251 +160611167dc9b4c26146db926011d738d8aa5737c40898ed1cc7229b037b084c +68f670165f20c4f8a08727c1dd6eb92b394faf04bc77e90fdf3da1033bb7c3fd +d33d151dfc599bf9da2d862c6bcf393df16edb7b223285d2d4438e53bca343d0 +7a28f892c841766627de275b6f74035eb047a7de0c6e43aa5aa0c41ab5a3759b +0a8ac2a36c0c4126424fc41fa68eb69bc5f9211a912e89d5e8174b5e27d13cc7 +08cdd7c9ae5549749aec7669cde41cfd3350a168ef14e22ff28ec57e2e800b54 +656ff194ec9dc387b3989d1914b50d212c731ca0c752062772c55aa101b72fcd +0690c27ba648761ef907db118873161d4a86c1355c875a8abd699bbd0b75b155 +cc2158a7d7c4d7a2136d6beeb01a3149a93cafcf5b13f07d4df811a23023899c +73352156347236ecbc2135509934bb7da79777cde9a6edf50bd3d741226dd875 +80b657e3d565bfd82b87fc9dec95c4915bb969a449f8b7371196667fc69dca00 +23db54db2353862ac26ebe4df2eed3cd92afa9d11699a3cebf2dba844197d502 +bc288f2dd6357e905d87f08b531d949d982221dc77d316d38649f70487b14484 +0f66d567180ad91a6cad0ae5ca20ad91d74f24cb06467d9b628dda5a5a4a0651 +a79e61e5456d0580b6b010ba1c711404fbf7a6e9be3e36270014f5653be9ee40 +419a96c8085276e9e341cf598cea6ac6e524d11bf8c63693c5e20b20a43a80f8 +6e41f798e7ce8eb88f2a0326bdeb96148ae6071acba35e4b5a5c981f0a0b1dfb +75a2485b07644baa16e610b7fd06a67d6b9e884425439ab217e89e3e2e393eb7 +2811dc3d1be43714d6ea9a41109bb90731bb83bbbc14919da1e5a6534a1f663a +ee276d2dc2a8ce1e9dfa50104c66efea41a893e3980409741fa3c149dd41ea40 +f802785b2d0a170b2fe5081469792bf2daa1904688bfc0518a4e70511b641713 +838a91750115716cd9be0aa19d2c466eebe4bcff2da969521e558e74e8ecf065 +f03d62b972572008b6f93cdb6a15789cb9b703e32786da8d9b16029f5290edba +65a1235d78ca22063623735626e962dac4b5de67c7c5937685446f74413f7459 +384d7a60507c144e6d622d7554b72a26a5a95572f1da765aa23cfa9548c33237 +0319b25f8aceb5b26f50bd2492e0b58cee82a1ea363764778a75396b0d8806fe +d6208844b3cdcb1e47972002e5017cbd98097a8739713b7f660271fb821da095 +90fe4aabd2952cec384983efc97cdcd430768b8f8e31992f8bc332a40af48222 +5c940b126e3e4e80148782f32aada256737ca6479dd9fe53c74f0793f0861f88 +b750d7a508fec57e5cb93986cd4367bad1f3e730e04e7c63cf4e6234ca4024eb +ffb67305dae82898cdf86830be6b416357c69ea1bf3749d912daee0b454e686f +8749f80ffdcb60630f9c8ee5da26565e0978af401c4447f93242d3918e45e83c +528f2c1b0cae1056a1f70d8c0cad7f47a30d75aa875a5db35e0a0474e4e9145e +21f6ee607f03d7e2d42a9fd11cbed242d09cae0c1b893658fc2e9a0989a4f470 +89daa601b73311e42dee11cd34c8c16e2432471bc8b8f3d27a413fb4c59dd6dd +ba8f6e6fcee0d6ef74ff123967c1cc9d6d3e683ebd1e9adb099e2dc0929f0850 +ef45d865633cef7768a05b1abdd79b9102ce0eb1aef614c7c3a7d8eb99f4964a +9a8339fbe8938cfc20b92e42be5135cf737fbf40933ee4f4614ebbadda64633f +746e9ba93f03f0f61df3713aaa8b6dad9f51cc3bf03cd67a0f8ab1a5c48adfcc +2a18f1e473b8f46b690cbb73f3a8826a97ab9ec26cd27bd96a14be86d55592e3 +b863edfc7bba0538f087fea2c8763084bcc5b78a85b9190f039bc21d5fb4077f +22c09e3e63325f7e59d179503dd5a5c8cb47280a65c1804527e269f16d64257b +32868f50ec9960ecfdd38396f39eddfad2308d41f9c0e953bfb34332de83c9aa +6bfa6772064ae23a3f1f34e6bdff625582e5f8fe3562f00bd830e61358fb4034 +9e0c47600ec33341362a32de8980709e1bba02c90cfb13f5e6f5db613a456f1b +59b8d543c4e3ab3dccbdc5afac4d11f6ea9e10569996f71c0a69bffa4a379cb6 +a1f7ae3738a4d42bdc82805327c989e992590f666328b617be24d744f2c80d50 +3cbab9d026c045b5322f17c0fb7d6575433343c63b79615d8bccc8837992b4dd +c083f58eec9aea4597267f71bfff9864deda483ec8aef047c5ca3ce79ea23bf8 +261b872085835c90fc6f61c9a27dc5e40e02f922a9bc444f9c567971458df508 +8ecd0fbe1a0bc6f32ebda1b99459ff5a9e82ff2733ef5f8b0420755659ccd109 +696bcce1ade2f87a0690541f4e4e49a81300b98fa2b5505959c9b15223af8248 +35ba7749a13eafcfd415c2d7ce4b08f3bbb0d72dc4e9852bbfcee181af86d5ad +72023407d0df8ddcf13f937491ad34554535d82b6f3c196c5a8941c7192aae4e +aa0fa1b01d03e4e3414e68897809c9303784566537eb511bd5a1ca9b9e93f0f6 +ed8b98fcbdba093c32619cc68481c7511f0e80c1eca5c6da2dfef7fa0afa2663 +cb3159d68e0b9d153e82570dd295418d73a69074a53878de896b13a82d4a87a9 +f6b5c4a9424fff38f6dcd4d1f60b5f0feba7dbebf6b901d40e1907e0fe79ccdf +adba8f453d7db198bcd916ffa9b5d28b405ded23e684034a0b468de798e1989a +e0142419f42118b7731d4f8e5fedc15db9e80279eb5ef9bd8e1c52287f23d250 +3cb68baa7d424517cef3cf785191f4a927f75bd5111765e24f541277b235a522 +732409643c0f16d2989d82f2e9fa128e3c47cf5767510ae0076da2de9257d529 +e47f93b6f682b6893010788aafd9046f411cafe2ee21bb351adcc14f8d479cf0 +b4a8281c93d067a5b8c59f4f3161e542dc9175689bc883cffd3f8dd7cb634c66 +865146eff87493d955018723da8ffe2a83473a8432a4cfdc673a712d15646c6f +ea84bc8af5740c0623e3a55383eb667e0aa45e7f72ce2e22eac1f4ebd4dab4b9 +c878473a1c2605b8300e402bbc7d6d46d2bdbee6620171eec20791e9a0581e61 +7f91eec3b26934d7add813437122aeea8bbd34f6003ec520544db0a655a2490b +9314fc64e7eaaeea0a1062745f600e285fcc082f6183d2f719396e5b7c2c6615 +8364be172ce150337eb679b9b34a26692e6feb02af1546248613bf2e36150769 +ab337e2c57995fa8267756e76770356d653887917210761043968f654b0d387b +936bd887c8436f5a08f6676754e2510ae99a324c6f0fc6942be813ca2ae2034e +e8a1c5d41b6730af5bafcfa8d402126ea8e3bcbdae8292139ebce25ebaf8b152 +cd3bec82671635035091e03e9d24dbd6590f8495dc9e190036b70e02756db4a1 +ee4751da17607f9802f355f7c5b2df169660c9eccc8157bdee610f78aed6df35 +698ba5a7447a63aae6d9fe00a3124cb15b38da7b1467ab2930436f70b1c102b5 +f67d44949634b892b2fe38c36b8991e94c808d67b818cc3203df3a65070b63b7 +d50105061d189373582a6701ff841cbe43ff92e92a8f24550929ed50312b036e +367509085f0347369b4424b861471836f951b97444009aced38283fd987319e4 +6b9b271efd11b12134f99bba51a52ac9e31924ac848e99806d86da9a5caf867e +1c458f8adfb5c24e6219e920287a37b0d96f812666ed48e0f287868ed38319be +3b6fbfe0f1a988161cb64a7cd4c9f065ec78913ad0cc60163a5f6810750d2286 +298594c4def5a7a4c66864b1ca399b4a8f9789775e2cd933974799da178e1cda +f78506452b9582c177fefd3e3b3d08d247f6bb163f01e5827fba55e11e1dc153 +336e6a832b0dcb3d2e7df5423002264710c983dc6853ff80cad2b5cb7b0414b8 +c29939dfd53aab6da8588e79af11bdd4602c5628c0f16d547eae72a654298a26 +28c8f13f9cc547564c8b577af1ac991db4e2e1735906e9187823b9010864cb6d +f04f8cbba9c6391662107ad4a1b9eacbab6ef0f7811e38fb0e1f1ef105ee406a +a3a9aafc306518c89d92c6296376c64c8589ffc9002ed44f57840903076ec7e8 +68682de7f75ef220e63e5c26b13c4f831459f23ed40c5b3340c389b6df4f348d +5b9cef5284493e120b885e25859893ea9cfafdb979997036a3fb46916d642fcf +706da21f6d8d90809459a27f510625ae05c302a2f286fcd052b1b1816e368f30 +dcaed1fa71f8c633eea1b44cdc74a05ed3896aed3e3ac67a9853fc48c67f20bd +4dfa573494f6274e088ff373ab905e1479571f0e599e04d559b08fc49f3ad947 +4feaa74d3d0cf5e7dc369ed6882b21e211009404bb47e27cf20b5d53f02f4085 +761c73195540b544eccedad184c4effd2b0053fa84d5ae4f81ada45df697d062 +113204fe9135bc3d3008a8b3ba444f3530546f299774129e0d1799fef670fb35 +a8ca117ae3556e2970c1ea6552f8de10b3d2d3c70d77d1452c38db53cdac73b7 +dfbe33121c223233c241bc67a7325c1c2cf62fa78a19c7a9e4d453aedb7ea34e +9bee3ce928680e7ca082a139ab23786f0fc28acbf23022dfbc073c7940dcbdb9 +1f915acd578fbc7fee5079b76e1dd31b5748b4985dd7878b2de452bc5d3caaa9 +999797c6c679f4a0efb213fad56220df3ce9e06184559655930af3243da3a0f0 +bdd5c8d0c2852b1cc69816e8b475fc0cc9dad0725ee623f4d3cd88cb106722bc +c38230ae273c88eddbef1a0b8650bc95fd00647514e1f97ae4259c386d230b1c +d40e8ea57179728160e81ba8e70452dbe69650ba2aa55e2a52d4af6a3d0869eb +473d72ddd3f89d2c29d0dd9636a42a5df01905b5fe89212015cb7484bc43159b +5427596611e73a2494666e337e30fe00a1cef14f5517a9dbe62224f3cda97fc7 +89413b80b074fc928d5482269e438739b6a0211654bc31833c1d12e357318737 +1ab845439f795f33a0a505042651d06d6bfe3a82cd101fec87afde8cf1359fbd +a8fdbbea8d798e25037d3760e2791d33675d245a919b872174b9a93daf449ad9 +45fd9ce410be565a54dadd68a305497badbb911c7d107e585a616de51f62d35d +f8d40dc6475794d0104b1181bcc6d430866361755c6c9e65b1f851b9b445e612 +c329b7fb906eccd869db66e54c1a74061c8d4ef2bd9c3ef99d1aca882ceb4401 +6c1fc4b2ab8d7599c9f7e055bfcda026555cd2c791ee66c7d44b5b2b0647d4f4 +6792f2cac05a8cc7003f85dec01081ee05ca421ff9f7db06e213d02d182bf68f +15f212bc09cf72c5b3c8e882230e9ab9e139045f0a4c828ac97d669c7dae7940 +84e0603e3b52bf10844fffb049e0f0cada2eb09f23a6508502b7bddb097a7443 +07319cf4b9a4d5bb01e5f47d8bcc6eba7abdaa1d5d28f5bb68a576e87b84a8a4 +92d89fe23bb0f6bacf6aafb223b11ef92d19ecf13fa40b520e6ae6d3cabc7c09 +4e50503a0c54a46edb1fc5b1f08cb34c0ee3f747a6835f39b080828dc74ffb53 +7e325298dff9edfb5aa43f8e390d42c066a5032cc526aee746c6bea49eefe6f1 +10be5eaa0d95c87e363d0466d85815590463d808960bc0b02f49eba0a22552ea +20ae5277c2242b2cb34dc1bfade488ffc05286cf1222b825e98fcbf4d73325b9 +f06f753521a12846778d9433cc283977687b418651519bcedd858f45557cd0db +7ff7fc9b2dba107764709fe07a2a33767b68afd248548cba9b3761898cda7aff +3920b85d7095379fc66aa854372f6dd0bba86821ddc1ae34c600f3a8f747cafb +c7f94ecf8cba8b421afc55fd627b04ddd81618e05f84ada21e13d23835d054cf +9005ce579a43b1ea968532718d2df5cef8d1d9d0f63ff2571ff92fe3ddadb156 +525cd797a08b9138221edc1e76e9457ea7f972d93ac6c5d644093b1f76bda9da +df473259f8f64c555fd042758c28301c96c05b5e076c155d76a114256b88e8fa +cbd16c3072a65bd78bd24e8842634293963f8bd0c23553dfb27bfd0f800b3bd1 +89d70619c8bac49ccdea66141c7e319f00d7577e0236496b83b122899a03c860 +e211d8b51ea9de542cb343ff475cc3a2cff4f61c8453b3b3ff38b5ac78fd0230 +079ee68fac45b787c679fa60d57d9b7ab18bd72e77150d61891bfe50bd704263 +b5d7861a6b5323f2eaa60f80193273219b7d51d26a1502d9a42b88a3ff1df36b +55cd37e46ccc0186c2ed4992a19f88882e136afb8f2df63a3b488218ad946aa3 +63c378ae5da00acfcf8777e4961f1e19d10c001a4ffebe6f6787082eaf069ccb +3db56d3932d3b267e64507085e2e08f110f402614dd5ac4e712f7c4799d39939 +71c4918d4b8f216b7bdba9036edb693507441ae36b06d1cef89227e20449b027 +7284053e63d6ecdd6a1fd7652dfde9879db49e373a9100e5f0e809fc321973a0 +18af62718fb01ca9be68d91ae75c8265fe9659a765e3ad180d545b762f1dc0b5 +a14f3dc4df4f5699f98079d5ca1344becc3e494c83a5447c117770b0f20eb2fa +688580b38dc1a32a68a06266e80fd907435f075b2aa6c0623752ab90543243ac +7d06d0a0a5b620ee8b24a6e02f30d68af394be9ab29fd07b8265c50c7f28ea95 +6386093518a0739037918c0d6f1ffbdb539aa7b8561ae2910c718113cf77a256 +7784205cdbc3bcc107e2e458c9944e0254dc2fd7d7467ef8613816363cd46df5 +e87657bff6894fb0716a6c53ec84095885c2e965ab65f12faf404e17206d3dc3 +2d8392894196e39908eca31f2948247aea2441db010d4933a80a9b232a7b115a +5c983d13074006f675a33ba6794158c6ad278bd035bff1f9acf6185b9d7a1c58 +9d175467aa29dd3b5e8728e15a6de49b99b762a24fa0648fe57604e4270675b4 +60140c4d77f2f738db28698b49972ca2e46880051ab4e156f40b703964a7ab00 +e19105cb5169fef6aa83b852317286f073808f418c69311b0ebe406ef87bbe19 +3141a7488f918215edffd2eb130ea00c7a8e598cef6aa4d0cafae9c8bc1b951e +a5e7c8a504837f3c6d3da9133f86d0ad99b7068b819791994d0f88cd9c1107d7 +a625706b3edda1762a075bba5299ba80e89e657961d2c828a7fc3008190c64b9 +013ac4b904839fbb04c588f3da320e8f7accab6673a1a93646511faf653c240a +ebcb831312463f4d2661a000367c3e887485003bc9c90673028e8381f0c4eb22 +7a71c865f58582726278042ebe35dc0a8ba36fba5fcdacbd32765d5322c6e766 +ffe82e75847a6bf57be1da82ffccc7c3f59c987bf92013592146264c6907db90 +347b14b8913c3aabb9858d076e0ed6ba7b90aa30f1fd1466864a54b258ed4a9d +2ccbd7c89259ec7955ee27d1ee2d2476c318ea558369bf70f0fc891c046bce6d +3fb12d1e75fb9ff231ae18839b4eef27e60acadf27353597b10ec7eed55ebdad +a60ac4cfdea2a39e53241ad5636f993bdef01463e81082de8b41b0a83b6d8654 +f59aff6c192159047d7e64bd2bc62f425752d0a37a35a2b708289b4d419ad773 +b6273514b4e0aa8f3d8347f5ec9a28c932cd0cab3bc1ffda46085004a17f20af +8f1b92becff113b726c379efc313abbf69a70894731dd50f88c61f60ec0deae5 +b4402ed8a9e2dfc1eccec4c3ba65514db3dafdc1b4074c623fe34eb23ee1152e +aad99827508566c232737285acfc38d8c9414335fbbd8a08d9b19b8fc0676429 +0c5443b1530d7d52653a1c28bab0b876f71c15f9c99329cf738cc95cd296cde4 +768168b8b056017599f1532bc86c9ac482699f7796f43d096ea329825ed91c5d +a3b52ca9f7c5a9b92ca2999f7591d6562a93be5655d619fc4869d8d8a5e2bb1e +9137d59f9fab575066c34d0c8d1adbf06d76873e35264068512e6a001106ada7 +ec5a0c6fa495873497640c0065e754632064fa04f145481fb2e039ba7493f5e5 +09f0cb9e66a3089d8216fd855713078dce9b8cbab3905c63d2771f26f26ff186 +c137fa9351f1d335316d242728c8d89f3e4347381696b119f82ebf1cd96860c3 +379c9c1070725589a0fa9626a59bfab6de21a50e600815be7d6af70edf059f3c +7414b2a496fa424e5c0863f36fbf4299ba33828480dc75d2854e1c008f837b88 +0e30768d6bdb4b26f6076b3bbd9526ab9d3a5784515520539cf18719d48d9568 +63f09677ed807db69dfd95d5f9a265318daf569a8e4b99c704d1f0e7a3be4d6d +5d0e4d129fa1b5d8d26d9b9965b812020360999a6147b707dba37890bb2696ee +98622237d4e7ef1118f685e1cfbe8f78937b20ae02dfcd3a5859d6a4f1a08c81 +6f9558627aa174d26b5c33d38cd7972fd1fcd7050313bce8503aa9ab62e8ba95 +fae03a1b8eadd2075e249e8ff59ddb3ac07d267b4bc54ef2e8431246c0139c94 +21c0e2c00d337d535ecec130bf4c059e30410ea58875979829427c54ea6f760d +6f6cb09b7f48eb6d579af6156c71e51e662a5af97f7bb7a5eb9ce1c9047c4c5f +9d50f89b92494f584adb7dd6ceda9c7905b0e57ff7c9c53af0eeb81cc3eb95bc +a8708f574de82ca496d423629b3d9df0644ef07bfb8f43e5ecfdfbb79afb5081 +774c908f167d44cacb2540ee143858bcf8ae4c969d6adeb30fff0c256bc191b8 +a38011cc11f7352179eac9e039152e00002e941592290175fc8c30abf77341fb +703df70ca47c3ef54ce613eed28d9f303113f0a0878708eb264892dfd863d1c3 +b82f1257c40ceeb4c2a8cd54ee972dbf438f9feaf0052788f3815726642fdc33 +9f7aaae569a8ca4d76a17f3a48d85cc1f926afa47ec0eedc829317cc832b43e4 +1c0a1655ea3393cf7fc15557a248c09ce11df06453a130fb2a9461ae69c533cb +7b8756df16cf9d6f6ca5a32eae6ad70c3a0a9d176116559380ef84393cb44e69 +82ecc4b7cf336b078777818ae90353f1357d058cd46812811c4a0930c56377a2 +3ea8889ee4c95fb3331a35771a3cd78e8cdaaec9256c959ca9873f71a06e5066 +2bee2342b541805242d28b860a6139431bc02672d34550a909ab85a455b8cfa1 +85773081178a29401b1b3ca794d9c5bdc4b848d0f4cfd69a7f98bbc1497fe4ef +880f6862e2a492524b0b92f5768fcaca1a1f47db980288d8220523779ebf17cb +2147191098d5f49a09c8f647b00c5848b8553960b2a9e7df67be64724a561519 +5cff2d191e70272427fbcfc4941de630b0c4afb41f230e41645a0a49d29017b9 +e4e717b41ccd44bcfcb4b8b955ea847bd5f674ec89b2ec88ae3cd795ca1dc2dc +09c21ea5589264bc17ae372fea55b526138deacd72fe2c5157a2d4d076a12c20 +2cc5627895ce8ea7b11c9f58b48c8cd14a962cbcecc508a29d9e794ba22de572 +b8155da65549c3a4ba7459185cb1d4dfbc5c2c2bfab19c2ee172140b459a598c +6f818bbce13e7a6c7ed786484ef58840ac1b7461922905e7a1ce67f4fdf83c07 +a756c7578e7c7714bf306cc5d18847856df650b635f08e2dac52cfeae828a2b2 +f4e8459577e7008ec0445f378de02310c1a460473d63f7223cdc29c1cd944486 +c1311a7f803f80b441e4395f44c578810e8ae6dee64a9db68a2a9cdaf6885d21 +f480d78c920ee279f3b50f5fb06212b1b1ac62d7e21afae7bc6b2982bee2cca2 +f042eb6320bb69942c5541e5acd8f64565ea956366d5bcfff90a807277233754 +5c5ae6122a1931cab34425a93600a4fa919c9aeb0dcecfdd315286967982df57 +586663c789495a0ae8d4aa255e2bac33b23cfd40a7259e90021786d62cf558eb +817e1591e4aa336dfd0bb42419920cf29832e62b9ec4b07daf023fe55ca4cc6f +2691e1028e87930f98bdffa54397897b1102f5e8f720720545df853b0b523c26 +37c023be7b0c326770ca8c57de5bb86b5a802869627a2b0db966b9cb7a610d29 +7f0a91f2a48a093fec14b6c6af2832531cb57cac7ef21152aac4f1b6c800cc95 +b03141821bbac69485b9160261019aa97cda2e7c0d0580e0a4cb2a06d9a78bb9 +f711d2c8dd8060c66b153fd5914ee1a861279ae759c43a6b3ad3fd5c8a7c6754 +140bf8e541b3de9b4caaeb32f2b670a891f96e5ffc46bb860b1eba1db06691db +02f15ca84bd65b8bb6fa65b417bda821894a01059dcece02a30cd708f1b9066c +86932bdd2d74f632800d703dbc10c002f70c7e4f96b5162d388286bd22a3c4e7 +8f6e2bf4c9710d22700b88755efccee724fb27daa0e27c106743070cf9461f50 +caf42ad21f31c12de754242ad6b282a48a8ceadf3242c71d6192385b238e2fd7 +baefc8b06f7965221ac8f2ddf3c73c0bf40d297ef34ba57fa9c0358c8e83fdc1 +d59ffffc8fbc1fcccd315d954b945e0e0627af266b1272146383563a52e3bb67 +30490dbb2234b1e3ff5f0ff22a25ca5ca277fe574f8cde80d456511080a1623a +dd20afb04e19397cbc6d38e70eb0eeb7afca7c775ed65c7f8023445c2c3ef353 +fb0d306f8957a9512e727e3462ece44ea1a1f0cde3d11d9cf796aa92cdcd3a44 +7046f234639d52b16a8162be089659e1e64b77038537307c09dfdd2e5de2b2b9 +bbcb14225d91a9b91e4443ddc4e30bf9f950c6799833c4af363b74afb0668511 +4cfe31007a14b52f8a91c7d896ff13b22d043b6b943ebfefd5589a8c5d86a1f8 +f968896da3071ad93ce32c5604037321f20d329ed8ee7460e8bc1eb57699cc11 +89431b8dc29ccc26985d898dd7aa4907fb287a3fcfd487393ee789e0daa4a49c +e443300575cee6a868c1599fd2fc76e3c689e544a77e6e85af4813824f1664ba +e1eeacf84f757e26d9a7a5074d4d7719448527f8b46ac88c081a52c98338e9cc +4b6ccdd0077738f002825ada3ff4c484c5a7c253bb1be1fd106b100d54fa994b +75a7d001d4431f4322ccbeaff6887a352912a745fb6fe62b444c9b3b73e88c83 +91ad77f3dcdc62bdf81d147ea6bc03a1c48543c546f92612276358eec846ae70 +75987063b20d93da9c44ceedf236ed9f203a03dfb8b30192501d75ceede38093 +281f4014eaf8a3d01044382e57e0e3d2b03a1dba8c7da747b456470343564c51 +b48fc59415e002664f633f2563786e80a89f369af0833b7a381cfd3302134820 +97808dbceed22294e2ea05aa750ef944a60775eee96e02d28628dd0b115bfc3e +4a19454e927b2236bc99a3b116aa5322a4dffb4dbde8263437dd25c62ee1fbac +9eed3f264441982e6160fdab9adcda3a849d06f8c35abba81844e8ba2aa5f06f +d723fbc182eb8927878a86fef238e0293ad6c42dd41ed020d7b6f26fe224b3fd +995894adae1c301c9edfb54e0166db3787444a057b2407a33db046a4ed9a1059 +635b5ff06b705b4af8296be77cb8273e6ca8dc756747aed4ba41b6a2a1fa4681 +12109dd8de5f06839fbca58904b1113239f9c6b9e735e3c0771b8ef2cead7dbe +97c8ba54f5b546d68f38436cdee49991e1be80bc273c74473d445c5a33af6eda +1cba12fefa5c2b33472768317430074a5e3dc9746b44601eb2dd633b98803287 +a16e5b8ad9206b62ed761868cb054e9deca8669a4456c8c4d750aecc7985c278 +470e4221ed1035f2a001a3570f0c0448c325308963e5c49b630e1d7b0970d532 +f5711470ca047d62872826af9ad44e3ec54386ca78d221e712dfbbae7e7d78e3 +d4125d976bc796bdb29e071a6b0db0f024fa77499ebed03f5115f2864e16bb57 +035f34539f9e91fb89cc28a20e34ac2fecfe32c6b488dca969c872e8a8b83889 +955f8f4c2917f58ff30ea7c8a5a0a07915865174d761aeacb5bd38e5dff4b2bd +7cc23e8021a567fcbea9811f55d734f464d101249adbc0c9ab8864c902eba01e +16e3bdc3aed922dd27d1ed693e3568be82304adcfdc28aebcb815bafeb435854 +f6f858dcec7528418257dceb1ba1b76242854aeffd4af8c48c95496071458a13 +594b52dca9b06b6b5aea3211ac1cf5705783a7814d1cdb8d5a88834b84ba208e +950935223f090e4837b54e5d99e9223756ead5160993201e1883ba689006399b +47034fafc46dd2de661bc712d78026d362e05deb800fac4b8c4770fa5c45d349 +4ac12fd7a91a336864cfd5f2d68b03cbf587be1f2c0d799d0d95fec2b4896cda +172ee74f2c13defd1ab4fb05bce2a2750f3c4d0952f8451fb5c736268bd36ff6 +e01bbd0b324d1bc364e3061a4d5c292cd5dbf73cd2f39907ad8e000706186d10 +d2165a3dc4f46981795609d095fe8b2aa8af73c59e72c3c823c6cb330c878375 +8cd5682b6ec6f23904b3f827540d09680fb391b446494eafde0bf1bf2386d8a2 +9168d29b8378f3dd2f062bb32cf24630c15819a94bf030a4a38737eb9c206713 +e139a677f3c1df1d6804d5966f259b44bebefebb2f39f024d969b6388983acd0 +a4566c29dabbb050d825acb7dca9a223df210d49a6896b6395d6dfbba0ddda7b +324e5bd3bedb728774c033194183a904ed80ef21c4e44aaa22593b728cac32bf +c3558e7105c25aa2cdb349d76e21b67aca8f8c147fc79bd31f23fc4d37e2380c +f8388f594f92679931b90fcd0111aa94b6d1a97bc277d543b841dee3d0dcb29c +66785ce5a8d57ad364d8caebea4d44777f6f6d98c9c1deb473f09cdef9048d63 +26d7f755d0ab0e344ff40c4a3214108cc4f3d0f2c3f1e6605e24cc0b505bece0 +e68ab8d8d03dbbe90d70959a901677c765e5a0217681a70720ead4c92e69f905 +517d387a13cf0ac6e6f67580c63abdeaf3d2b1aab71606251369a4c62a682504 +e2ca6f1b2cb0217a97918cadc778464843df5bc64ce53d5c316b51a88caba2ec +9a82462893ba03430facbbe10da9bd40ac580de81cd1cc1481e2b12040fd1eec +28aa63e4010e7edb1ea207cd0ce4894999dc6ffec2b6ee67650a7da941818aae +f1f5b8ac2522bede838ddbd30f403daaa718fa1717a427a4a55df5f13bced40c +6e07b825b8d9cebc2e54754e2d2a06260b65a4d3b1c8e75f3da3fb496c295c7c +71186c4afbb705ed27814f529739dc6e139aa6efbd73b16d47ca1f4d3f78aa66 +47973698d9a0e0d4238e0d27c3afaa9aac5fd5eeb837548ed46e879b4fe58951 +ab908c64d8073538d18dbc5ee2bdc05691f359cf0d698a39fe59680f70a64439 +44f92f961f7d7dd641c4b3ea991a30c25d544b2ddbc6403856bec46c052b314a +44734a9c48b60cda6ebc42c26d6223e978afdb2e6506f0233e43041f99eb8db2 +0dd54cd79fc5907513a86f9e11a7ff9c5dc5699d12513659f57212d689db27bd +dca86c04be7be584f77c1943fdaabeb2b354b1a3108d4a0a58f9b8c0572682ac +a1911651f37f490a61eeed6f8794006bbd809a2a6e6813c9e759076bf79f85f0 +fbe71b4ee87040d1713657a57cb25d7b6df042bb9791f251892fb4968afb7f80 +c4b069cb4eb9b22e0e98e19e3d6caede07d9ced397e04824c5a2d09d09ea0faf +12e6eb840a43904fe5ce3421511e46d860929e1f72a285a3510a6eecc4b671d2 +a1240d6c584933d4f8de2d0d3bfebfc3adc78d63231a51c00a95db0d8235684b +18b703c384e73ded9e2d58fb83c6b76ceafcf743a4d757a89d922b64c59842d3 +f01605ef7805d33875e54cf0e00ec4da251065f1b289733096c66df19241afd9 +0c2aa05afa540554f76816b3bccba52fe655bf89b7c1baa0dccae3018624edbd +6c0fd58e39eb116c2a77ea129f3f2eb76cae47b2a202299b37ca6e0795f6e6a9 +86a6e2174c3e73cb2ec30cc8cdf51dc04068cd33b9fc33c7e0fd8bfe2c5c6d94 +a35d7735d0283ad84c52fad1a5e1e6e18655c8605ae1f780038c987b5737fb59 +6f044006e0f8e505e0ade1518a4d48b29b60f5c3cd19e9d36ad0e27a9a7278a4 +8e679ea1b4ad3c36c57cffaf9c81293ee6f30dcc3b356537480c973f97fb4fb3 +7054852442e64431b1764b50666194bc04ad70e4a80f29e086b61c5b79e1160f +a22cc7f6ae35ae45c24c495aa46ef64d0c2b66a914ce7c21de38569266f3c7e9 +8c8a89aa60d22d0342adfd66dadc73dd9481a74fe9fbbd8b65e6ec15251e131a +7d60d4e0f52d6bc73c1acaee6fe332f247f6e6adb84991a03122e3ac849c9250 +dcfdc1e270942a9e5662883124afe0b531f724d3b2c4381e35d7e1c3d5c739fa +fdfe351d0f5b651456397520651fe34174ae571ce66f2e3dc76d814485ce19df +bdc84807b41c22bae70216d341cbc79e13a5258ffe518248ffe32e6f6cc54426 +28f57a1a7d15228447f929cc8a79d0bfc646c237800afc675bdcc3c3bd441582 +4b3e7110c0d2530df209fa6220cae437687de65c8e778fb4378f09f30606579f +1bdfa9f0e90b6b36badbb6e7d812a594cdd8c8aa907d8c2b3822bcfc8e9fa3ee +c28386d491a5fdccf9510419bb25314e110e51e2a2fcfb6890c821e7b2c0a4db +1150dc02fa2bff860d6090e60e05b8e614f8c4bd0771ae0264103ffede3ffbac +aa178b74dfc2fe81b24b773c656e632dba1892a7fe5c405dfab86972f4b967b9 +e5c5c5120822c99307966363f6f025c229130f5fcbc0a06f58b5f15b7d78fdac +408081b83fde61681e3398b17f8c40b7f40ad59c5378c627bac692a8294457dc +b335e0400d8ae441941a233a9d6570883cba9e9e7ff5360e79fb5e944cf4eb90 +755a87bb8f7582daf7b0c95b1dc403d00e8a758319409f0e68345ee2030b9df9 +06b3e25ed585e703b959d91517a79a1937f368789a5006e41496adc614d1da11 +eda102f79c40e83233a35b4fe49949873c68c1cee341bdfbdcca0fa88feec616 +ae388b0eb6afa50ada6d7fd8cb40dc3c8b433dc863544111a792efd8d66c1e60 +67abac191dd421c0b0d763c3436f1f4ab2a4f04c5ee12418ab28c4a608f8b7ed +ef20b902c3198281a247fadf84811e1b34aebe512969ec7f396974bbc1b27c8f +c2be8e65cbde6d22201bb536ef16aac4b7b5bf9608f4bf233ab92f4bbf65161d +1bf7e9eab022b8fe6ccb6be43e8732fe9f758977e492d82deda6fe020d0426a5 +710cde0f9f0fcf5a61f94b0c89f95adbc85365594530abd4bd9fb7ac88bce722 +dc1dfc01351ddad66b748a5d14273830045dd98f5da8ea31519bde3cd64a9e58 +9ad93a390637940cdf4aba5af19b2039546ec2502340a952cbe2bd995caab163 +e11f7b74d4a92600fc30643d0b23e0a40009fe1c92197d95824207c75c81e15c +cfe1b508511f26ede8468852f80f2394b92a109628c01dd414aff5ad2fe629bc +5d700bf1233e052b1504ab4ddbfd13a8bba907b9fffe16d1bae172e90e4dd680 +cad5fb251c1d42fa7c405aca8d5c831cdca0c32de5cc1c579dd2feed00cf29b0 +ca06af69bdf6c05fe07062a25ead6ef2ba332492b56926449e4d1eeaba399ede +5a74d5a8d4d40696780346512b3070cac61e17e1359e9bbcfef81f7706a88801 +72f59b18126dd3391a44e9dcc754ccb9028b327e6f9d111deae18480de7d6fbd +bcad0a5f6f5ccb31fbf1fb451caf13e1fae721243cf1c7f8a456c8b6735256b6 +4a307ff7f885fa8059999cc09b7243ef152f3e24db881c0e17abbd23b341c7a2 +33c1f32af9477a8f7c548b3bbdc36f1e8718f51f69edc2e188651f296ef26ab9 +1c8198bc9c979e7278e68ff1d45ba6b5b6709bff19a82f62b950522b97079a04 +e0d3385b591316f3c6e23ed81d8dbe025386e236d1a1aaa5caa6ea216460c8b1 +d03fb5d8424c9270a05ed1dd6b8a94bc8dfffa364dcaa94b552a50f25bdf3fdc +5375d950c2ac0e336844af01d419cfdb7637d7feb548512037adf8d2c6880c56 +1cbd72230995f87cf262ce4a85a69c737e863b3d67cb69c666a0295bec5d07f5 +d6f6f3c416af59cfb6fc6f3bc789cde377667984e0605264f92df509fa2bc575 +e719e9b11201beaa1d8d0b8d65ad070dfdb921c43e224283b09eb09352421352 +d8d129d6e734c1fb8483c7745677b1d4504137c1318785955054f432d1b44f57 +c56ca3d8ca8652fdd942d03e06758f3fbdefca6f9b961d313e4acd3a7a83659f +4b88c8f2828749f49227e51300c3d008fd73d0201352c4602af41f57369267eb +41c97511fcdbc21d62e84104823028ab3cf743f83d5eb0e13742e7466ae5006d +30aaa591a9bbb197e853de5c7959784d9822e030bbaa46207857f6cd4b0ba80b +4bc8885e02aae8662c1717a198a5b5e567ec210c963d7e773b2e0d0d38d252dd +bb41bc9c730b1a15695c8bc9089cd5f1c192fa76ee88552a109bcd12ebf51f80 +5eddfcaca70c38540eae624a5da94d25947a56cca5c479496d2736bb4a3b7b85 +f635b6ad111915061cf4d144611747de8af6d31bc284d2230c33967b3ae173ba +155396dee935f33703c1edcec21ec872eee1947bf88a5b374dae028336f3758e +b9140369eedc9161975cc979fa619e4e205e5952956ed40c96e62d58242b9af6 +d79b8abf6c1d4262ac8b06f5b1e7d5be0075e5f5ef549319cb27ff2249abe912 +9d9bd71488f5e33c3974a2383ac182f9c827c86b233e6d17024fd5a52c9152fd +3f18fa6487ed3cef6850dbf294fafc00b86fbb27569b3f27066a883ba69a8ae1 +8044950f2e5fd7052d528bdd0a29a6304aed1c90bbebaaca4a9cb29d697b6a7d +3d6e7da65afbfa39ade6ade1290bbbfc1334cadce95140c777fe472461d48af4 +5c8ce38cced28b9e17026ae5b31938132b10aba90c19d9c6d6409929a1565598 +fea9879e7da3c571ce7b413841442c9d2f0ca04984cf23177c9cb9c52f801c58 +c69ce636cb3327830439f2cba8657cea64960d3948cfa5fc843f9cbce193f231 +b1b478c53c757ccbaa756ba54451fc299d60af76987008552b6feaa09bdacf3b +5d06736dc3bf403b37ae30ddb53c6d3fd74eb46c02a2dfaf113d0a7c941321f9 +00111f61ac86c945568b440c3fd67128722a5a65fc3f052c63e29ada9ed7416f +833c306a673b542a8e42de6fda9502652bed824d2205c91093c819040c763db3 +c9638a580ee3aebbba0a5f7146e47275d8d19c05d3a8bb3e236933cd46d826b0 +89a24b089567e61c6b73d56f27dfceafee35873377acfffe3a0101f63c161f2c +1d2129049016f97d52768e92e3ddf2d669915f3f1ad324b8ff23805671b0a88f +550445fbf17934c34228fb1aa8c60eed5d6cf5abb16bd9b0358e55ad0cd0c9f2 +00f9ac308b2efaac33edae87a3042c2293acb4c6fc72a545482b59ac3c087cbf +cba59aa4f9f94cdfd6a57aa854124f86fe5f442dabee2a486529a213c6c45578 +5d1f031d8232ced3ab69525e66916f88bdfc834d8a8f78c357affd55cdc6a43e +7ea10bd1058e36868598d5245894577187acb79aab0eb4875424951915410ed6 +d5deec09f01ce38c74290dec6eb7cfbfa08ef363e9b3df98b9ab2a4009bc82a4 +40f527c5ac5c1d14c28ec7816fd28f9913d816e06a964fe602bbb8616173c9f0 +30befed588ef70f635ea5ffa034519b3cb0a31ce93da4f78be1fe55677d3294d +c6dc3529f1f595aedc4e1996c3beae1728ea73be494df48d8b995aea301749ea +ee2f6b4b245b84bb272d19861c08c795ddd66bf08428bfc389c5aaaa4adfc07d +34005381af49af8acbf4324d16a4b7ec45499c6cac7e707fa018f5588bd9bead +2fd9d204f7986715a302c2735679684960263f8f7b8e805f768acc37d98f347f +ebabd28b06197edb3f50c3c97fe77caeb643d23a3f6ebb9fd9d08b968b559450 +4bd119ca5e4d3782d9a2aa30653407d4de122a98268ec1d55424e1aa718b6bb8 +34845a0f5eca756d7768ee2f03186ec4d73d86f9ebeaaf96abddf8bb733641b6 +c0747e733dc391450d3ea7ada5c8e0a943f6e2524846965454ee1fe5ea23fedb +039dc7156f8d2210a3136e670638ee236f560a84e2ec239db3c6728d5be10ba3 +1123036bcf240cdcfb398a9dfdf4b17e4b0af4d02cfa89c728d2095aa4407026 +0d692e5522d6b11e7002769ce6708342d0da684a54537ec49b2b6efd3d0f1d7f +45b16beee12f2b2acc88425bb2362c9372fc0b03f5b49766a00891858bef4e8c +d255c2af6fc2897d6c4c4eabb50a8414e173006e9f9080b611bcfbeaee826303 +fa0f5c3c9f7632620c7bf9c19a86dae5d8f8f9ffa96795cdc2bb9acf3f22603a +5c76063a21074e67fa7846cb320abc057a17e15aad301028bf7155abcff60d2e +0571c659eea0a98ccbee0d8638803b5be1153f9421d026bfafc8efe95d3ca2f3 +d1fbbd30fc61e23fbde66a1d308879cd5b55328a03f2716a6e053efef48af9e5 +f21e8fc26bcce5dc36e0705ced70c738d1b0db8d7a6d6b64de4c07a08cf11de1 +3da1f5b36793e7144a0be7b6e3912e6d5e8e84781504b7448f13618910c8e92b +9435e09487f12fa449e9e898d890af6eb2037fab659ac1b91aa157d8d3316212 +9943a47e446a9057edac23e79f7de354a43008069ea29959938fdb5a29dac095 +7076234d01acdfac2dbda44b6fa5f7b3d4625482e6e72481d9c90ed08e9b84d7 +de5e0be0b53e7d50df9937b3e47b885a5d1cce0021a2f37f6fe71b9ac1f5aab7 +0698d8461a2d58713a457d210365f96c4f31a0ba2ee9592d07adfc15fd64b297 +b9adce74312e75a20002ea7e9a90ccdab10701f2b3dd2cbbf71a9b032cb2bb94 +34a2a429f26fca7dda65ffafb631a773f9f7f36b12d1d0a2688303a883c554a1 +a71a50c712810faa43a694af1014bb4cd3689c3c476cd46acefb2091fe9c68db +eea89443e674cad4c415dcb5f652b438575b0c33f125dda22a4a81fe36011b3c +488d47c1eec5ac7cdb1e68a17e775bf3de307b5c58fbdff8e2263c36e9841cdb +af80cd3ddb83088d409a9b1c9c7b6d13e32cfa28d68bbe88ec9bf78893b9703f +f9e5433b26afe90ec9483f41b8d2213d37cdff6297d285e905bd2c0208558f70 +0450bb34b86e68dd331e8f4233a89b566f5cb4e65d27f8bec4db5dad248b4a2a +d229cb092f4a0bf337d779a823cc5ea78dcd208d2522d96c50d13ecea99b645c +0977d2f7d367d7b3a8ebb4b09463fa9bc5e2eeeb93dcde380a46bbbc2ba72a04 +d28f1866de6029d2c2b64685ca5f23a3bba98e5867b0251d89432b84ab4a4b36 +5b5fd1b7048bca4897e02fe879a1a9430a877bc058b386995ddf0e8df17b1465 +2cfa68f7e111007295dd0b8800393f230915824973fe79ffa2b94efbc9d4f8ba +454d4c6c4fe2580a25ec5569fb27afb8b3c6fd879b255e257ca291ee92970682 +983a56dc55c6977d039725e9918d00e98bd598593ca1d0847b4391273bf5cb50 +31b80f8160396bb957223856d0bee2bdf03eb24245729bfeea0350a7d54cedd6 +f7a23de4e7d6d45938d52cd1d5bec8509bf283a60f3845deedd77a3ab9acd3ff +66ff2fbfc811dd292770204e2c0bb341e79a3983e06d3dba57806cfce215aae5 +c2ecc0f73ca9c98df4dc1f9a64fbc426097cc88dd0ce3bc7efd7c53a006d3ee8 +15a5e73f8199fc632c395093f4755f0205f1d062c42cd4c555bf685196a41e75 +54741594d1a6d78760e79a804c20d119cf73c5b7f2624045e1663e2d713705b6 +dce835fb483ee256aba2e13f6341c5f161c77957ad0c0035e1dfd931510c694d +779a5345eea3680a5f485fd7f4afe55f8bf09a8f832f68088fc095e4dfa1a490 +28a5fb4c5359819be76eb2e5072a6d7df64fc87dbb48550481ce0752f8961405 +1b389c93b9724d7a6183ff9d15cff8ecc79504d86de136c63211529a0ba6098a +f441b2351a64a7831d87fbd74210fd2afb0d775bfe4d1efe0033714bc35719c5 +c590c942c3200e9f59a0c9dbd88289269a1b225be2ee7f0c005afc568fe6f4b4 +f7a19a04950cd2de96dcb6d40804b6aa7b93d13ee45ffa89d1aa6b0112f5b0ff +d7f53da808083724990ba8a9c8cd757275af9c31d60fb7afa399f7c8d05ac3cd +d9bd905c8dd80fa063f579e0b8665c85353ad28d199052eacd4b79cfe233d032 +ee7408fe2e7f8b58e928251df122e81e05ccba3305ba2871c5df1e3d15092963 +8612636b7e6b06f90ab40705a5166b8860d46bc5c3c77ec825b1f4578c830ffb +3169c26616f550251dfafa68db1f6b36133cb57af1da9cb62d94818e2af2af1d +3cbcbf14be745a3094dbdf5248740f1a095da9b4172821de1ff8897d3a804d88 +91d677a0fb9060e510b0d174829cc12222035e41022b96103a7424ca3ce710c5 +d63644564d2efb112b806c050a5b15fb9ce1ab9ac8d9758c6f81dbcfd12defff +d8e3df6b7998d2fff5d9f6040c3a3976d60d0ee91d0aee69124902b8c9a156d9 +813c07eb9e82396e43b0a4015cdccd90df4e7b7ffd2cbf66b7e1080567f748f5 +9ab2ebd561e7c6cba8cdf99a3875892e7da05b1ad54e186c3f3d50111fa7864c +9cce800c65bf76f69e573e3c88d7cd99c0ff058b3d6e15f220ec784e8c0bdfdb +fba4d5357789106ed6700e8227951ee21f3d2170147b10600865865687c112ca +e8c2d9a8504b514e292dcf0aba49c437ed98501a190719aff0297baf890a5b54 +60b98b2eb469ed66b4a58f8839ec57064e85888248016ef7f04a2fbaf437c6f3 +1332bb42b6e501b42f00d07316d40c2591ad9f4125b91b43dd4e1023998349af +10455a7811e789df84b0333f34235436273583641528f71a15aa561edfa776ad +de4555c325f275bb854a37a957373ccd78bfa5f643b606f8007e921dfdfb0c72 +a2a11a24b04e17368320d4d4fc3124e193afa5e591fbc43057bba7c1e022e933 +fa4f07518e2407aa364550fa344152b424fa0eeec4cb9a1b7eb946321ea1c5db +e64798d1f7cdbbaf3e8ddd681cc5b143d39fc263a399cfe8e39bdfdf4917e3bd +d8bfcc19f0adb907cbe6cf7a50b5f593a8a26f046742336389849bd8677ce4e0 +0711c7798de5eb55ee04fc5a0336e70fb2553dc21b857c27f10228dc041dcf89 +3d4dfc40c0ab2b136b4013c1cc7dd1ff12375e759acb607ec8e5fbf792eede95 +a77b8ffe21b109c6cf566471e23e9b55be6ce6219ff3357a61bbc427527b3b8e +fe20ebb42b6a833c3575a551981f6a8cd233657a6d05375af38e63b8c96724dd +6d26ce0dc023798f08749c399d01a1aa5540431564fe34aa87fd38e8951153d7 +64ae5823b7b1c003fcba38db5f20f801f59aa0baf4e4802c7a84edccf1251036 +0b27656477bbf51496eef9545bae40975fc98e490073b813e5b87f4d2965e860 +54a3d7657be58591499961fb36a96003037e6f3cac0733754a5ad259a4cb541b +c2c56999ea052ec49d03b25d7f3d92104b36546c3a53bf20992fc4336d3c8db4 +89e0764625c9cb43c175bf4a2aef78f2c8b557243f37ffce7842b6ad2eb31adf +9f3060a2d77d7d2888f0f236d6d9d1f5a3fe7ca9baea663430829982f3363574 +217867a517b1f8187d8fd94172d4e40289d14043c9df828ff81eb099fb8998d7 +f111b475c585d2b5f72f35eeae5ccc006ced57e916d0c849091bd27baef60e09 +375e39b62a5795d070f0ab56e0aca9c419b142f470995a6f9550457857edf640 +dc9cecaac87ac628023484b4b73f96d8e54e41673f29329c36dee7b5097f2377 +afa2b33afc3f8c126bc5f6399e222469636d96e8a4d78a09c16a35bcdc8432e3 +03325e9459c1ae99fc443f7b8b3a3cb01b2a2b17a42b8121c5cd8c83b375fd75 +dc9ece643406f374cd3359115f6078fcde350a42c5a3f0e5b985f9887d52ea3f +ed7077a3989b0a72c4687d0ca057043da823b59d954139e8b872cde65bc3d2c6 +97d17d13b01c4744963f3e5ca6131fed27e6204e2ad2f78b04b40af3ff8416a9 +19fb3c01e4be5f4b538b400a6da152cf59d1898b2eb2ee3da7832c53fd35d468 +09daf2014fe3f03c96045a3ef2e6ab907f0c895a9e9e357bfbfa1b3c3d1abc1c +a9878880b62fa1c6f86db005d1b1f0d1615edc57231952adcaec28210274d336 +66e70629313e8cb8a8cfb8a7f25943b01bbcd404a5c123e8efbf985768e733a7 +e2876912d7c4f91b1286661b46926e048435a379a17b3813c86a992a8d81f489 +187a0ff12839f15575af999e06b5cf3591556c7c4b39936dfd48a5e711f7b6aa +b0e8090e7ae2e597793230bb4f3daaccc32c880487e465573bdbbb78004af1ff +d90ebf4ddbe21e8865117fcc940ec652e771c6fda2d74b6193a128429ee5f8e3 +344db13d1cac71db2cdb96a12a3fd08e88a7ff677065cf03646c14062db8c61e +0f47c8cb01428d3d1d8272363881e33a9b06c5caccaa6f9f99fa133b5a80bcd2 +4ad11bb9ea5d19a6f169720ee170dabd382f5363936df80fc37b1c80b77d017d +85d59a9ce722db2f1f307c8038b8ce5bdd22d0aaddbfb11816e49cdf32c03d27 +f3d442afecf9a7d5644111bf7e544a8e31d2a58f6558a85fb1145663f73c1a00 +3be36e39f1aacb33a2b29084ce99a53398193ae5ba2cd316f004eaf527f0c73f +f5625f8964ede138f4c0c8617062b96406bdc8db6101e4ef70a780a8c0440ca0 +c717f24a62c365d78e683c3613ee02e7303c64b63f5fe5b35cd9ff0049bce0b8 +6ed26ec48723fa93c261054c395595d6079f55b1137a5670ede801f14c37f4ca +3345def35b72521392a1273b815d5cb23e8262a1620c3915ab0e220bda0e7f06 +b24bf27ebecde1a71433710bcac9553b5ea9c1c002af96c2cdb832f1719605ab +11447e20b4e7ca4c7d4ef7b49ed412582414b35f4d457df735f6ce053067a2cb +54f4bc004eb3fde8620b8ad01c5f0ea0127635ee3a1d5a9d963cb9ba199333f1 +fadb002b14a448d34bbf9d5dfce51bd0aebb4c81d87edbc12bddf59d01a50d6f +c6185699a63b8892b8d7b544ba59675b05f42d633eb6fcfad34bb1de4b8d0033 +39f135b39f4207cdcd0f3f1dee591bbecf4ce2ee1541e10a14665a030068668e +1365b4f79a66619b8fb548df7b6d212a6b46ce295804b524075558838c582570 +12b1e8304da65a1d9dc6f6ef9c23015678c25188ccc22f86d44ff8858d0279ad +7527d19d8d8ad24648d2bc964647f978fff63a54cc08c60ad76d5c3bf0e700aa +83f0912fca4eb0293555f49746173d2c84e33d58b89974723926d19b63f72091 +6b823a95f305e1a0dc3c94b8e5a7316623b685810be033082caf73b46c98cee3 +acb17565d92f814183e1f60447403fcd136c2d9f6baf7227e2358194c89d1dc8 +ccecca8dc6083c5066b3618fc42d6fc27745c51ade1b82d332b2a6fceea44f1d +590c00626ac0f473740794be7264dff08dd93bcc895757ca8cfdb61fbfa2ae42 +48fde2fe6916d631a6d9c142cebf0556b3a7c308f2b07d2dacff0748090ab7ca +2eabcf91e7cef03aa214bec4fb2d1db5f305588de5de1ffa15f707ceff038bda +db2dafd6ba4e1fe84d1b47361c4671dae4c52902a276e30f16e6c6baddf57662 +08a507b4a8069c40c63d0256070520a12192df1154bcf73358c422286d2c93da +501f1dca1da3cd30ebdedc570901243597c4a41366bc4c4ce0bed7f9cff00c39 +bd2d87ae5061c82abe0a96ae5841ca8867a82bc325737fa0e024e1a7c71d44b3 +9582540c38fce4b277c47ce50be1cc3b53640c25ea14355033643f3766d60c66 +a7e90a9bb2d8b2859c7d71df202e09178cea8ceb6c07c750bfb570506b273399 +fff6d4a266c8d84f5b84baa172f79e1c4e45eaa49bef1b7b4416affa667eef98 +eb3f8d9ed92b6b80c750aa909ac172774c006ac0170f41e3afa2b58ade0e6eb6 +e0e325321e0d6f442c5f06d63cf22a71ac4f23664b67dd29599b9e4078b0bf14 +68bb0c532ce0a655023a8fd4dd061ff26daa53008ce716557c83ec76859a6f1f +ee28e8d75b74dfc41ba2bc798e0037f23ca71f3c19f5cb8e56ff12e17b33cdb9 +74c8060014daa6a8fe0ac7161cc5dc798537a80880e87feb91db18c6b737351f +a9fa3ab85d97ebcd2c84f49ecc659203ba90e8658fc82c7a3136be15d69503e3 +22e841f0fc5c09a31b2fe74e822a0d3912da4e26e2d85a4f962158f69bcdee5b +f2933541b288fe96a9b2f93873fe0890cbfd03df9f7a227d2d620d292d965b58 +f1831a16365de3675f4bc483e68efe71bc77b944bde5f1078102cdcedfe6a24c +8e5d2c73f16db3465998dd683d089d8e2aa810781e71d4973e84bc5eb6f0402d +6fcb105002890cbe38c3ffb6e2b911012f01fe5652b940952149605491b637d5 +52196545e745227070c4d76229c4082406f2c00817ef4ed4ea296054930f3cea +2152369bb55306c2662ef31e432878b4a7d9cef62ebabf310301e965da618370 +6e735436a7832b7db6b22c6693e727bfea45d55c50af9e310b4e791a1dd5707b +bc14e757792bab302541f79ea24e3ad24a70bcabe2e77d2af46d462e503ce98d +7af5d784ed2347849ed485aa544625cfc10da0bea977667e9c0dabed82130418 +bee1bfbd8abaad8530780e8c8fc6acc2b7265bf0159c61ad0f638c1f8405ab79 +189c856c7ae3d307934f59a1b064090b80e1f3eb81070e39aee4b6c80ecfe2c5 +a035830e48e29e5b963038abb81e6fc553f31715d4265312e14ef667bccafa1a +0aba11501987b8d65a438a57b63936f44db152fd9164ed841fce268189f65fb8 +312f8a15afecffa85fcce4cc111527be81c5f8945f0f28e70d6d938d931eb49f +354432c6a726ef3697f5df89bd938ec4902f6854e0aaaf5a7e030a1c1f8a9a96 +b3f02f8b18412a6cf4531863fda8db79aeb31af12a15988f689fbeb7367f646b +2e6a965bbec7097f3e25cccc37c40e1fef6679befea3ae06cf274d36a4195af1 +0f57515099a25b5632667161b03bf684bf0a8ad34db4cf33dd090f530eac2377 +300e34a71b7968bee2e4ba7b96ede2da6709830e15bac7af061c7f5a28d0d02d +e51ef88f3ac585d5a776f026977dc229c81e68b360f8cef21dcbec48844f90ec +59c212243f01db0ac7de0ff0c9f64fd842d05c32476a2a2a90f22822592ca6c0 +62de6453795c5938bc18cf6e5a19c547af1f69303a7b02bd747ea30b44319f6d +7545665d11e598abc210a6fd3cf4623f2fac368e6cd102fb49efe52615a6c834 +78916d8645652a97356502ebbec2c258ec461ce55eec41c78a9eb7a7da43db72 +3a495fdb1d66229ee0d2f7eacf9b889d116f6fe6bba4ffdaeee52fe8b845dffc +ea1a2c226edbd132f7ff4768fc4e43eb7f78719676e26dd57d187817dd53b853 +0112433201659d58a096a9db311fa9d23eef66ebf81f8584b9fa29fdb89359c2 +4e4bd49a3e505a86c07a46338a054c9a2b155c7680fd544628c793731a11bd99 +0d0966f323d36b2bde35932698eda6d2b6f236a205c1ab4becba45132062483b +d3f6882c3dec342bb97a67cf7dfab7aedaebcc908c28c67a20f3397b48e76d2b +13fecde99dd23f1c5ff5c8130135ef51d7583692079d7e94a757095a82db0ab3 +bab3cb6ca7624513c290056ca9d9dd768e1e24766c8274a9f16ed1d8305ff426 +7986de51d61c642c7fab7f6dd02b33e4b5afc2e7d4bba2ae31650232b2256bea +87a2c54dc30a58883b1527e035ca6d366331fa98fdbbb4837ea8f0cc79d741e8 +66e623f3dbd35672fe01ba210386c3f06f2b7ee5976567cb2a26206b14226bd6 +b7c49d82a0e862e2357d9f7c51334f120cb6fb62c71849fade595597f4696dbd +fd7b01d012f67f58e12e076369da78c2b63f34b25598b3a114e59768ddbbb3a2 +084f663a2e66dd6b9e06a90368555686ecd1026432cb863aae060a45ce09bf5c +24c04d16e5ec907fe3e2f8dadca1cb60da56c0be7ee7fe1a381bcaae45b7ddfc +0b73e5b6dac15f3f86904b7de03b5f187736679fcb8dad008ea8d99ef2f7baca +6c4c29a4d9b71a3d86b93a449d92ae04f37cc4f68edefe704e5c99742439df8f +710efdd2e808747139c0a3324334d05f9e24dbfd01edf79f5db78b4a99f2c42b +850a93d11a35c42c331ea1118385dcecc91cc7a14dbdeb362d2f0f7f86d362a0 +0f95236f5fe00f99cc298898414505efe544c524e5fe5fbd8b407d3330abea64 +1258be5437a50d58789e1d0c61750adc6da099930dd70c126fb6beb28d3eb6a2 +6f56f6feed657cb9dd7fc7a59ddca27c4aab79dd869f46f266e892072454ebfb +74b00a8c7ee0342ee58dfa96fb877a04ac3edb50a2efa679940dd9a99dd76ee0 +bca987d06080b6551ca1ab9140efd153a06e16904d8974d660e6e9274f3b31c8 +cd98ca77faef729789d882a361305cac0757b119d3bf3138c39a426264fb8aa9 +9bae1d40b1f917a027934f1772f8d28d72d400332bfc9ca196532b8cddf0497c +8d8ce538b3392d4c81401966c2482a0d81c0c5e5399e5e9c2f976a865410051d +834b926aae5bd04d788507b3272efe068f608ec4752c58d5c3e15e814fdf5a91 +865d313772553067ddfe3d7e8b46f6304b519d30b8f7c81ddcb39a8e324a7537 +e372a37847ad892cd11231769275ce755f332ba342551a99743a7593494e5159 +5c458a84e32063db9a7ba4557fc358b25d59a2b06e02ad37b770b05d08fda2d9 +d6b1cb49ab22800f4a865e89dbcb5ec8c6c752e047e79ad0daf03648a5acb6f1 +810101a23bc648c46ff97c04d6a4a4eee229d73840688162978c5eea4fc77ee6 +a47628794691fbe9ad4d4adc4a16137945f46a8ffea83bb2610d7f7ec8d02ad5 +7e654703002c0dd479e350a9025ba807f74d8eb8e73ad0f31386f378ddb0508d +a4fada66644d57fd902ba5f9058afc10a79e87a2b3b494355dc178a57f9cf55b +099dafdb96d3ebe65b795b35ed22d67a922901b6fa2b8337c92289ea7e4699ae +6af574d4a3f3a76f0480e4351c2ef40cf6d3803e149b29007138697f1f52701d +e99a3eceb12cad45beb08064eda2587f35d9283e04f1c7167cb5a660bbcd40e2 +312e9fe47f42db52dc67e0c25ca13104a8073630dd1bb586c02b175634cdb469 +7732a0e3ca2c07ad4c0a823811ad5fe50d23e258bc876d905484809ce11af5da +6e23730cc2bf817dea1f097a519c6c1605f51d7ccd254ce7f1182f9c2a0155ba +094c91997fed511b9b2251dd5c4ffdbe4ff2ae8feb28246839b5dd50ecb72e2a +b69c622628dc20c41d7f4bb9e1cebdedd2fec401638ede17fd890dc4cdc2bb06 +9fa4ea0abb521e74bdf866b58a5e95f1847414339eaa93dba4b5848e716d0516 +65c2015cb1ec8679f9d6065f3e45633a56ff8c7606915b4544d1160209db0c5a +7917ab0dc977e14098f07bdc236ae5ca5a3d91ca6b18a58fa203794b0dcc9b14 +c97af89a9475d93a2aeebe1392d14f1fd44352c3ad3e391d0a8829a084bdd5c3 +06e1811a3da521ddd5d666766395e91b68c00227d8969c4b411413ffba0139d9 +b53121907425ef5ba688a54f1cf2a36598bd070fdada6d9e96afbfc417f896af +69481cf47cf64685ccc68100279fdc8c99281cd262db350d41dae9285a500aeb +5b4a6f3ec40c0e6721cf2848c6ebf1b12f862eaeeac74b2677567c5e32754996 +ec4b87ca63cec112765a1fe9236581ac5567c16be9e6c0cedcdb08283a8033ac +361ee24a89831b766198924ed196fd453142cc3cf08f15e9053652a29499d8e9 +28bf4ba94b782e2791817d19d62f5a31a5d667f40f6eca9717c17597600cb11f +950dc7a272cdc559569325c25bde0f9407c18aa2c4da2c56f2d9cb68fd1cf1c1 +ee5ee9952e61530a3932fee71d30e8cd39b8854708b042490507b7e1f8372ebf +ca5d35940745d1e532690f41f5d13ca83359f51c505e657d212522187b93c16c +ced917cfdeb5dd8944a5f36005991dc924d6ac3381a4b3d9c189d7b7476f1f8c +788c72af07fe924d9a123ffccf649829baca970f0422258be1f631ec9dbaea10 +e6d2a99f5a493ee6ffff83763628685bfe5a379d7887076d1885ba2c4f63c35c +dbae38b074978923eb63cb4f1cb234c5b4d32562d4ddf1c6b54d74609d51a9c0 +3629be2204f5f0b6acd3f65a2afe85d4d2d19d9b9d7874ab8300a441da628667 +e34cc4735c87aab7078ea8d6d3c32ec9cc6cdbbd6a0856d22e58b0418e095e63 +9f417a32dbc3cb2e077053f85d7dc0aff7d706f99ffa1a9354ebab33cdd7e94d +6eb72d5d1558dd1c754f718441ab50e79f3d87e2728a45bd1856e20662c8999c +86003177b4516596a8532d2e420d77dbe9996a139a5234bfb302b046a5771ec0 +d191231663b36f07c88f4e17a401972629a3c9166f23e76242ca607fb130a322 +33324d7e00beb6d9daf773452a55205f9625adddfd34b0ac0cace69bf35a77f9 +e389596f0337c8e0f011d0befcedaa787a012ea63cb0b204a9b5012188e237e7 +32fb14a5a33e69c9743dcdb7758fe961d22b1949acc22584253a4b8c3f19e99d +ed3f6a1342dcf2655aa958ebe375aa2e5b3fff62606417097b0e2fcbfd069ef9 +d0e548351f498aecba1225d9bb2a141c7c4626ddd7af419c1e9d654d36a786f8 +deecb1e16d170c891e208d3546784901ce2668f827364e72f123edc4572c5238 +bf1bf8ec7e8526bdc9d7b349f74d5d4eb4ca2b75ee96c456d84dc108fb76aacd +68f51ffec69db08d1641d1a92194b82ac23017940dc29ba52b7889127a060f42 +0ff60ba02bd223b8c76d26292d41de2f291ac96093929e4294aa32ebd924cae4 +09be352a5dd60b3612413e04bde1efabadb801ed660ca0d48be24590b70946a4 +fb6a93a024fee3c14806ea03c7dc494e53c49806ad9a2d27ed008aa63dfade7c +2fec51ea954112b77af5e584ddf478bfc1520fd2205b85f68bfcd4b6c909a791 +d5b93229f11a527bf2528a0554b95ec5dc21fd451380332b17385543cd3c7a83 +2cf261b770e40d61165f56df0f6865ec5249e2113b95d91aaf3f5ec5db468055 +b3598df24340bca2767fb5cc23949a174461015e7ff7dfd0fb32c8eaa0d21aa7 +94bb4e5edd99f628cb617b5ecf4cd5e838074c56de3c2ffea4e90d7a07410853 +76db03fec95073190cd0faa9e5fb862e50d478cbc2581c4ea5de20f82abcf8ed +7e65af60e9c8babaf0dded1dae5d84bc6461c23b2b61a59cd2b9984f880b0abc +5110ca4464b82e4b1052b926328945326f4e466c1663a8fcac45dad23cb451dc +5feddc06542c849b33217e91abd13e8ebc208b35fbaef85bfbfe5cd88fbad54c +51b8347f38e38a12616700301b96c4802945f90f576497019ee885a7bb82a89a +ffbe8a79a8a9de1243f6c022358b2e84e051185636b79631c5e7a03ed8abb8ac +67b831d200d7a0ffb9f2ac6d0dd16a064e0b8c0b75b1d0cf0ac120542c6734b6 +798d53de8c3c234a412d140fd3737cf2f40cfc02cc9f97a8b0aa611df717f9c4 +f3091d0577b5201deaf460a40d636720308f7e74a2d8eeeb2be2fa5f97cfbb73 +99c93d938649e1dd5d0e6af35ded02d574b60338f3f21aa031b57e3bcd6f9b24 +8936c41426939fe5f07e1b5ea80ed2512cbee0c0c0818c443a5f46e6bd59aac3 +c6ac2906b977ff57ca817dfe39bfdfd1271a2540e2fef5d00e678f5ba7e426b3 +bb53f10dc3630537b9b746f2eb1d81e6f5c99e59081daec894fec9a2dcfcf06b +8170148e5d12bbd05811c45d42f65d85acf667e40c4c0f8ef4be09eb9031396f +7ac45f71b22b6a2dfe7410933b6dd3dd9a81aa120f2036bd80bc0eb476e8b1d6 +4ff31616cea77b08aafe30723b2c26a998287c132ae2ab6e385051642891e375 +82bc13847269f69b43bf20f546160f90fa3002f55b62923f8e2c1366a231ac8f +de8d8f130b29502bec3394c5a97faa613ea136e6869b8b0ea3e1b29bd1a47a78 +74068dee21ec501fea4efb2a6fc041a1ecead0de85e7068d5c582c2a2d7411c3 +995cf6864a28cc3a9d82c7c7ad4f1609148f0266f15eb697b559469c56511e9a +07b71bbdbaa89574a815f5ad6f1d696ff93f6a80f3ab54a5df7ec79ae52a1a4e +280428476ef9fa360b6e9334fc1085371a6991a8dea25d21c89bf52303d4ed27 +6201e48a9feed9a81a8d7d444a05dacf5d9b5486d3ff7fa9691a1432b85a06b1 +13a82a31476ab851296b0f836e541608b41f9de1251393b9d38bbd198b52f819 +709156584c9b6aa7a0190f1871079e5a9fb38dcb31964bc6daedba75eb7e24a2 +73453776866b747b32dea027e09724b278aa1d7297b49296f7ff6965a65b7342 +b43f068822e680db306d3251dc1f28b60b45a6286f5309b2ce02f71726139656 +ba97ba9d8a277042f04755cfd8d92c08f2ab7675c2da2b59cd1f8fcbcd5a69ce +92154cb01b0745d9aea6eb9e419824b1fca60886f9684a2c1dd948751c1cc687 +4e3f7578572f4b54fae02b769e63357c6797453f9acad2636ecedbd309aeae25 +b92567f94794880acdda0c59591716d223a39319e10259972d8e0a2640b9710c +d7c7fa415d51ce81c1259cd7f6582fa89a2c16619c92ba6797e70ce07077b877 +bae5e289f1709552a03e09283a9c4e44e400c52de6923491b4983e3a97eb7087 +83008d633b3f9725000f57cbfb5a1735b40dc93828f13b7926bfd9803d0a9af6 +12c79c875a912d44341f779fc73831f207f11a0876d870e8ed35c80ae9fc66ec +5bcad6dc58b6a2568ef82fdfef18f5cc8e5f189cdf43423ddb4234745de64194 +599de46d79b5c57a62676fefd1c08d80e8cf29663623c2556ab0981d302cd6ff +d94debb273dcbbfb93d9424a553508254538038983c7d4082da35c6205913494 +4bf4b1814d5bf55ac3f0a36d2acd116631ace9241267da7101f07ddb8504ffd4 +672f77faf0fe8727e4c26a6ea93826640ae1cd3d806be07f810e6ad4e0407fc1 +71c7bcafa661757a838e608d87584fe02618f8ecf9f55bf4993fd0b33ce6d635 +e8767a2d0201007cc3f195d20d509de95b9aa6a1bcc8ca0240ea2f7618af4f88 +89b5005a00d77d1608540f2d27f6b140e0ef33a01ae9c1d1b6aea027bd00b16d +b5e88130b170d91a980ee8ea1a2952caeea152b318d38967a1bee8735ac94cd9 +a02e2e9dd842efd5a5c16d4d8461dbd31455237ed09d0675d69dbafe0ca46c0b +6a1778bc880fee5e38b9ffd5dbe2c23d60f516855b97ebafec02a1185a06f9f9 +efaf97118f119fd1c18c1062088a1f988cb682e5cc6d74b15d3a03d10d881854 +f824e7342eabb60dea659908b2260d74af16e1dcb0cba75242578eb1e1f008ba +057bfb3b104bacbe19fb9467e3eb7c2a3947a7c9b2467d758f2cdd82947180a8 +04d660433f8c890a814e683100f3cc8e5805d0925f9a9e3c42805a03e6b4a36f +a69d401c73704f7e9369158673cfc0a5b3c88e18497299136a06597281410747 +fc8237f6d970b1366af7a0ebcccf86613a142bf91f3b81e7d3308f1b9535b6da +bd2671f710ae5a43c6ddf9bfb504c685cf5181358079180a2fc682db3e9675df +8a688ed2b74cda40548d4fe3f3c1740fb017f83bbcdaf985478eaab1365da4f8 +57f115bb1460d42ed38bb9cc4e167fa41895256c739c1e9d57b26fd32bb8329e +c5f8bc0f398b69700b270a4f8cdc53199f84dafe40f3a09b72e37848aadc2aa1 +4ab33c9ffa7bc55d05965fa9b3127920f9ab5c731dc8bf255e41504ec4f24293 +9c39060aa8c4ae392b2a128aac74b19ae4fc648b0260af15c1cab6a5dcb22496 +4651b78a941d1428530d8c198d2e689a7a3b630019e75dea624cbcb547a8c1aa +e91332f27f14db1c488f70d7dceea5a3168bdf176c2f209a1a96cb7a8fcc742e +3936b1dda4b305615b0d849e119eb6e4fd6fd3594d27230fc5f31d689ed5b0b6 +61b7e3389045d323b3c9b03bfff11f90e17e4a642ca3b44c889d91ee264df9ad +347f1d83975d0d47f14d8d01b0d1d7164b956802838d9bef77a723795a629f22 +15bd9c8055cae840f08ce601d2d3b460ee7af95cd52b1f4f671020fd3192edd8 +dad41366b8fb11235ce8a1b6a0837d5758c08e0cf4c710ddd5bca41546c2d577 +67c4904bf3fceb9d4aa6f233071e95abb4790b4861536123ba8f0b9360a8e1f2 +9b65d251d0abee150692c91cb261d7517bcf26fe8395c275263fca21641c43a8 +e5a8658b0d25cba6b4b0e02149531c400880f8e753f9c141de54358fd9c5d860 +4fc857383643eb2573dae3c1d7549d86aa7ee7ca9ae22feca7587f542d34a6b3 +14149db3e7b8d54265b09bc84cc6220d91cef8644acb4d9226e62d4e95648264 +a85a28b5fd50e7fa846a74e37f29b820ed74f4fe2dfcd08b506bb54f4b64dae8 +ce16c86afae21d8fda9e0b64ef7534529b86f5565e45d569068066d2a9c7ead1 +201fe9ffc664c12b5b8d4d08d241aef5458a20da52bed83b8baff4565606ba9e +5e3d459986161589455cf0b7e112212fa173fbf20f65bf1f4588338e2b05620c +cd23db06e4fdbd08caaea9e27755c596c84efdc4fb34ca96cb44f8b8e3344aac +881f33f0b38947e6f66bf1b7ceb3bf55e7d4586c9ce107134bdc464dffc72d4d +ba4c3021b23a950e61460b9d2fcb8c9fe2b934ac4cf0cd2af9f44dcddc728fc0 +6a008faab6f4ab963ab6871750e94cca4577fcd5b72c3206075192a0490f0fca +37d6718c76fcb03e05aefde7182ed4378368aad4dd06107066d83fad0fce5f85 +0697e06274dd59e1a5464b4062a3687f7d3da144a0ff165956c449879809a832 +3d9769a9b9f663f791c53eeb883808f27d8313405bdde5b7c0ede1797110e0a8 +4b315e9ad6622d7bb257f890f663f440709a11fcad5ba2b1d480e8bdb3b89fd6 +b51ba5049f7d61cc53d15d59313c9ce0b6946ea47d58758fe044798ffe59aca5 +dfcc686e007da302916416f5a5b9a7b6d15cfc8f28c23e8960930afcd249a739 +efae6869c74dfa9c6410f7a9294f7cca8af33edbc6e3536f95deb43e2b26efb6 +85079a3049884bc1ea3c0a1e28beb44dfce94823a60cf1f91544244a68b7f0ca +3924149d3a71dd370a151d1a110defef096de93be043cf4aa09b8b765665efb9 +56a1eec4146d50d985ca1f6d00c89892778dee916fb8ed206c09411090baf039 +388565b9f8e5780db9c571f6b55593dd84b9a16fdb1b74f8b32d9a613ecef342 +8313c1e235440b1f0566e1ffd62ddaeac071c3c38668502df5f94580fb0274f5 +c8a6c5ca8fb8661dbf24e21b5a9fdd2bcb38ab008a9af4105e679c47ca0d9ab8 +30c401c42baf178a2c88b4da2255c0b7c5a0baa8f42dabc7d5b545d0498c2126 +30e4a0c0e43b6dbd67a3055e126ea11b491e0a6cd126e31dd9832c8f94d6fc14 +f98462fca92caeb56a67097c835470ba1e790266d9d3c38a43f94c2130bd6786 +850908f55bd7b512f20895253f09d2edbe775a8594b9d9649e714425ccec45d2 +9dbafcbfc78f1f71993186add1d5bceeacac926ce70a133f3c52607abcc3ee97 +df2a2982263273c225ee5ad2ca8f788293509bc8f1d52594d7d811959152d627 +947d27c95f36dd28310e051499b86e022ba1c7640dcacadc0b23b722fdc36b95 +f5b817445c2df1d76d7f57729ddfd3002c4cbb2c9310d580f0646297fbd40a23 +3cba96b655ddc470cea3c92564889027d32e71e8552c9d0a38d7d1221676d484 +5d6628d684f9038cfacd56ffff298c81b31348b1615377c36052cf2d6db3f377 +142ef084bce05900e3573b85abd7d51e79e516481f675f0769a1eb325d824159 +f3669de7875dec76e286e7d6cfc90751e7e88a9faca280c0ac94d73e71a80647 +9d3f41ee6e5ce577cd11f891ab855135d71c9b924abfcbac52037b7b5d018aff +746f92b625d7fc474c0925da4db2162dcd435a749adbe5d3d67e9750733e0254 +fffccc385a9b1a669136dec710e826d99da70d6e6a2aaebc6fce2202abd0be21 +69e9dd45dc1b86f0c790e2af06e6474bb415b4b747dc7982093ff5c5b2bc3d61 +bdc2b96de2957ba3b848c881aa026ff1f959230e1f2a352939fe9ecb8ef712fd +9c4b6a8295f1b5fc177c31ab1a9ef73e6c606425a452d972d1e2694d8c03447b +985ca0ea62cc60c2e075662589c1c703444c6611fa8940c7e7df7de25f403ea3 +1072e4694a0b28a1f684669fe308d6914954a736616fa7526b72b12885f95a51 +d4ce7496aa10faf9190362da54a9649c5703b9ba17d71e234164fab84a287e93 +e00d7629fd8cb0fa5a80048300d94cab55746a9c21965f19a595d11b947f6b5d +24712f29874d05505fe4c74320dd51cce66def98b76fb1963b26afc1945b4c82 +1442b1e915d2b58716dbb0da4c1c8ef7b0eb71166046d5cbff1031a01e21efb9 +d683f4a8105bdbc816b5afb478845ebcddb8af1a3b24fdebbb1ec7b3e3fe0ddd +f391707cd506589a73d10a6dd2dc1d9e30208ec4ece130a91181a4927c57d051 +f8edca70a6c8f62d463de6baca59072b797b0ce626649c99acdf3d6e38545903 +01ee8800c65f6014a1e2dac189f34275cf25179f19eeefeaad35c12dc31539d7 +2e820ffbb7e6c0cefd39b5f1b4b3dfd099acf5449be3aea4097245f2b08b1252 +12806781c7b8efa84ca538d752183cc2bd0852bf0bfb7f5f1f646ef1a18e9cce +8acc603ac9514f7855327761c04d49f5ec931178a3e207e3627aad366cc9f21a +fff5fcaf7a52047be6ee64077b44aaec937275306033b09ebe03a8a6e85817bf +07743db28742237f90b9cddd4f5611ba1d1f53d71d56a6268f563080185919ff +3d41cf85b32a6bce7f766fffe54d399be503d7d4310e9e39aaa32d31a7831817 +b9ba348846b31c40da668f48351ab4e11712468d3213958df83083f9da85dab9 +46eec803f5e0d0412a0ee1ec4d9910bdbc66e55a420a1f2b7269059c8ea6d051 +f25ccee0fa4a91d275179ef2761f0a7e79b0b561b0a8020d890ccd8e85e7e57d +51296e74b398a4a00fef0397690ec0df4a03b8ecb9e4e315df71e4280094556c +29cec9f021239f053eb04668e7a46c7908580f00e1a21a055590216531bcade8 +580e0bcc88214146b362948cd2154b9100d2a895c1741106165cefe487199054 +709948ccee295a472105e289dc342bb69862941bc983d78918a8a426fda65d14 +859b3b38d452ef89a659cf90b2ab373426e5784b82268b1361761917f28d625d +5fd8db2754cc035eca5273f259a7392cf7208dd577b607c52afd943990dd34eb +491148bbdd57a7fa3317b0f643bd22cdedd983e2b5a45ad72580ace15b30ce81 +aef36205585544f7f93ff7822b3a95628191f9823b0cbb68cf96157bfe9d5918 +cac2d97e43ffa089e71994bb8565690c348a7769d0cc2752f9d4822e49a90bfe +bf895a54b0c2c31d18435ae46b70ee47f1ecd6e2f51f3ee59cd6eef092495488 +a5a584c1bf3bc1ada56598d97506e82f38261440874af90dc6c21712bde9ba87 +eeff905bc1556d393129ca90c28416c18925229b7d8c6087e44ff78a9ae9a164 +f80da0fd9aee3bfcc6ba2da8be50a709d11c390d304650c37ed5216209694149 +5f2842a66f2641c128340337063d0f34d5ed4e6bde2a9d268e54fd4cb7317492 +36c6d7c5d4cf186d97911d13aeb06a37ae2ffcd2f99a4161fcd494150ce841cc +0dbd7be61768e4e716cbd33e40c4249b381d563609076305d01b84418055dc0c +67ac48d7da0db58b0af12005631b1c9072a74bf8d3e40e75189e0a5bd53c7a0e +48b57178b5b829dfb580ecc26574a05e866f6242b86de0e0e83899b01c710116 +ccd7761a162c39d1de08a37039ecfce8336d33e9a4590b348ca0b9f53c2633cd +d5c92eb2a4ea105a1f4331eb49899c00d37bbd498a0097db32ef077c9356a4ca +7b874fe174a874fe777d48cd30ba10edf3675858644fe7e888f54bd5ab4fe80e +7b410ab6bef4f847ed227ad0470742b425978678a22afdf3f57b2306713d64ae +ed9f5648ad024995113b031b7e7e28f14bb8a8b158ab623e1bfa47369ab6549c +7039f15f6a34f84a97d74fbccf7df103aa4e5df20919c5cbd9239eaa20e7a30c +faceb9ac544738ce1d3d7869e15ff79b4dc642bc9691ee785c0c629988a44d6a +9f486ebbd5f48a4f6240bda2b0df8aebe6735388f11f248c0b5dbb463f0b28f4 +d98a7db3ed34c2fa7b0d72a17da9da49e4623e7928769ed659fd0cf68822fa13 +4de9d497ec70162d2e494b69b079b3ff3285623c8f708f0f29779d6379b4fa82 +432ab9b414968295f7357b75c9d4c124e0b86b547233dfea5c86a51ff59468c1 +3fe980438d308ddbf6f6370d8b2de9f9c26a17d0ea9b17108ea6aa57fa852c0e +8004f2962ed7b8b436bfb922ecd0de64ee6c093cf26a518a9f97b4e248e88a6e +e23c724e9dc9eb215bb603f5ec69f33a7a863d01e0c33cabfd52be253c826a75 +ac8c10dd4462585c7a4dd9fa94841349a0c3ab8e473df5a4c2d42dcb0c1d86f1 +51b8bf059dfca4ef0288f050c9536b16f646b84a62008ea6d3668a194574766a +d4f0e7ab0e8643b608fe22c68747cdcdede17e1c87753db6df06227377b6aab5 +3dd09627ff359ffd7d81cdee83cc2f91746b3b9eb2ee6f08ebf795b788dd5551 +bd3cc90f2f7c72b6a092ea1848828ca9461ebf249df3cc71d0b3aae1020420e5 +34b61da7d49a8dcb63806f824af5f9e71411084e9f3d82c2b23f670c353aa6f1 +1ed1e232ed6faf18f7b9111005949cc3be914e0e77d72eef1bb58a439c208da8 +251434fc78bf0dc080c3b1868fc26e7863ed49c9da982a9fb3768819e7451c0f +8d74cdf783d64ec2801a5c9018a3687bd644ac42d5f3bc77224a47b90dd359a9 +15bb8444baf346a30a87af9f53b26a3ca4be96a70a168b387ec7ba3f839c1c78 +f9bc168e7952e99aae7e9f519216a3c749a3120b373a9cd6a648cac0182540f1 +a2f304e5655bb15ebcb38ff17d815fab5ef607b9a6dc680f58ff92f47249032d +afe05592e2aad56bd8e04eae47195187fa54defa5c3d7264cda0733f64e4c3b5 +4e21f32c2d60be8f71c8167760b3c96ab9b83e4bef4a2b9bf23f2439f4c030f8 +f05c213f692e0b291c10a6adac7ea9baf4a1ffbd9442599535fff09cf278cf66 +9a5b9d06ef2abf01182ec070a002f89da53db6e62d09d1182a01d3a45af0dadc +a39678926e85713d5ab0202fe8056d5db3265fccfa27247315bbacb3c74c3d89 +1f908190610cf904552c33c5e4c384cb5adf9994e240973824c7316fd1b12528 +e611412545b9f7a1ef0feb6b4d2a0824f51e654445cba2247f0e8d7b08118cc2 +2b4ce090fdade2c9cde0122f338a527db86dad3b7cd35f25ccd08004348ad08d +957e3dd7be77c9679a5508cff9efbd4034aa1a6d39d1ed5564e3138bae30bd90 +36cbf593ac8db5a927448625b29bd3f415a5583b323811142e1ebb8e7cd28d4c +f791cf14638b9f654f96bbafe52ceea831e2556bea9de2033e5e67aedb4c2c0b +bd0b9a6b77c80a21f6b2862bde084972eac85886328881eeeeb868f83a1fffd6 +14101f3ad8e341ee6f4dc1698c7017d4442e477bc7f6ec0208de5989689afc8a +0910ce5a8ec25f0ea0b68854b0a8d3e8eb0c62eb07b524cbc573eaa9c515978a +88588f6378a395181fccbf09117c6bf7ff237d6647406fb17d9c37d9a25149d8 +0635c503df05d4c7792522000bd03f24eb9d861ce394f3226b9f060c7d020ceb +78213d2c3b55606c37144b41238f9c542f2570fa701d1645ba58d38b2e838fec +3c38affe90916235f0bcc55f352fcaba80ac4c04dc8e2168515f366be1afb2b2 +481d8c328debe15f933c5d933f95d4402bd443ce841cbcf1655ead76d0c1672b +ec7c3f7409cd0151c00c44d756ab583fe5fed84f8814db603cd3fda6b3bf3cba +9d42898a238cf252ce4d2895d413d8d3c96101e15aa905a9efb7d8cbdbdf5134 +9b1d1eaf04a618ecf910da6b91c861319166b9aafc3058ffd24cd77c14f47e47 +0f9d246f968425d82d181e49d8e450be32c5bb712cdddd6d84e919b6f3c20cbc +b0c051e09b83e3b3c3d7dccbcc23b57fd1a98f6714265d5dec0a5dc74af8a8ac +e0e75863a67a6621376fa95bf7518d2df07c03f85226ec80c9d2e751c5af9dcf +0d2f1793b999109f5997e0a3163090c1369cbd67d9f7054084d6a5c33cd690f3 +74b87c61314c7b135af9cabf5144088ef35aba29dc97478cd2837d99d2d1c661 +15fe5fb8fe24ca483dfaa3cc86a477c14fb43f6dfdb8a2f562f2d467038d88e0 +3108cc6b027e6237946bc42e74970b46a097c9e6f3d4d0d6e9e0c62126e27b6b +516e334f04398b626d016e0c56ee6c0c63d77cd0face69426dadbe9d4be62c54 +9e6b43fd39f9f0c3144997005d7ddcdb3bbd64af92bd78c6cf47eb1b088705d8 +3d80f3ef189bc602d14e512c6167ab9b6ecfe474488d29486a58b129666dce3b +c570983c576a8b69fb5da570e314f6fca38a4bafa24920890827023a37f3fa54 +f9e253c799476886deb040a4fb0d699e4d9f5931ae14e40232219f8fe55693cd +a0848915cb9eef4a07d82159e4ef0540634e60a2d1c2bd8483abe53a489d8869 +dcc34853771f1edc6c43fdf08fc89266735f9dfb19ec871eeae7367778ba2cae +190fc6168a57c4771c15a7d272903a80e395cd0c87b288e748e52eac36b6e801 +fe4c776e320b074e04ee47d31c9bc5b86389700fb48766f37f10bd4f667df076 +b28792a3d802c6e4c5ab62a0abe4e5127e7223ca18342d4271c7169dc92f7678 +bb06f10a5ec821e9e1e37e7dc1e0dd202874d654533130e1af010717cb4ba389 +5b71a31a8221ac35b163d2bb9ef77a66573c92549afcc0cec4e03e1234981607 +4e4b14e0d955bfa053ce4a7c3245ab1253406b480f70247d7a5ba92636d13926 +422d2381214bacf030261fde77f51d0b5b747ceb8d55794e7a2d438521449c22 +c365c722abaa90b1a2dd9cf4038cb0b497159d84a4fbea01cb4927dd4a5778bf +17373fef0a0014bbf38918157428a4bf11736288c6e7841b35664cefbddfc5d4 +880e5231be0dfa9722db86a78f55c93fb220f9f7244301b03b3d2a2659e425a5 +b9b515b6171b47940809366f5d58591a56063db03fc39f678a03cb2b455f9428 +2658579ef19fc30c16a573883ac4b0d97b943fe7f4c8d3437fb94c4aeff6240f +ec5c4a6b03d10c95aa67a4fb95a913ddedca98b373525af4920e2c6d171957f8 +42ad6fba1ce3228dbcfbdf82322b4f2a600fa5a294b5c8226d8c201736d8ee78 +4fb635d71dbef87e1e9b7208c98bbfaceac8865135a6e0be3d584ff581170534 +57f2c10948eb0b47aa36ebbd374b4a557c7162e4cd12ef3eacbb11d1f8655814 +d128466ffc4c05b162bb51ba979b747b90c98f9704ebb2832ee773c3ac195e46 +22eaa05b8c51743b4e6569b976e10bc293b987e5c6c18c26b65c0a9ad64bbb85 +ae093579f0656707564e77bf66fc8362ba2bd9139120dc55e686e6ec7c5539bd +bb0c45fa302281328b2e3a3c639f127e404f424f05540bdb16104287cc6d201e +fe0d8ed75c2a91d8b32327c190b73809f9e9f4ea41f227a2527205b1b85a12b7 +4b1b4052c3cf9110e0530f47c0567f072aca8a457abd767de06408867b3ad9a0 +101633e4083da4249bc01bdfce62c02034afb5b6c8a4bde9077a66efa61aa173 +e41855740cf3aa23299d8693f1117f7fa9fb469b482885f787d008e5469f0338 +a21d6161f3a480950627a73a4a616821f1b5242bf254a80e248c819f8710e967 +6da8b39dd3ddccf7b72b010006db7832dd7f5e15c7bd3168333463fa617407a7 +3f3f53269ca1ca4f8d4f6b68d88475467fa679d8892d951ac35613b69e738aa8 +c179d7559de70342e5d495d3f98ea803a1d5ecd7e965c0546b694a0960f676b9 +590931dbe64e9d29c99643015608a68c27fae5a69ea2be4bd26fdec2733a63e0 +d5794254038013648a6bb807b81d1991cf33fbb49a05bdb9a92a66b54ca088d8 +dc4cbd599bdca611ddc90a4ec272b3741ffac658c5afd409cc4585ef211f22c5 +9aadf6cd0aa79eb81ce45a9a444341bb2d7e9896f3525da2f4918ebae3a453a6 +9153a496c8c2b2a763b309fdd7421936006521f7f30f1dc293c84f3a6d432dc4 +42f9674f74d582eee387396d1b9e5ab7d82de80e30b2258a79eab8343129b81b +55870eb28ab7890c01af34865100a9dea848e86321f519fbb65e28fe677299b4 +a5df47167a1f2328533338aec3e1ce461e62a54d285ff9af7cd93e286d0d7bb6 +995c63406978ab13032fd8122e894e096329a0afb9e676a5f065fed3c2337ee2 +0eecc0e8090dc51caf6527fc05b576f94639d172f047d4cde2d56872ce967637 +651024c2f5b403fd90d78d9d39a5706738d775bfbfdcd3e87b8fe8c45076c8f5 +492a62304b27b891995b2cb4dccd8fb87e678e9e72c0bf638349e5f0af7437a5 +de43c9bb210eaaa271e6ab2082beb64b911aa3548171ba8a9ca18b6f4a6776f3 +f24a966f093207d23fa00d96d5bf534059b3a6bfabb411b4c3848d3e9f99f97b +8d48ec0c5083ab0f5428bf26360e7221fe02ae2fb371c18902d27241b072522e +92018026d521ced65ea9bf282b6d1dbee8f72ab32cb8b12b7fa515bda4be60e8 +e45dab08f0604e49556872a99f5cbcdf8193110102dee8a0a257be9c08c226d4 +db294a844a56a26cc99e66bc257c08d9d5067ed6370d9b42c2939a05e4d55c9f +d702de002954b3ceed1ff3fd628118d74a42cf0bc887413866ac2403b642aaf4 +59cda35f306894e1b6e9a4b427ef92d6bfd624fcd19e95c6740fb7cc5ce60a72 +769b64eb3be285fe2cf9fd2fe9113dead0408eab705fd28488151840615221ea +1c8e1f67804dbadd9f2750f2e558714bad0121fead00636150310040fe21306c +36e4a20442fe3193db200877dd46d763ce79e3855915a56d631bf12136aebf10 +dd909ca02273d411a7eee842a79a0c9beda9f979d3a813ce4ae5a0484207545f +f652a8c9af4391f7b6eb4216107db16cdffb9a60ad55f9ed432930b3db265461 +b97b13203de7d12249db2b0620df7cb2427e6285ae52fdfb28aa5c2ad8b77360 +0df4c61bd5dbbae778a595f568b9e5c62be14de45c40294df5795c31d43f9221 +e0fc996ba9cdeb4b29655c0c85546e17c14411a15961f5fbb3f989f7f5f8aa52 +aafa02310fcdb44433418ed35683a5a142e2dd00946d16a82dcf827d56730975 +baf45c152415f62e545f316c9048cae5e19d8b6da792633fd82b55e49262f892 +09a17e84ea6c120ee4bea8a25223630f04363561517ded52656cdae5feaf0762 +0cfd610197f872fcd0a7e8bdd65a26be95cff63303cf709015d0d5b90f802af9 +929861f3e45bee090aee9980e5dfe277568db1b2d9db4efc222e02386efffc88 +90d6ae0a84ad2090229f539c045edd6892048878d2801d51af0ee066772d8503 +836b141e7f726d7bce6a5eb0c24600d4146f9bf06335974c64f22f4336f87492 +2b7365c4a9627a49f73d7c76dbf597cd73119455865633ac903a2cb9a163ce61 +24642b82df86907c73b133b78448f501fb5b9408e3370bdcc91ff2c1ccb5e7cd +713402a6359166b60f3be43795f9311852b6d94ed4230f96f44e690e13794eff +a700974154138071451286d230df62539b20b41a00c913cb86777ed412d2b28b +2dc24214f7bd00bf8949ca36dac389fdc6f44ac4e2ab2cb6a2b6974efaebde80 +0c8fe350a0473f8dc81146ce3863302f420dc463986915f7f9ff7bf802c2ed66 +9c3b58309b3cfd129f6f46a803732446716002f5c51b5784d10a9b7bfed4888c +7ca89786bad85851d5c7f07c78af2a55d82d3d012630132ccbc48dac54821c93 +34ec8111bc0ef280f95b1eb3fb27d5cec116efdf5f341f5c5f882e35bdb1be2a +2152cc53b274dcbe7b114aee9a6062c810ea20fb5ac75d8d141937f5f071e4d7 +5b29d088c5ab34723733e608fad80e0e6c7ca95373617570380f8834bb5b7411 +9356d379aaad972d8ccd01b42526052de6992979bf7cbfff7864aa33f86c42e1 +5a4e2ae70b108a9881be2f89b28980e48d4e81614af16be4b6a749511e0d4acc +f1ed307454b5e56cf88555e5f8db1f3b179d772c12913cc92b8cd13cbf5e0aa6 +8ecf7c0bba830dfd339172926ab4ae8fa3d2bc054286b6838bb41e1a519ba5f5 +4453efd52ecf49e8021779e34de0b95d02270538568c15ac35afc836a2a5183d +eff5a7de30fd63106979278777865c9b3a07aa7699ece648e5609e707df2dc81 +8dc13ba68b4259cf1498be2fa79bce99fcf62f90f3118ca375d92aae6ff894af +ace8e863d1501b656c6431d66be68484feb97b54b937fe5c976aa6978f2973a9 +8a96b0bbfc7bb493fb7e192239818fe7b58a35a161044e112789c75022da555a +ea7b142483ddf0f065469fb386afa9a91a74aacaf3fc57a85facd78abfae5708 +dfe4a9f0e8d7d221b52667d74d138157fb35a789165605d4a36da6d5aa9d11b3 +888957fb19c8bb1bfb3143090b919b41ddf535783921ec5c525ae0598fbf4b69 +cd755ee383e30efdbf74e7cd9c5b418a9501b9e4cc74a3c568a5b61854179a24 +574e7f44eff9051a23dca024f731e3fec9c15e4250ee23884dad95c9fe2c7111 +ac01477e202086e27a661ede0fee7b0b4f2759a83ea8cd9c9ced3b2bfa05de1c +e59bebf466bcfa540d475fb9503b4a75aa40cba89b903b32350212b5b653eb09 +c816fb1a46e97bc1736a2144d57d9b7c61202685325e4902077ac08afe66db1e +65b6aaaae223dd472d64cba5f68996c7fde4f69c4c28b91dc65d9c5cbcc6649f +d46e76a0c8559b76ac00bf5a2f5df1aba3aa2fa8dcbfa9ab288027813b627f24 +ec6a92468759cecaa854e9b0155d4a94e52c897fe96607d592218664d1165a11 +5f6a12139bc9380b924bed4e16e30596b9119ff7bf3accc871b9bdbf27dba0f0 +c2d4276a8be2affe6e5e7824baeade0c504e7cbe2d7a2d527a349827b21a18af +c26eb08243882c7b3ddf1751978866658b7e1d52016de9b2a59a86d0744660d4 +4fb823aab5e47c7fe40075d699a7056390cf0b96d8ac78cd85dd417785f3452e +5b389164f3c8e9d099b5a59c4c5f97eebf5d5419b665527a6baebb0be3b4b78f +015a14c8dc0d3bfe379bb05f6bbd7aa54dfa9f119d6056212f9ff48119926921 +1814376f9bcbbe6bb15cbcf2121ac0f31816a8c92b93febb9e915286b003abe3 +201a923eed5a3c78dd4a9b54bce519965821eb0893630c05f3984acce21e7bb4 +447d49d08a3eb6183af91cacf1920c169b421bd7a6628edc236f7d7b73b9bd1c +0fbc97ea33417d675744066d45fe70e7136ccca645bf88a3c1dbff3745ccfaad +fa35790c97843ff87c1f30df70ae076ab34f741d7586dfc24fbb39d4dc18c829 +f4dfee86be991393c624a01df5a7521ffced29857759662a305e991d0940acf2 +e90f47ea0dc21ebae48a352d34ad97e415a601627f292d3f2ef490be17648bd4 +16783f0e4a5d86534514280720695825665b2b897b5b1df5e105a1b3b8f4558f +8519bad4672807650d689487a8f1ed93ea0d75875b5e5bd3b1e02e5b436fa358 +f76c2690fe2a61749face27bf143e74a4d334fd201a91c579e96c6388354b544 +add41ddecdda826244d2df981768186914ba33f995ed6eb7dcca3c6f5bbb8f55 +76fea291951f12f6d4ef5c3e99c10df53622959cf5406b70360d78fa0d194bf1 +ef1d43f65222f84cebd543a8853c2b788914c77d6e86a586a1d8907f936d47fe +2951bf02e2bd928f995391b37ac9bf9bf777d7bf4c8208968af57cb086111718 +73939fdfa8c8bad14c374e25fcffb8ca2a2dab2779f9e505a5b6c0f1b73f3bc6 +07bf86e8efd6b1df59a943ac27f62309b1f364b0dbf245f572651ddef1497c82 +93d44af05c9ab8002147f4b206d32f9b023f5373737be39cb4ca7fdad2e9975c +07394777bd2115041214a71e2f22717d53d81c6e6e69f7687df2b4b49b14bd4e +ceb2038750e3a6026f622889e891579d4d24e91adb4c5ec9ff12e008441fa3b8 +9e5fee97f5d24c2d8d17c24ef56d802b60ef074f71ca2170021a24d3b32ee3a0 +bfb16aa39bdcc9f429a27f16138ec7362e0becac0788969d24dbd13b6c885902 +b0fc8033334327cbbdc961a44fc3c4029fe81b88fe41132803626954dc567d0b +1cdb93b6305c24ef228d28eff9643bbfc83edb58df735081e00430d7770457e2 +0d8d190939273706c5b2f6c73d6da73526d153e6d58dd1dc93800913795d0895 +f45b6b9a7416aa5aa83b0f033bedb4bd248b9fe7f7f04085f582c5d64639e53b +f280dfe5b266019e2ddca06230f2e1fa70d96ba8c0636bf2bd05aeb9a4be30a8 +842c88684ed2da4e7446839bf1e2ffc71b202e31a6fae8b456ba1afe1cbf4de1 +1b4b0f9e0e1d08e59b714da5f7017e2acab322b5e1bbe305d2140a4605e41a12 +b7f4586645360c412431e16a1bea0cc3b86f98fcd1a49ce7874f28338698d8ba +61beed27fb1975d7f6eeebd07d7c29cc0abbfb3c601feaa2733aaa1d96a9d388 +0b3f5d55eeb521db6d723bf179198224b11003ecf1ec11f182c3eca7112abbc5 +a9759617348388b645b07127c8c9049925662dcdf2eba504a50303bac77b14c3 +86fff7253255f412eb4456c3ef80318671c081e96aaec27b7b90ae0b51449311 +3aa22deb849e0e2c9ebaf0ce145628b863735436504851bb058b85d616f21d8d +9dd0d35dc1b94d282ec222772347403d9f777c12b215dda4727faf422cdbb611 +36d3492ad5f22e98f29ce0cfe575ce63a3f7a04b599580d33ad8b5251a396fa4 +39f1049316b05952f89dfe0ea3b895645487a99d8fed522b94390d08e904ffa4 +1a389eba980f86431b48dd0ddc86deffab02abc169e77a78a607a472f2fcb785 +9bcfbc5196c5a252b6d87d75ba3c6d13cf509dba984ab34b262a7b40cf24cad9 +b94771d38835fa037b5f1b6314c96be80516b3d31522b5734d239d9348959499 +88221d4150b9f0d1724979d1cdd961cad637ebfd4bd23bf7f9e05205e8749233 +56b466475a3fdf26e5cda437a675f907962b12d32e8f208a3b78921dc8a37859 +0b4cd564fc665c688935228518f7e80f889a63f8edf44ceb4ed19bcbcf8c1f38 +3a9a0495bff6e5011e19da484a234cd4fdf65ae8a1edeb676b18ffcbefe4acb9 +74e9f6a5b336e4270aac4396cc3d38202050b84d65d580f0aa4255400814aa3a +79acac470c7608cf3eb70ccc80aae09760370e022f5fe2dc560f5a21ad29917c +35881bb03a8166d27115e5162f0b75d0f507cac7c1e5a2d0fa7bd07cc3302b72 +568b435df91db90f7ad0fb90d5d9b4cc6def653ebe975877e28c81a14d3320a7 +9bc230fc05458503dac38f290fa1bcdc13d7d6d5832dd01533fe554a845d847f +b304149388deb1bb3f4058237cbfe6127b407ed8ecf161024794077da85710c3 +2df4840889eb2d34a57abf0a49da348d9dfb82f0c68c4a0fba22495baa9f5c59 +a6358193d34904ba8bb1deed398848a161bcd9b231d9bd657d21fe185f5e6ac1 +b49104faded5904d0174f084e5c54c16135ae896e0d8d4a94b5d3f1e5d0a90ce +ef118a317e9872ffdccec6fb495f8dc947d8b5e493d83eef30800fec515cb1ef +56a628c6ec9d0e183c4bf8e631c9b19627bef283e927c5de83f45ab3bdd911bc +6c11b26933f3063dc3bea2a0fcac465057ca73fbf67b9590fa1168a7a351efcb +afcf59575e99e413bf40fd7d17c614e761a0930039a2c701d4e237ad9bb18a84 +b1143e6e3d6b5a5280781620c9b15ad715091743bc7de4c0ab310f25d02086d4 +38a2f71bb34781c85a2ab780ddec07ef7dd7a13ce23165ab11725eb4aa8391d0 +420fd0ec6f59a1503da37ce8467a6d554944922794cbe8750ce5aa0a09810268 +8c38b32c9ac7190ca6694e1e7c0da40022e3e9a02445e70aea54ea0869a0dc01 +0f1fb147da2fdfe64d6ece99f07e3a06eefbc841824fe110e436ef96b35a1d22 +d2679cc0ca41f8be79a4d2543dcc7fdf2f42ea3c5d3a04a643be427fdce0b539 +f72124836a446d30325251979447bf01e31c7549b2edff61591f74c8d495e0f6 +94514171a18c90b3d1805067978ebd1c9e8420ea54b9133ce5e7519c6827e79c +a331f1cf43f0add8ae45d1e27ff743e691becfb5a47ddf0afbf335d9d8c5963b +3b172eed6acbea718b20e244cdcff248833dcf7b3a7c72aec6285764c60ae547 +53af8b0766e295e36c2a825107fae5e9afebc612c129d6c9b9f49f31088d3f18 +217ee161fa60f64ca4261bd85212cb45e29365773117f1be89fe6a4cdd943382 +716108281f794bceb4706ac207fe8820b3d6ea90139b7ee5e490a2aa7d7ad09a +3d392ccb5022723a96ed2b7840998e2f6ea3e5d76ad05adbfde0c7ac908e15c3 +2a493a9719c03c11032e65e6994219c19f4c81ee1b9590c11f1da4bd972bba98 +6f108fe86ba5f26e94037c1e0ea6626e0fa526d6af14735e091f6f02a90521e0 +556c97252f2d2525d9440e18eec7a8673e4518a45efc9b0d2ba9ee5588d4c39c +06c5707fd408f1507c79596bf4d1b51cbb39e8eec0ad206744767cd7cc326ed6 +5db3d17bd5574703597e2fb846b421a4e1debe5c588958a7b03c0cd800bcc7f2 +2f1582d9863eab19abc98a8f1bf820245af8955bb691bafdf75351a8290b677f +1a905d194085bc5f81a51d6c1410978f04193c410cafd322304c9d33ab6b23b6 +5e7ebf071b1d7b76d502be1580c1e58dfd274b74b17b9e5e510f1afa1c369c6a +c2539fcb4fea2efdb1b7459f3598115adfa5caffa1e91556d76b4d471aef8225 +56673e60252efb5acdc122410a5cf43c9f5a6a56022509fb125b8c4a0b773a1a +3844f46b5b7fee24773142097b3dc80c397014cc2dedf024454b9193fb1e2d0d +500cd05774a3216b487adb5abf1325fe8f3fad0feb4a4911b5c9f85078bdf7f0 +14da81f02e68a29953f3a8c395b1ca46cf4818a98a3314458cde3ff77b84d3db +785ffb4252448b4a42cf915ac83c4c0cf0b71421e3c5bcfa81f0834a373a30c0 +64c8923e3688c047aa6ddf17b96a79100024d2a6bb159585cbba90b1684045d6 +818c13eaf4509e2c83ad278749a4b8163acd53f8a99bea07b58e23c5e5d08c00 +ff0f62bb8a40886ac861660b18af1b333a8eee725591cc0a7242650c28b515d9 +846660cc77403b96fb502e1a9f0021793371087654d11febf841a6104bb58ce1 +a1dcb5610150ae850afe21806b404c36be87e5849b289038a3a4857d9dba6148 +19c23828a873fe92274bf38dfce673e65caf3e09ff8484ea033858689c480d0e +76e776e16c6fa3c026df105ef700edcdd249cd589aa45c601b79ef76ba786b5a +4b6a8df6a02588ab1d0781d6f817abb53d27e8b142986a36da8d3b1aee32ad7b +0cf9bbb08d0ef4d77b5e0e83b18d92a713a8198a7b8c4de9d4c25b92a5014386 +43cdae759e6c10cdad953e07d348814685318b379b2600f79e6b39aa17ab7462 +e98e3e9f81cbc3d9e748e65ddeb3d001721efe02893412c151430943179d5620 +f50aafcafa591482176abb913264d5d551e2e2569785944f07348c523885e86a +da4de266cb78083af982e454488e4defb19d79f28916301e86f05e1dc10b89b5 +8beb4806122ac789520d3e71ee205ef1f418003f2c3a4dbcf048bc68713fef71 +afd1c214f49650101d7fbec35b2aa188eddfd01070970e0d4328c921e357077c +f893d36a99e2e7c7258af907966964270e2c7d9505cd9b0192a880946f94846b +81a0fe44119b491d5beea54e9253c58f5aa030c9ea41273434fdeaed4df51eaf +00e51cd4fe109ce4a505e00ce348e04ff3e841925f4164c073d84d638a3bf14e +50d7b599606078402612e0f650a24326ca65360d4946cc7b62a23dc0639e0033 +7e7f03108e7c6ecacb9b0b901530c7d97856948c3bbabad7df89948f3b9225e0 +e8eb8b5817f4bdb34ff22f224b8ebb70c50319787019643adfdf5900280c0fd5 +de0ea5004bacb86063e4a7cb57f58640142cdc001a2a11a64aa73fb3e804b3e1 +8aabe62b04d72c6d151d3f9618cad28f28679d122994f772667fa1ee0bcc880a +24e3949aaf10e4155e3cb7be1f4eac2db9dea04e31cf904c99737fd5322dd0da +14c5b91a3bbb1ec6e219938a1e099cc72ec44b1a33bd6dc746c7227c522b061f +f911c270502646024c02ad50e01c618c2e0580df3344ff172f6670ae8013749e +c37fb468455a38e9eff5a78444c564262c330a1f7b07da38c70220e23d1ff168 +a485c18a1e807694f57393562f8887c215c854a4b569c309bf83a1ebebeaf947 +9cf543dfa3de324aca1c3df864e025bb978564f44c00adb52edd9e1b05319f65 +5ae7ad71f06b2727223b76b1bdbae43aae5e3000936988b13ca4507ae0252ddb +927a44fe896ad089a853dbe41bbec352e70091fcdc56d1661ada059c2b29f887 +c2d5332e5697b475921fbe72297b2d6f303fdb66203b61af57158f6f082a3469 +5432a8729521fccbc677edc93f2adbc205672eaddb4a288fdf7bc30e37529293 +1537e2b4cdeba15f41e670f69fb69f1c340b0a87c0369ec55fe298f1f4e227da +bf3242cbcf103a1b6d739f7e419577469c33846d03b5caa91e35edcd9ee9b130 +9a08c19ce90878d34fb68fbb19f0f66be2c612a56ecee5e312a6a7ccb70bac88 +422b73a3b6617212157f5b4265b2122f87b3e0d08eb1df624ed65f7abb1b79c1 +6d8f052e2b4e842617c8341c666d10e573f950d009d15610e3b4a297783c26fe +b9552bcc1e51485dbc28710da11eb5ed97e0881a9dd289ee62185c8ddbc1da4e +6b1ef950cc199b43cd6cebf32aadd664da90d5808c797eeb947de621e1857aa7 +cb1778ee4faa3202110ef0aa3c1a45d325bef6dee6d85c3c94f954eb726e56b2 +3559ad9fd210412cd932e61ab27d9e7b1e81acf50016716d9c218e5f99dc5d1e +ed04834fdaf8d436f3749ca77af0dd96384c19f3ac8dfdfe3b0c68ff66bebd53 +5ad89a12c41ab9773828961d358e1411877f13863a7be21cc76c73e26dc68674 +fdec7f88568b3eec307146a2d1f170922f7e135c910da21c6a7aa446d39f76ed +5a6773112b8156643f2ad4dd4734e500e459a05c245399b71b0344affbef5e52 +81112756accded95114f25478a7aa7d5e7bec6f36a3c415b2ca14e466bbf2e5c +e92895fea3a705929e77f06c3384ecd6d30ffe2057a0f5b654d2bdc680e0427c +158b6d0cbe9424146a32819da7c743756ea9c18c7ce97ca7c3556ba03ac38c0c +2bd577b17c182d9d2fa2d97c28a2ba99abcca77034dd68a68ec6fa2c36cb52b9 +046ac6bd6752e9d3496289d761a49b2a4787a33e253401172ff36392d42db73b +d6f32fe3b0e7f75ff480cd94ac77d44c8a82b301b5d1380984eaf7fb961368c1 +973cd86eaeaab97fa821a8015bf075682cd08c523f9b8b2ce5f91955639ba791 +fea6e9ce4923da1ad89a8b28fd808496d370e0ed3ce2edb58866bb0cf94f9c80 +607090de17c627a10e062c9da4f1814c5a1a3800b62515612c2445e0867b8891 +685bd9a78f0e4c931d5c2bc64811713395af006a61bbb70f682e6760d2ad1e2d +9de6110647892e3de1d6d5c72bc19c1fe9a3f9ff6bc6b02b98bae900444ad46c +04e59b502bcd1886fb516e0095a819699e21bdb138694497d3b65d913a0ed011 +e409e3d99ce2007ae1beb5621584caeaacf84d8388dd01ad0fe2b17524b79551 +197e435c3fd80a2549037504aa0d4d450505ae69e09615c9ab87bf6f50eac1b3 +36b47ccc67090621f690c296ffbc2bb92aa366b9467896ee93cef2168f364331 +fc30a58741bc5dd02131689c6423017b60a70a795da14beb0f6f1fd2db3e12fb +06f66590d610151f94870cfb2be27b38dd798bad8d0c71c3ea558d09ea8911a8 +8b560155c1240016b6f1400bb53026b0791c7c0d784f0afd2a31251a9b5c5dd5 +661717ec666317605d3ccc4656e8dfc6d98556db2853c393ef0deb9a0b4f6fe1 +e475b8b813803365d0434f1cce9de3b9606435b1b766d78612cbf218249a82b9 +63942840802540be07abdd2e42569863f6af11c3be95725c1bc1aa316eb805b0 +3e343bb863be7a3526da971802089970ed815e75a6b8f66e97e4d7751634f695 +a9c381ab918c40680100ee69e73cd6eb69bf7904ca242b2f2ee9b9715312c3d5 +ffa1cb97b886bf2c0eb4c3792c245d21850fdcaf25f8c5550a2d225f83ede50a +89f1d65504c630c0d1f51b7c63e667d688642859844da35836b5b6ee679c0f60 +7dad48614024ba1e2becdcef5017e14c02b77eba6d1853b2387655853abcd728 +45eb2e3ca7c3da8382684412a7eca888eeea8dcac7bab4fc791248675af414cd +6bd1806df951cc6074b1c1019e3cf27ab493e421bda830ad54c2965badda0a2a +0b8afd413460360a6fb5c4c42241d3bc1811c36461b2cc17d23984773e638e70 +ab888d167025c10ff0f32d7597b7dbcee65be7ca9effc3a62ff8c70dd7359867 +c1a25668b3a665e9bc4eca605bb028355a9edd8533f9af7db56595f93fdf476e +30af43bb9eba76c66515354203eb790f122af9fe9464312247275f039946c061 +d4b91a6e34cdda9bee28142e877a18c47e6e2bf6a84cdafae98a1547a8585909 +cf03b2a0b02d34d50b7132beedb5e078960dd5beb0d6f9f940734ee15ca7eefa +31b5aaebf0a1ec18f0431b7e5245ade279cec5a41971176dbe04af4322f2f8a2 +5c579402f5524a4aa730b3f77d3eb5d9b46ee68158288745c515dd1b65c1e666 +0deff07842092b641c5f003f85b19eb7c03be71191ee1b4cd7af6581edbaeb76 +a445248a3e231617bd46cb6ff42e1273bcea04f86394e3088279594d4b58f36b +6000ac5b4e2494cdd384e7236e12bb8d2b5cb706cf6336ab7567683b442dd224 +ec15cc1674e7a9d565d941132971ac7db252f684fb4fd4066427444a01fda5ec +f348935fdc5e0dbf9a922209b8c57a9c5258b572156f9e51b8b1a5292ffb2bf2 +2829cd35adc5226ca37e0dfe5e33dccb4d444e07add3662505aea5a835c8464b +1ec355cda7203d797b1765948ab00bca9d915a4408533fc316ff141293fbeb3b +3811d5d6aa94be52baa7c848298d8069d8ea5e9c162f0b68102a25d6e0c25405 +1f7484ddcf8984cca5e1ab7e4a6215b009ac9993c45703fd2d59781e191983b7 +4dd70c7440fc53726c040a6f0c6c05e11c8c3035afaf93e692e35b2915be55b9 +70434916738ea44ca15798387f0ac552d2b1a0736e0846ba19d883c7601ebf55 +296a822931db48fd38c0f4a02e3a54cbdf16fae0eb6bd2f320926eb5a9eabce8 +c37c81f56f36d4c08d3afd5072bf8779b12949abfd2d081af230ada474cdd45d +5477a03998c161fc5813272756d0874943b56c1cfd302704062e578d289a97e2 +130bb15c66e7692c7b9006bce044ef5b59dccf9447000f4ee7f1bd42e07015e0 +c39d2c12b2138bb255d3fe116fbfe0a1e8c956c859aa20f2cef45732445d21ee +530f7730d4a6670ba2435635a04db586dbb1985ce86459086d96cb46be174a53 +20ea71321c433eb6221f9f7baac2d61cb06446315d517e3eee827b6f73469306 +88410e3cec606eeb8f541059246f9eec996f8c9f0e34b8a26cb4dc074eab74a1 +5d48dd389bdca9ad50b448c514be1568f401ca191deab793a2021bde0ee2644d +2140946d1c013b4b95eac0a05c16ddb3e5e486dcd6b23068f8cce0dc449db812 +a7a0e921e577b1d7f084d30100026daeca01b578d487a9c72cf90a43e5f3b0e3 +903d757c6a163a3ea489cf28de8c37ff4fb81b7977912a1f86747971752e5ccf +31e2713d95f01d1fd4547b633d74afb3222ccfe73db304a335501e955f80ac39 +928b99bbe32706c82832dbcdd25cfb96d57ddf5e531c09c0631761922dbb1cbd +f366961a77955c9bd8ec42a120a97b0e798417b53517e31d0eaf7e8259de4e72 +0b9216a8f249d1c2ea14d6b7e34f4596258161f743ef0bc2b27f98ee2b558ab2 +05d4d804ecb533a1c9a0ab8c8a7289b0f4707afa603e4956b88e5de047a4f5c9 +f1ae6b8a1e3ae6a8746d867ac44807b86dc72889f415eb543bbe6018bf2c6dde +e1c6c8650c2e262f08e1f37f1bcb80f9bb1ccd466274360559c345288cb602d5 +f9c129d6ee345b631a7987abffd588e043245fe23e2dfde3d7c1fb5ba9bdda34 +c062954d1a4e68cd1ae2e972d8dfb540dd12ecaa72269e5f62943fb9957437dd +3fa1b8b75472a924de57a6fe84c1f64afce2e7629f9759983e823b18bfdae81c +c016cf9ad8fdcccf33a39eb7878cfe6ceec4a716d4234f47562d99d0cbfc026a +a0f8cc8aa6c395c9205ec70aee522e2def7b6b1e99cd71cf0aa0de6533c1a56a +ff2d506b129521e85a2602086216efb73d5c309ca284b20b15bb63f1a2d8c7da +08bfae28c6be7c32ae8bbea86186e47402cbc3ce70e69e753613782c656b74ac +d147362789d7f704b438161d0cf1228df40c67600fc19b048070d735d0c3262b +dd79d1b9e73c7d05ea6480cd6ed89a9a2bb2abad67303107b1cb09c7ae0e299f +c087efa078a3d8b212ac0407b6a90164469eef361c6fee9c5b56b4540834d61a +f87fed9368cd06659cad3d76a6b8dba70c6b8d92974779fbdc7598afc869be36 +dabf1c832bc4f6d086ab654a775ddaa8522ceb770747ce57cafc2ebb09e302dc +090c0fff85c93a288d6dd82f59e6c0fe2503d6c8ef1991979399663e91557b26 +15403b21c9121226a27f89e7c077f70f431f51006ec1c6c0035f8c6a035c98ad +c3de1cc3e93ac56d6d779c989a8ee2c6eb658d8bda3eaa8939554bb01c1a9cae +711a81921099383eccfdd7c47f267954a706e47a89bf6d7265d52d74622f1791 +54cd546bda2927cd8af5d9ff7d649ab35e8a5d92f5c3a8708ec336532899ef4d +2ca16f592bbf77c311e2b49636d8891a8452847f8a0f61c5374df3603e11e8e0 +1acccfd9134a3d7587a85b32f414eec72066891045336e9bb6c1ecd35202c6b4 +f8115a1b958fe276baa599d951a8bb118f6317c3c5550000aee5487a5eb8a8bc +dad984f66d522bb2f71a9a1a5454ed10f090ac48d220a3ae9cf429959a3cab05 +c59c09f01974c399ed6948943cc3445530c972ca51031e8b8f0fb1ef3b987156 +41ae8d981e75a54402220564a078d5ffcae1a74650b0a98667e073a3610927b4 +e570cb7aabc88811c2ceedf37ce64b814253abc5f5cef9ecc88ae9e6789a9c2d +97de7da8c04ddfeaaa8cbfadbb63e29a9b5bb0d7ba27d4214c9c363817fca8bf +05e831824eaf2b0610b6f511471aadfec9eb2b088db70359d161a51e767085ab +0ff69d3309fc80237f4fff87aea27cf1db5b03a5653de099f7543ab8ef84278c +8e3a97745b758681bfb940dba95d63a759cb63d10c55ffdb98843ac014fbc260 +6ea95aa5a2134d9c6185aae6b51e92b4ff44e811cb95577307aa552eda00eea6 +7c27d4483e0e236b3d8b9b365cc9af3fa466e9d5ece2293fef59922fd3f258cd +93004dc6c470dd1978d0e9d3e357dc0a2a13401313fb5e1a2e069a14d3e09161 +96b017365ee89d873059bbd4188c8c85397c1e4a55c79dfcaa6f4840b15e2bc4 +239abad46af37f676d73b160d653d43d8bcd21818accdec68c2ccfe05b767c1e +666550865ed723e6fa63a862160a016be5e964668913b2fc94a66986a3973a26 +ad1a22a1a56f9ece93a9f60ea1cfc99b28953b6d1c7d37f5267185a3b38dd2e3 +f338ef264d9e2889cef554c3b0535d2f04d4d63028a5b31326c774b523099a6a +4e330922eb98fe785af7e08855ffc179635a6446fbcc7b03037d08ea53a47a7a +e51cbe6271c5e60d69f81019e9bfe11d65f3ff458f812e55d256b22e7b27cc31 +da848fde5d4a9c17326561f2b2a02de1f111a6c66a0f08dd0357fed1adcfff78 +7dd2f192fee367ea866844722f8dd9714417a24bd19fa4e66573758eba95e1b1 +f67a541344e08927d4f96275f4e60c86a99eb07284cabcf713eb9d84290dcc95 +9b62adb07745b8afb1e48877dc2915f48c5dea19500f8657accd2d41efae5f97 +97d1b69d879a0e150649e823ddb65b643dba602061f5a3e3c4d11bbb857b4506 +1744dae99992254e3478c80221703953b4e7a327b18a7e196c22cb7e873e245a +651d3d15cd67c720584b591285ee8c59090a0ebd63ecddb25df96deb56c4beb8 +3c1133976b67df863b5ec1cc1ca7bef319fa89ebbf69e61b68e0ca238558e194 +fcf954cf99fe5627f8b2cea1fa76d56872a2ba7c5f1666bb8640227263e3e6bd +5abeccf01f8074a8004b4c10492de2274eb6fe70fbee2f07da0e9693751fbd4c +86e07b9b5052aa371de87260945555c358155f7bc461e8d5b50ebd09d19c8380 +f839a3289a26053c41f8ef47d4f91eb0f6fc7be7f41eb2c64b0052710ce2ade0 +8ddb7fc82ec8baf3c4d3135da8f40d6468925d01dccb01678351e6a19dd8a187 +f5d2ffa85eef5d9b1e0be73942a7c8d179e2fb1d81124dd9d6778d2940068a7c +09e6c65b9cd9b595b77cfe5c31d6c9eeb28540bba0494d90939639309d600e39 +2856bb4324a31a7a27f801efc888bf85a6cf8f7df4d19c1013df008a3602ad15 +c25f73bc82e3d7d6f4197d3a7073c5f84599bb6ad646e6b543c0b22b2446c3e3 +e5475fa306f96418265f79674ff58bd5b4690bed88805f6ac4bec0dd488613d0 +b3a06366e8d84f9da4c9bf85ec1b8aa8eceec3b95543f214c621ee6b2580ea62 +eb6a5bf60f77d656d6205509c2611f69bec685d147b5e5538d4d03fda578d3de +d7a8297ba332ac9a518fadbe70bff80ff76686f1ccd459e92e0a2925730f0690 +6b474f895489a8052f928dc8b258f80ae24331fab35650b150a11c396bcc378e +ba0777fd2d720c9dfd4da24b7eb3f2c658c5f872263d8c52584d6d1d3634ef5e +57f933fabaa4675f1cb84d09daed8df8ba6ee2497eb1ed746be1093119a08945 +dbde1f7d2b5d2b15b4afc82f05d8fa176257aa5ff3c68c91b57f878dda5218e2 +a7d7b66f67ff1cbb5f22d0d4cc6ab42611e31b9d9dcfd69dde62c5aa3b416265 +2064b4c6f0b236c18a55737212689f86a1c0884e3a6bf03e58cb951ce351d7ef +0791e8bb1bf90039eae6e473095074f0d01a5394c7d09fa3c71fe9b6fcaf5d99 +e00e93dbb808e4912e4fd9c9227f41fcc7ec70e684f30f8fe9a9a1d36c8b0733 +4b85d1b19deb737cbbb1944c6968a331f4b7f1ff7f21186b3a588bb6762962c3 +4c89859bf389749528907bfddbf85e9ff13448f1f52a72c78f3ff4fe0e3625dc +bb29d52e697f3c0dbdacf5cf75704bade769a25d9e5acff57858f65e8acb10f8 +ca8e4aee00a80c31b5b249efa9cc3aef465cb810f918e3fa1a3ae0e4d8fcd839 +f4886352cc49a8aadaa77022f051c560a0fc461cedf3a429e4fae07346fbf746 +f35ec75fdc77f1b7e6ccca702bf66dd787e6c3165d4037cf2943c59e5fe63839 +7d238f7bac06633852f23b44df8a96d60cdabb24233c6f2db52a4006480ddfd8 +5982a1b482ecbf33192fd3425c05210bcf500dcbb3df20abbb7f21cc73d18766 +b1040c4ab8942c097e1a0a8de7bf2a97cf798ab1b0402fe729ad3b9faa69c65c +5bfddc04dd473fa5f66d64953db2ec54345bb9243eade72e08185fd90ce4a39e +0490bdc0933495c993bc551d6b34b7dbcb89a807f39d12d9118de54c3b2443fa +3e2ecff17d797418a1a5d76f187e04589c435a654cc673cd26cf956d2004824b +c0eab7e1d5cab91a06e028929ccfe0dc8f3f8c5b324ac3ff628c12d7cfc117e9 +1e8dd11dc725fe3f65f3310412b9d56e9d8081609aed8cb55bce973cd515fc72 +a80873c4a3c0cd43ea1624042aa72fa1de93730e52cf4aa5a9f6523d92d97790 +96b1d4f4bd34d18af1d5b9a3c2b813050f629173e94b6399fa44b5b6ed83b63e +cecdf0e17c2e8d917566d25e6d77c2781192e12906850167c6f84b1a718a7923 +03bae5863b110431739d2343b865245c9c1ab4da8204cd5fbf5545590d31edc0 +624317485b0dea29daaec6fd6bb7fda31cc6ea00e3efb5efed4aff14d0068a4c +350cb1f7231b1ff85666c2fbd77dd406ac72691de726950ee7ccceef1579761f +7b23a22bf29c5870802b9daa3d1d8bcc08da974f59380623e196403b635f9789 +bc84ccb74968514ef2ff5b56caeefb1692fe64eebb38bb8e361519e06f92085f +953b9a5b1ce8907ad7497c3735052c8efca51c927cb2c72bdd8ef5c8093ad928 +8bb000328b309efdb77d70800b3b22522b3b1df74db78e64c07688ce32b6889d +45da9ce932d5a27515b5706ae3ccbb005ac08efb276a9f4e49becceab45aa940 +3a1f94014461289c06602f0807c16c202660ea67599271b339bb10cc246f85bd +575996b053ef583b96a624e9b3bb2088392bb97d5b9e55a5c1b78e54d01ba157 +ae1939b4f3fbb1157fca1fd940b68ff3cc432bac82dbd159447d5b7dfe1d4555 +52ab7cf7292824041f8ad6d45a64b8cf768a19b5d21e6b53cc8072d2c3df69e9 +effd174ac956c3bda653da0d21d84fa5a1bc0a12513632a69c70717002159629 +c9ade905c330beba2e75418d2b41e589630104465748217220c87fbe466ba118 +dbc8a49e7c149d4993d5dd89b58723a69c6fa9ac8a7e53133cc43ffb999edfce +2ebbfcd5db130aedae0c2053d8f415dfccf3be21c197b937218feb47b81c388e +f34b2aefe1c19bc24688576d3e0a49c80fb529accf23e15f11eafdcd0b9180ce +83cb3b3708d13b1e4e54b5248710ba3efd3fdd3f1b5eb3178d232732a89c488b +5016303f5807c16d2228e13f9e46eff5fce2c138d7845b86046b4ea6987d9f7f +1f667f83543442da94b0a414d9cb0e06d66216e91efb083ca883ee6ae1421d2e +7c5375fb3241169471150411cd1f3c34f3b7b20817b41ec80d521e88c15aca28 +5174a2c02a0f2296568a4e157411c000433c89d718e54603780133b4b18e66a8 +542aa5ed26b440a5cfeafb15044601b3bb5cd6c0941dcfffe3542ef71969dfac +c88aeb00c1d40dc6177eaa3db6513ce2c628551ef79c20fdf05e9421eb54a8a8 +542b2f4401a7f5aaa3781aef83889f7f48f9dfb3eefe6c8b51ab1f37ff916735 +d2832a6064389fc2fef5b60c07d448f0fd8281bd0830371ec93bce9531d74c6a +f67dc4fb5cd09c2513b293727f428cc5e87510431ec94fed40652fdaf0248673 +84f48b67aef958b5a386b607949f45486266b34c5f9ba3d1c902f7d33ad99f93 +67a54aa6bdbe711d12100192fcb22dbfd0df60852736485537dc2dbc4d53f43d +3407a06ca624ad62cdd5e94764aa5eea9b31e42ed1ffcde8ee443756c844cba6 +137198dc7ae259859b651cd2f0951627bf701eccff266621fbb50069efca09c9 +a3347bd5ebe0e33bb53666eacac3d7f98de8525dd9d5bb4140f177a21410d95e +7ce153a8e9b6e5c2edf91fc7166e509122d817fb1eee02eb97f9ad78af8838e9 +3261ea87aa7e8d461e6f7e8d0b1e6d5418e1ea542ab21472eeb0bc2c997fb01f +07799483358740d519f7cc57e0426027eb33ab3012fa5d565f5cb2fadadde7d8 +70f0a1af051e29d1f96ea95b5957e3dc193d17a093f6c7d5a2b386bef82132d7 +16fd8522e6aa9171c97571b7f14cda309c75fc34293c2989da0634522d9dbc0e +5cbdf4b59550643e41cf560cfef5da0a1b4b8eddbb3d924be04410376a0ddad9 +325272eda32504f899194b2729578ab8a8c0e3a6582102b683bddd1ae7f1200b +f51119d7b4a3f6c79f0efcbbeda584f1ab39591fd8e13a342c463d289e10b481 +315eb3c7e15dc51ec505f2149e50ecbcc034df54486c64fb2017902d7a557693 +1eafaeb3781779227102720ef8c9384a058e60b55fa87cbe02eb5e5a9bc8a6d9 +71f0f8938aa0805e168a744e482bc5ba215a4718ee07f0e37d6c4dca84fe5a44 +78b4d2e161768ac9d84a4a67f18a1fc759d49353f6c3ef0a39c49d7b91431fc7 +8de3a0296dfd7dc41476adfc06e4bdf447227e5dd0d4e9317e32bf2231b900eb +2d29120f2cde14ab562dd6f20a358e175748ebabe994b2cdcae8400c6b899b70 +445c1c4156a3795f3f320f307b2fba6ea957d8448f4b10c5241aea2adfbb764d +2a57dae85320543d698e929ad435447f58e772b1e01f56366903dc43702ce609 +31fc57f6acd3b33ae7697f3eeaf817de2a6a9f437f152c4255fe2a25de292be8 +cd0fb3c38bfab2e68e32030e1661596a7c99445de845012119702ff713156095 +b55ecb85efc83512e8a464ecac7e1a02e5e827461665645fe8260d7c295159cb +1fc1339bc37de2a72b9b15309409727aed46bd2cb568f33e709f897c54dfdc9a +e393ec0a65eec730e361f6540121aa1dd2cbd24dda389ca4a717b9613d2570c0 +e5faeb238a40eb9e6b37c49d4d56a950cbb84427d1bb4815a7e9381402f15e6c +b09703aa3602aebfafd0a9a26f8e961e5ab02fea46b8763ccb8777fd574c3870 +3f31000b38fb2fb8093129ea7f84096f933f4820c6b255faae747b1b0a293c89 +6dcdd3ea6074fdb9e7f33b965e693c53fb6b48d88f573b085475374d43b30746 +49f37ffdf9f4f3842cc0d02e370634b3592588c2cc68675ca30cd40e2d6b2a06 +8110af875a723e73df94f7181ebd915eafe93e462c635cb69ef14f78593c961e +8155fff8375a8f96dc2e1693c5ffe5030d5db906161cd339d849afd515acb075 +8e7fe0d1e7dd571f8a298d45c06ba5f3352ad6d4ae32bc051c0fcbc8533c0f86 +36667f7cc0f220a4c8d0bc503abe00f8dff1839c22d77d4a525004fc38f7fee9 +365d15023ad3cf01dc64efd7abee52477376594bf13b9cf73973fffeef536dd4 +f366fecda0ac75bf2b7be980fc1facad7343fbd1863f9733e7ae204284f0fb3e +cb3c4426481a51d65eebfb245da73157ba1a20e72aac408f1f6b813e598be767 +1dfbd942249f1014eff7f699ce0ac12cad661c5973d959894d9701c8d3b4480e +aa802720724db4fad5a7acb36640161a964aa9cf01e917768efb377ffe94d744 +21d58cf7235903a0f65ee22cc10ab07c86df62e006da7985d86f9b08dcc0753c +7be958c8be68558542213fadf2a5670838b7c45090530fb0c678e1456ecce593 +6671528f5ebdaaa75afba0ae30f0a19d11b101420a5b1c45b719fa79b94451b1 +fd3617676ee8f8aa39c079ca7d90194b87ff560ee44cedde57bfb8fdedefcc11 +44d41293db71865529359eae42bf18a97d57a1fe52cf2f4b4417eb58f34e095b +ab3d9b4567447335874824ef294eb60b243d06edab62ea112f489030faf3ce9e +a62ef44c94bde81531ec6938d9971802c6b01492bb487d552052ef83707de93f +3b2c413bb3a8ef0fb32d6ecd455d3a0c40381c2fc88def6a87a69f8ed259e04b +f71e037227fbac50ac83b2e6c2c235ca3469d0ac4f78a5fa2bb2b1cde92075d0 +cfd8a64634075a5701a232a9d8c88cff05a4246732353b6e6838d0a3d32f673e +6adeed92841ff92692ec0aea23818ecb1d4e037b2ab4326877067b2c797cc809 +bfc10f019886a48b1ceab70bd9f9da45a648340fc3dfa63039c74876d8c9e749 +d7a2791bd4b91f83b028ea464ccb71471cadeb15432ba0febec6b444b762a7d7 +e526ab77add39fa2bfb4e6d47f69adfa4a24afd9eb068629324b526ac89e87a1 +5ad2b66b5b883ec2bdd3eb2fde47998a3f629f3274b156543b5cefa200833962 +7c936af2f89636ab56e1ec7065a42fdf2c82b4177d42a48326198b66cb1a0e5d +e1567e80b846ec7b28f3c5fa09a0c05b7799b5b70eea2db7f5ee1dfa65f8b7a9 +14dc80bf5e4b280a9f7b4d7bb2f8f784a233d62c3bc9a7a5b6de1efe9dce79cb +9457b7551379a290f80d2eba79f9913dd417e21113cc3691597c3519ec39ad38 +aa63d04ba24a2009b744888777e8047ad6f4ef85fc21293fddc6853ba64901c1 +fc2da1d1b58533985ce7d2d4903c2fd404f65926abcfb21e4992feddfef95d92 +4a2f0a8a367403530586ea1e2262b810eb1ec80594ce0f4ee8eb3c69f13a7344 +11d2e1f351394e9afd3fd561d758c1e23d65723f66daf08268ba07e794bdb29d +751b69a45135c8d1c16ccaa8b340aba4831336796158f28b216c341bd508f8e0 +a6ac5c479e0bfb55603721c86e6783b6f29f1e803e8761485ff80de5ea4609b8 +a67bb2e009d28be0ed05ddf58bb05cbf590993ac586a84ed65b95d7d7f363542 +b54ea07a9da21367d1e83e979163f6dac8d05e6a748b95328b25d4fbc9546520 +3f2897592602fb41bf4f244a80bfa043fb3c6d29b827673dc9933c0beb6d9de0 +ea43ac646f84855be48e1b8b6dd3bccf11c2d183d15aaff37730e27a7f0aa1ab +97e0ac16d62be1b320b11d23a2ce09671604115889c09f4070d4666a26821693 +79c511a99a426c64edf129f2c272fa7f3580e1f3b41a838d714ed3aa50c4d991 +a5a042f615d022be9f2a11d88e4ab384f656d9119cb013543da577d340687dc0 +d01a918d6de91bb81972a85d28e944bd7b3c1bd37180e144744063dbe1f7b5dd +0b097e47f264d3ca34718261b92242841ba9d21316c4eb682d54fd48e14d0fcb +6644192bedca2dcb2c4b193a91b91edf32193b4e7e524cf42676cadbff6aa8bd +ca371c508eedf480c948d905fc079b3e5ee06bc49e9678a56a606438f98c9936 +ec525a76e5c361d26c1f2bd1edf8c9813365d174a09c5ac26d3fe23d9e35b0ba +bc596a5c0f7a4279d0981c065b0c257d1f2bb646346cea24e2a63aadd1688136 +652dc85226a255c97233f1f5e3ca922b9436e304d326c1eff0d83a3f0e6a865b +99c654266f47c0be61f965505b6e903fa0d586a04a743e85399d5a0a83630cb3 +2c44b87c4974b394417728de9649e96f2e5d3d3ecaed7a3ff55d63b3076faa90 +0843a125fdce55b2c87431a295a90732d02e8e547fb0b8e53e9c088d9e2e441b +e2042fa7d7cb4da795c38bb17b76bfb94965e0164c0930362db0c28bc06e1fb2 +a2d9970cb9bd25610e22db0370bb8ac8acd2f5e751948f4eb4562c6594a3ada0 +c799c8e22fa4b7c700730ecc48cf22504e0a704ab51ab5b4082e9b75bed526dd +32e58de7c1d857c53e68d4b2c8de373123cf118c96c4251ccfd7e5db263444fa +0ea513eb2ec4233f630be2a7e770334c022ed693860018c73af89e010bd71d19 +aadf9484d610b7f412ea68773b9633d9c042993fbd304534b27338bc3f8896f9 +3629de258182cf4ac67476ad6eb39e8f8611eaceb0404b61d23f3833eb0abea9 +9f2ac6a4ed7ae1e9b2b2826377c13cadfe9985aae0a1d7629833c9806cc83047 +9dfb8ba71da5da1f65f919718be9bd2dc0c7cff251f5550e539831e52f9b7077 +e73f78835675822d94944052ec91c5ac0ba28f4fe1cc58f6e1306ee167ad51af +95051a11278974f4ebdebd576b0c5c0b53e0d1c4935f299e279632798ce749e3 +dbb3b10ca1a6d0653bc1a9800de240020d6ac9d89fbebd77d6eb530401c8e5e8 +041fd3a18ce6785ac452ce452ddc21ba73de2e1788e9f4926d7912c66f089bb1 +6badc744d4a19ce3e492d3b5d8b644c754a6110debdc45d4c5da8f336c8540eb +6fbd1c31fec886bc6763e714c59f1da081bb32edb3cd70ab395f0549cc482df1 +8b9a53853b065136c05816068fe0ae929e570f4357f8d58bcc10655c2a2c0cf4 +7758863fceda314b0a792657d3ed3a71194d496caf394e72fa6436007b6c0dd9 +59267ee7cda1f660422d54c2782ba0cd26cfb1455057a6cb6106c58aa6205d0c +2d5bf0bf0777b2f914164de89bba6321000f656814e15779be1818c8cb0f3ed2 +4a66e0756a826dc8552acb17a13d9ad41d7f7ab8b15d54182c0d18146052e08b +c685209916e58ab1377d531a6395edd26eb65c4eb2c8d0647ba452322ae93161 +1eeb8c671001d602cdb54aa7ccf3f4ba3e36ccb8669db6070c786d680827d544 +d14ec7c4d565b5679e996fe49e0b416432ea7e6760665bc233c7a65f69621b0a +42b8a782d76195d7813e3e69aad347199ae3a3824e4ddd116be7f834bbafd8bd +89eebff0a487f183bd971cf0a3348c8f8bd845a5e4b400d8512007f550578ec5 +2200d50c8f74236f7832f49cff4bf957ed8be648236855355d5340028f2397d8 +520cfb216ab710cf0a8aa110aaa0f57eb86bcc900cbc3da339d326ed61318f9f +2bff20008caad03ecf8cc9fc863174c6ee00679f8adcf3a806b2c188e6a7c841 +837a1b88d7d0a0e2894e75e6276c0496643c2b3f6c56f94187886cc2c8b62cd1 +78e10c08c78c032bc73f74cb922398e1004c512e0b8cc7f9e157821e25a12141 +d70ab4ba77730304d5d4ba1a890536f689dd31f1c92c16f5f210e4c3a1fdb036 +62794f160c85497efb4fc594bb01d711dd4872d14382afb9edb362bd4d7b163a +695ee155ab71adde88b72d58cbbfa6f83bdf2cb79a151f248abea26dca04e88d +0a8f5a644bfb2e88ca620f47b7cc3fb1b2ca4ffa1e7cad4f9c5e56c6c84d3959 +4bc5ffddb0975fa58846c332680c56b72083a3a0b4054f6b8a3f91ff8c3636a4 +e0d08d26194d481fedc967d5b879018f1940bca1edf672117f1d53076063b6e5 +6df0d989547117ba1bd109a3e35329ffa085bb7f1f7b5663dfde4cf70a10317e +63b6af2f0253db0765575ae7a46c76e329a5553b36f5bf1d42737041181d8f10 +50b51ed2452a11166e7e33d33e33669cb7206919850f396d983a84c1c3f343f9 +4e48a3e426557e4283d6235739f19e00cbbc971dbae626a7e732211f88c4b116 +229399c8cb0e5809c0e2c61b589d7e4a6ea418ebb78e13a7f3246619c9b01fda +0dd41dac84856681fda3f92cf58341277a74e3b3a3328c4ee8d24187a571bad8 +5c5aac8345686bd466014ef256a503bc896f4186366fb000daea1f90a4143377 +9741f2cc66730e45feb28d904dae8d5586797fe832aa78c66b200a1616a526e2 +74a04f6734ad88b61468eaecbba86e10fc5132c461dec26a15daac9b6e3ca191 +9abb2fd6c1f5a4908c6dce248d3f2b0cd159df57967bab38414c52a1d6048451 +dcd522b3588c7adb0418454539e1a929fff936f211e5a20383fdcbc3ad8751b9 +fe27a2d8cca2be43ad124e6cb45bde5d6851e3d2eb9702e8a3bda55f7eaa01b6 +e5b9466fd3be35d969f4620fbc02fb8d5259b3246f5ca759153276a5b82570bc +db40e013c249571a1c29c19f1076986838edb1e9314911d0742fb666d732d47e +60e5386bfece072856f5ce14db38704422ca9b5300fc495309bf583f836d0a37 +f51a2c88523a07e0df9abcedbce12e768359a02704093ab088176f9e64880317 +b8f18f00a6cc8e5677ac0315cb7fbad9c8bc7d2a743db76811bb51968cf4f360 +fac85863976a508e52720e25d80ab2c1740c524da9d5918c5c5264eca745d6b6 +19019666b54a00ae519ec32e73a848416d41acab276a4dad5f7318bb9ac96fea +cbdb60aaf21ddbe880c3ee8bf6bd214dec5d6d8c7cb5fe764ca940b810b8cc80 +f89b2a5849c961c420c17819571c34982427b9a05c5ede886df502643418eab3 +383d662597b21a91deb0debc277af70dde2f294bba8ec4c11e96fb56e9c3daa5 +73ce2b15f1c5ff2a0ff77ba7ea7cdbb8540858f51302e466e99dbca2289d55b8 +ae325995c208b33ed9c03f6a1f2489a6596f1fefa5e1d0c20d71832eb0c4ff61 +f3ab26267ca7a03dd265362982db98daa7c6e0aee3e808ea089386a7a8312bf0 +8b9b054283e089fcb7831d7ce527a1fbbb58996fcd5aaf6a1d3fcd2dac910a39 +0d6fc0af665e0f5fb1e1999c6bd6bf6e3eea3cb4f24a22cb8b12a0363d234748 +b229b43b878148d2561aa606318e4a4d85554f08e53f1fcaa43c6cfcf76dbdb4 +579ec8430527135d872bb3587b5fc13412031bb8897c6e29f05df1cd2af3f249 +4f20d9828a7edec67439e2d8a2e9d8e0b0e38aa8dd310a2572aea0f30b75b0d3 +d61ef56b2451012bf8ee777670644bcfef1c55f314e0ea97ea8fa049bec38bd0 +e1b968d03fc7bfb177094d14770e1d76cbe19ed2569e9bc169455cb83e37c90d +04ba9160b2dd5ec462d5e5012ad7a3486b24e195e239c4e44e3ae5c99c7c54d7 +e8f453621d9c0cfdbf7e2db81d20438edb164556ec0fda938c10773e8205199c +cb253b802541b10c21fc48ddaba1c38c0fcde10d607171ca1b3f21893c0a73b6 +df13e2ae15dd7e9c3d34973724990022c2a8711ddf0dc6654685f91ee5fec87b +4295326514606b504fa78a0ea8c057c24c714ea57525e87a121d54b49ecc6d26 +15d49af2d10c487c9d517f68b5bdb0ead1d692c1839e7292c47d9c6d61893dcf +aa55d2da04ca33101ce4cede60d9bfd5a346e0f561f146fa0150056e69118fd9 +899a60eb9fe3eebedc12dda882d03b61dfa8baaae45902433f3eeefb69a5c708 +3de020a13bdb46b349eb1913f00268d85c3283eba8c7989be1d73286d1290611 +ab783bbf9f00e2bd402a52dadb00a06710fffcb8740d4a822c52070292026cfc +6b3c735b23898912c4df1fc5d4bdfb50f08519205bab10fea62fdabde57750fd +f78c827099a5ebc612c5961ea844a23193229463ee792212a836585e28872856 +b935f6a4cc18eeedefb78d3498ec8b4dc6bb61d038b9df282686d09f8aac7d53 +6df6c2d7ce660939b23a921f29a4f7077ff9fb998bb62c1e5e66ab125ef375e3 +80a119c5c30eeec46ce473e27e1a24f0541e80bfef5b38ae7c2d0b3526f11abc +832386f97e25dabb6d6bf0c8020c6572407d824b11f9d089037491515b50d46c +846123cb02f04ff614664339caac3d2e0a0b1f0bd090983cfdf60e302f005af1 +81fced3ba74758ceb872813e94ce557ffe9cb2bc61fe4d18b16c8f40f68d4e70 +793734e5e7974cec5ea1753af5a9471b67edf56bfd97748c28ace6392238d0e1 +82165279ff017d9ab50e7aba8aa5b6e7c40dce4c88d69dcaa7cecf50ab313606 +7a8e0f41d6e6dda7033f8a8e456df113d2eba6bc790742cbf67009eecb0eb580 +d62de862d1945bc26b1ab03a52f824935ff56f1cbf42a576cd30a1bfc56e1dfe +3114959a1e65e7740c3f3df693b81912601eb82926c0176020bc659edf9b7760 +1d9e3caa9bcda611782372d0186a69bc20759f65a10e6566cb79a375bd5343de +178e2d42ee98128abdb6e6b9a6590167076dfd05d333aa354e612f06170664d5 +bf9c272e608b3d9967c7aca7407286ccb187f774e36521fa684b9ce66d43bf60 +65e2138c619783e9887eb8693da7c354792785fa7cb7f80a30c7a427eb01ab2e +5cd18385589d5ade763ad4f72b942d08028294b5731dc27bb88c983f93e8c652 +c570b306209c8311893b1d3de24c41637335a5bea007c5b2c5a7d072c82637e3 +a394a3dc3d212253a75c8d1e5ba23ac91a5aa5597c979208da5f046019cfe93a +29aabdcec34d19c0ad54a62c32dea3df1d2c30cce5161abfdf77ad04333414fb +72cc00ce506867a2515a279fc5dbfd97dd3654261ba7723d36997e5637cf8d8f +05d6ba704be1c15b03f503c341076c384605bc14eedaaa7c65c949aeaf9766d3 +e809ced51d8913022e7d949ab42dced250e121e309b118320dc3d2b46b210c43 +a1e6f8a6da1a9a63eeace306f5f25fb32dac58301afc7908dc98236618396579 +e420bf6566ddbe45a6a56bb68ff442bd0a5fa966789883d049c9f6ef57a38d44 +9b052e4d7df178aa3a5ecc063f6c0bf1fb3214a745d0be4cd9854309ff84ad79 +83899a6e4500677dfbf1c727fcc538bb2760c326e417e82e7b55c03065dec520 +8b9bfd1407a6648fb1e213157458e1444bbd0d9663515d543409ada3cf276619 +f6e99477cc73a2ead882c02771a456ea6e96f72f5ece5ea165cd43b6eae7985c +1f5743b37436f40b8bb9d9620549f468e45fd1dc481c60647f3f2f01055da0a4 +e20293078d5f784ac7d24532097b9d39b5ed1ceb0ffd72f0b5bcd44136bd981c +23c9584bc4430d83f740b5f991f8bc91e46b7e334aa1885b57a898431fc70dde +dc517082db819cb62317c704efd749d6c31e3c98271d12d3ed7fc363027170f4 +fdb202022b7703200883df3686c970735d5431366bb080f897b6ee064f57d62e +66a284eb6a431b4d9686191c41ebbddd452b97ed6dc58e3dbf726d748f10efc1 +21bcd9dfc8feef98ca2aa9306fbdfd8bed0f8c2da3d6b04dce3463af1eb676be +11dad124ab5bdf74bbbf653bdf0e1286ff715dc0fda04d4e4078b1693593ea62 +feb3a69ce75b51da3a120fbb66e6255e9cd97520defabb79c1b812b3374a9fae +5ece3889c1753afa6f8949ba0d2959d5c07c2c0d50e7d30c91776739093ff94a +83e21a4166e71ac1eca43d29f4aa850416dc8462dfa808c7ce1e3affd5a46cbc +1ab4803a92eca1586329084df13a0e7876a8e144afd76951ae209222ea41c2ca +ac2507c9cc2b5f3201f09b1f757fce7fea029311b620ba116adb7f7527b2dbb3 +6cfb3cfa41d3ed8aef017355f256baa5f1d25b29d4fa159ad1e7989c07248a62 +1569eb5661775381952b58d67b6aa7e4ce17c322849638b12fdf1b66af8a9d5a +696fd560f01fd40d6e9f41f3662ded09af9967c6f84c228fb5ab4677d1d0d37e +a6177179254b203b5b3d8097a222c673ab8e885df8e7050e6dbc30ef5f5bbf17 +cfe32f794a75d1ac4f1e68c1e01892c56b76a3a630fc1ebea2835f4c93102b86 +0c1493482f3199c044f0a9f8ebb83a262544df349809d3aef5c0f19254235fc7 +712afc0cf65e12be4da48dcfabd800d34d13704b5dad99256ba184dddb70583e +e88149bef1cf7f539db9266c2329f1469f128823e45fbc33f2ad7210d3a7203b +485a1e59e27585d965346c2a1e6cb67f31f6d882a64d5723cb920efc78bab1f1 +a044b5bba2c396c527007c23ddd46980025378ef872065d051f7acfec1966177 +26532f71cc42053fdd39ca16ae2e3aaed9689dd41613902377dd2ac403a793f2 +03f3b3550d100d8fb265d7c553c16b3699aabb5628bf0cfacb2608847304bc6f +ce9bd96b1535e791f3e2d73d949e46b8e08364832cf075643ce07232beb939eb +65d9617ec0b5b5bc4471410a7afe2b38bdcea5464fb611ef671389f7d7a0622b +2578d2daa8e07914e47f852b4873804db065e98f57d64a2bdc4774cf4ecaddbd +bf5cd21dcc7308ae1e508ac9edc52491017e30c8abcff7a266f4684b18555dfa +2b134c99cd4e74b86aca4fcf3af32940f4060a106d9b750634d58a6643d70d8f +4ec3b5e65ce6179b75c7b3f84d3cd899a3a092ea36f1255340228a61ef2c88c5 +175474b6b1bbc93c4f85bfd7a30040fe2557b2c30ac7c99abbbd9d416defc5bf +04b98ccd521cd715ddef40d5d75d46a3c7ce6416be6aa61b8e51fd6374dd9b7b +4f4057f3593f2c693402d0dbda82b71f4b6db7658dcd2e9b8dc16b3f71598fad +ec456aadeb12bb8d7d55f40536c4bc4ee435cf09c763a31dda594c964885f7f9 +3d29c741dc75924bd5cce6b3854fde27419fb6b3cfb675e2b18d08b0013508a0 +53d950890c4244e8dbd82336afe012eefde038428156d193d504e2630a5f61a9 +07426656b6e9e984f955e4c8bcabb541da5b4c0e6c98c7a644f39d5fa12576d3 +720d7a60dc73b27fa50203fb5a8bf9c2439499439327da2ba21b198611e50cda +08ef1e78a4095367e127c7636e39b18f5902d90b4ab3eedfd32ed2d1093726ba +9677a00f7ec11f94da6e891bcda249b7cdfa9fbaeaef2122ada7d54bc695991c +6426a9f053bb1c2305f3af532a587297bee5d7a985462664853928f0b4c9806a +f4bb02bca116967c6bf3f16ed0b366cc36c1159a75eea2f5b81c19e411f59904 +290d50f8df1565f71fdcd67f62041e27294ca38a4d9342cc160c20b430fe72be +83c58937d81962c11ef39d84386438a80ec20b6f40eda53486f56c9c9a451d35 +f0c4d4594c70cfe64e2a0f3d8ccd8140799ae1405bcd9e9319943021f132ce4b +bee8ad0d0f3b3c739137f8d5d819bc30696fcf4afe0b7721547f7abe44b1fcf6 +bdd2ae2a5c3a48fb8bfc49abd5825e3779521a8ae5640cbad0e021ce8db9d09f +ca128a4a25314fca484d7da59b72bc33c28f746c16f7b1c61bbb5462b7184048 +aaaa34fbbce3383dcf4ea32475a0e0a32774ffc724e840190db16bf190c34be1 +29364198c66e564a01e3f0a4a9fa5fd63ed5a483fddca42ca07b7686af4f00c7 +517c2f3b6804bdb7cac012e8e2845003fca081c285ca1d5b68e9300cb5383999 +e823d854e6e7e4d7484d245f75936d07b021e58728a347b8f7294f6010083890 +234cfebabdf4d86761197ba3fefe7c9c398a9769db9b4f3fe494c9c40bb52aa8 +7aea63730475f78c5fd7ac7024422bceedc91000eba7f19e1c65aaca1b6e2254 +58a5e69fa928a295fdafdc33012e15d495a2e59899d9d203a79e226959a79e04 +5cd83673e485e607009d180f6870699cfcdd2903cc4e843041b1154f15a82629 +15d1a6b53368bab3265bb436990240ad9ca9638aafae6b565fb3ebbfb3937fd2 +9cee2d55ea102faecae5a88a54124cdc8ef445bf1ca81c5dfd3be73653269952 +3f59aaa9a6143e337ba0b40a7bd4c61e20e3b6d7e951c16dbb31312d51b604f3 +349cb04b0334d7e6356f685332151465992a23216a7d9a1723590889d379ea47 +7aed9dfc3e59496f50b20db4e2712c792af9a9ca381498d548e25e759fe82fcf +3013ad46a372bec10d3fea3b36ae5e50d717022e742a0fb0cafcae5f09384d03 +4cd600a052071b7332fe6e1128ae57cc26790380e0d608e67e55e0efc0ef0857 +6dff74f3fadfc512a89b2af803d782eca182bacf855f81437ea7288a0d2ed57e +3ab437c68d324d95c9c22337dc54e44c35f7076ce3bce2b7878cd339a60aa94f +e5c0a41f3cd0a84ffb9453018fa0fb4f97b730bb6c59ff330739f171810e75eb +bd287d8d771a29de860c1b74fa7fa9bd16cc4a2b410e65cb64a903eb3fee048f +6a5c4d0ab45a1aca80589f36a7238af0b0e2719f384bdbd75779b88b747ad04a +227f7c47d230f868317776ef6ca6b1eddd8315d61004824258e750d0513d5f6c +a3adbead210a6d2fa560f441354f09c71f695e792604b59127ad3af53b2a12bf +fd666abae3f3cd985bade5b089391c6547d6604eff3bfc779c03cea5dfb7f688 +41697ee8c0c57ad63c51c1aae8ef5cc81ddaab3708d95fc45b872ad35c26edfa +198fe38e4de28af59069a6d6df1543487694e5c7075c7bcab9534394e6d44160 +893662124d62666f32c6bb9fbfd87d041f0524cf21da856d72868c11d9d657d1 +c781f8cd7dcfd3b66f24b2c21531a19e9fd602647ca60fe57bc0eb6715051305 +937a145f0304716cffa2bba32ffa6cb5bedd36de1c6c03a50a21cdef5ee9e853 +5e18ae53b3f462710671bbda695561fac63793ba0d9d85ca1447550747d7d0c1 +abc3ce9b610e4c2bdf1c41706366e6c84eac7dab6657199f85315173dbf64e89 +601b9ddeee1f4abc1d0544b82cb4a19b1bf75111b1ef4b9455b90e0f970b148f +0c1899b4e5cd4ab52aa5c88aa06cfc9e2cd252e7708e41f47f1f1953d6a4d977 +472b247652475d72a024277d3ce24239f0807fb2bdeba74ee6a4892b56ff4a1d +b11856f44f96451067583d9429f2f675eb2439c8073722b315232e1d1c3d47a5 +b5de438a2bd5f16f8f7e5df0d64b4004f5cc96060ce6a6304c607b07c8964642 +c1c0e93e54663673f53a4e774d8d6b6d79199d1300731ee2250fe6cdac5cb368 +7dd6538900cfa180554f65c39b920fee6ce6b09ef27351ef30d0855833acf48d +905beea766571e433d5b41512ba86c11c5e504a430be567842ea1e974c12f6e5 +bb53327ec03747acd863c89f4ec22131718a1591fc5af9f9030a0bda8ff2e033 +897967a478729fd11f19fe5d8301dfb66eb32bb8687c13afd7c13a78f23a5531 +ff0539ac4defb4eb74e8da337dad1d7162668b08b01862e64d16ba6b79e4142b +c4c8c4b4e2c79189654ff9f7d88393e5814c09f19140f04e55b1f9dfb8148d59 +e1b77c220a5d6ac006f83ad892e476a27b3bbf4b6f32612d6d20261f3e11d4f4 +87ee281e2cf217a72c31ef2f075bb0cc7cf483af12f4face8a9b663ef0747aa8 +5bf51ea0e46a91a5c3efa666ae144bd7353aa755942459a48769574d3a2c83f4 +99ce519cfbe39d9910bac61c05b13d3b42f8918905df146cc2a6700d528a5b4f +faba3f4ba4e7416f035bb807e50873cd8c510d2d001ca449d5a7586e8acf36b2 +6b1a7204b857bb6070bb8f65d70494af8b5f99275205a876011b9ec202d57cae +289a6c36994466d9800971f5797735a9ad4355404f255bfe479309226cefe2d5 +abe02d0f1a34494af55d5b6fa2f159deb7d69041e3cf96138c99531d7321a8be +f03418fff3b22bde2f6c64af1c0734ca3eebc2d1b2ce0743be2a8cea9a0f1849 +1deeb28dd349ffa2560a82ea3d682be183af204e8683bcaca175414896d9bcea +9e9dbeef98833556ca38c34d5d29f94d160e893a0bdf8a379369b239ec918933 +14e3dba3758c1ce86a93f7b93a15f5920f3a88e34776609857ca3c1068ca24ae +aafa88e7f286c104e41f086a41692e56d4ee84bfd088c3a943249daa3b2dc1dd +4ef1b6ef735dcb41721edcebe5645d4c926af523489692b7b261310a677ea04c +b6473bc360635211cda8177545c9a80a05da57bdef1e43cbcf1bb10ad463231c +ceb716d7192820552381885584f494c4665a70985a5ae6d96a316b37e60bfe33 +6cc6fe5c12bcab2c150560fb5e1f0d1f2c8fc80dfacb213231680f101762a84f +f5b639eb8813deed223214a7b8bd8184f020e50e6c83e74fc6f1eb8e6eb59745 +319c346f1375204e421bff10a2de0d37adee62634b2fa6dc046714f14c08260f +db956d6f8d75e426cc1448cacdd556af8d1e1a490e5abdf326dcacdbc3c47755 +b35174260312d791235b829dc13e9ced9f5255a9a8b4748e68d8d46340a11a9f +2cc32cc26d079899052956e200f521a5a9e911f68717fc44fc21012bee6ba976 +6e80c831fb835d9e7de258a3418de08ac8006bfe1b4c8c8c6eb8a4a3220659c9 +26e1e005b59d0999efd04218ff177abfec30a01eb6af567ba7f4fd8ed13b819c +b54f35f180e7b49c77e1f620460a2f69742af2113cba7536dbb4bd9e72b076a6 +0bfde97052ae7f8bf710d2d93f6e2f7c4312014d0b159c6f9afc7f65cf34c664 +3f7ab26ffa64ec0734da75276bac98d6eef8218c3c21766a21af0515e1ddf7d1 +dc5ab4ea3cf0dc8b9d4f0e5a9d72eea4c94e4296162141cd74d1e4f38673624a +b0f09ad016f16bb6c189339795f62a150aa329e1cffb50314264264d9e9c6e4f +4cdb46ec69c6cb00cf418bb27b4ab0922f4cf707f00898edd8817dac625a0669 +6c5ccb215cada17f9c38a5b55b7bdbaf2b2cecafb51b788e90bdb7acd2427483 +75fbc30212bc16de3550a681fdaf75dc602840746a5dd76ceec9e3804907bd01 +98b3e5f730dbb19c6d3f0370df3486127a14dee10a4e4e37453176b8d1f6569d +434c39f79095462aa2d21850625189af5902fedb9ec65beae14fa89cde1a959a +f1dd35ab80bdec905888defc11730bdf87e8da0a244fc004f700a47d1f683f39 +117aaf616ad1547be97c088bff1fe0cf701a76b5576ab1a5064dac4bddf0ea24 +c61011693530095d135921adfc03ea8bf166a4861a4a3b69f02d9b4bade06f0f +430ab6a8fb6781286c80dd543519aa3a3991aca37564ab3c7fb8260823dbcd0c +4ac190b404b4d841049551c0f10e5f3a00cd5a9b10c4b2f2ca9808dfa50bbe7b +104a014e4049e1eff26ce38f890f2b8b0f49ebe249d28ebbde3a96da3d2fcb71 +e59470cf6ca60d8205310cbd27d6bc278d15957985b753bfa89d78ddad2cf5fe +16aaeb7ec76c76b97cf688d1b2294d800019c50f0a2ccc17f23c4159c8f70aef +546306d2bc307645534e58e2364dcd5bf342084ce38366b961ecaba61c5cb02b +f52f6fdfc0b7cc8c49e9ba70889a7c45f64cc9ed1fe3b978f6cb6a7d72c3c3c5 +93e9bad50782458a1581abd5b4e3a70be5d23ab7927dfb5a5104e40a4aef703e +5ebbd1965e6100d31529e1511a93f04fc72f02e85f5fcf104245ac4dffd9ce38 +08c78f50113ca5a4629203104541dfc746a0f4011e5dcafce88348c945374425 +a0b90885689dd8aeea0979ea1ecfc36d8c3a4bc43894e40d3992ccdebf9216f3 +2724a1b44749938f9af004e45680a2466833190ccf0c28e3eaf2611f08d4277e +3194b8eb6b2107062564ca40873f311d68119f75edd6b933e77c1f90bca84ee9 +15ac79322fa796009cd5f393d5bab90805854d9fb3276c67fcc9449dd454a277 +6c6e704aad88ad3eeb78332fafe378d8feb7174549eb23e2ec2232406e71b447 +8230a1fd29e3dc4ae4891e3be52206c7a55da54b44ad8ae6f6234b45ce37f753 +47a157bc3749e388fd638575c3d7fc0346a11de33f5503a1398fcfe9a9326fac +414ecc95ed3912528755c64943b7e6961b6024f22ab322d25eec64687bff8c4f +1ba62612eff3453fba41013c31bd4376ca3c34d021b4bd2939dd432dde6251b1 +af264d1b737eb930b641a5a92e568c22a30efb18d2055ba9b67f026873edc480 +bab7e3d687e5410bdf48892c4766ecc02e093363608c7a2eddfa02f45cb863f5 +7e981ee9ad6af81b6b893151cf26abc3c94d1720a01183b9254c6b592dcefb36 +dfc4c728a6708d3e19ba894b79271ba453b7eb48fdd21d8f33ee224970b7e647 +93eeb04421798c8af37e8315e4b610eee2161884facacdc79fdd0582c1dea0f0 +07d94cdc2da4f04cabe5eda0197b7371599dc4fec2be2033f89ffd4ce38ec8c6 +4a582ac99ecfffe714896c25d87bf5b46b492bbeef45806d4907d21189eb3660 +e2d41b1417544f35d823bdaceac0674dae5111d9fc45f13cf495cbdd2e8c0d13 +b432754b67b15f67efaab14c6297822427bde405bd9337e3aa1e9d028f42e47c +d92ce68d69df3e594c312c016fc4ba75001e137cb99304da1ab15d3ae09d8dc0 +a66a23b610e7e51079f6d9f522ec9f6df279e99bcdc922a55ca043828edb9b0a +bf925bbd3b48ecff6a2c837288fd1b9ce5849426c76dce143078a2ab249d1995 +445bac364df75a38b2d7f1eca270603c6bb6fba3a860a0749bf1fbd6a68f9245 +c0576a5b7ab4de2b02a0745dae9c78842d11d644ebc1bfbf7b34528b063209e5 +b1eca3f6ac95aacc306520efdee1b71902d7d2d2d16f6e999570d2415d63cd82 +a7da5c9f86c0e0580e691c0e8a754e25180ddea243c4253f90e6d2dc28d39fe9 +207dd5b925fd3bf40b54a38aafeb6ff57694c739e0a6186256c171eeeb680f70 +a6145f6b2badaeea9b7ffd053131ca0a8375856abdfdce0d4fb29f357d8b78ce +fd36ee59e1ce3b900bba48f18c9f0f0c480c12c4bfc1da1eec771c60d54210bd +bc52886407409c20cda9615d316e7bfc1d055bad292fc1b044a9045667651d90 +97aaa42dc27ee11d9928d800f9cb5e94fe1238d98fb2fa4d3c3312cfa79ececd +67f75a12b2c2b1cb3f667d22f93253005f739b82e08508c4b7d9f25ea4eeea8b +e061b6323b310936e8406df50677edffc087f3252504ade98e299bc42d55db8d +c0102f56e43338f66dadda74c977445e9ecd7354950973280bdd8f7c41431fc1 +cf7187f109d9a7508d791ea946555b9bc14b235908642a522ebdb1da88301646 +2f6f5c0e62d7d588185b138d243e1f9a4b527cd962b2f1f1bfc9c52b5ae0fe10 +5515285ee161a6d0e92b746fdb4a0d3d8727045e79e36fa76f20e0ffc132d987 +3559ae081d4927a809fc10cc833617dcdff12222a06f8fcf3e280a0769a6405d +d856eab2c8d9e123b115fca70d78facc68c789e9f5acbaff251459b967f051d9 +b93ccefc04a653af10abe9bec36c0b466ee6207beaf6fc0ea6cfdffcdd87c8aa +46214fdef4207c98b4c38a99f64ecc5f1a080643c01e264276cfd5b16876a43b +f91b6e0d6c8de00d3375bace650f7254a31ee55d90aa68a4615d41e2b0b84933 +faeecb2339b5b4c93c207953d6e6e9bf519c78e0b618b9cb08474645dbad4b44 +343e5d6656fd67b87977efa9619e5ccd8a3624a9bf6fb1d7b211ab3f53265a0f +9583efd358b4181a3edc971167a097d72b61f7656ccb810623f23b6b88bbbadb +5a92c524070ddea51b1f4d4dd01b2570b2cfbf80746fdfbedbc91da79d697f2b +6b575ca6c2ea30a60e647720efa8ba7e01498514c8785cf2894f363d3cd40fbc +e6278fae46b87933a015183d6a9f7375e79c3c9d53f507edf53be779bdc0cdd2 +5645311f22f1ac98df3acafdde5881f52236e8b07bd2c55975e76b588e5ad807 +8a9f5238a30730e1f48702a4b796cac78c54ab0cf9ed0b60fd0fe46fe84899e1 +bd90c7210bb44b61dab5bfd0241f5b52eccac77577a94f9871f500f3c415605a +2d3c5f724205b28e40ba20266c431e6a899041ceea1ef6532b4a25b2a8e241a6 +89edfd3453a10f6ce442b656437923479ca3b35b820f4ecac1c1b6e82cc93b5a +9bdb7ee0aed097a8b93e25b242c331a9d2a2e6e9d6c906601cdd93829dbb73ad +b2bf524e651095a0c04adef422b2d979a079dc64a45b9e538679e2fb06d516a0 +debf06297e7556e7bfec6250b4487d29c7f61e60499a72551b5817c836c5c025 +0d48fd231ff39aa32ce44e5b27f306f04d5100c844ccd43911159ba3caa7512b +15955d365a90bb37480b64b5229f97d71d6a6a040e782ac55f05f7e5ad280ad7 +75894976e962181a200c54977d41d71c5975c10de57834b38accbeff3be77663 +211a931398dcd6dbc471ac35aa741b70563110c01cf56dd68bba9fc79d4cece3 +bfec2adf933664b3c5a8e0cc35ce60516063bc76062a70137e6c8c22486e27ff +824c9b33d01799162eb0f5238337ccb6ef32a4a6999182c37316f4f29dfe5890 +7e613d37f7b6279739020b93ed9bbca3b5e8846d7338ec21fbdf301448b54b18 +301499c7699a782be3de7e139276cb43330d5a7a81e0346607cbdb171c828c80 +6f4da6f46fe3eacd86f19b981cdab577dc969cc9cd695c3cb9be7539d7604133 +797306d7b19b01ff951f5be13c88b2a25ac91fdc4e8f5b41453732edfa6b841f +30d2f111fa0c7156fcd876a1f35b973dfcb430d4a6880ab2d36e022d036681eb +1d15c5d0d8cb8c943e1f5938731b0639177a7735080ed3127896508b39343446 +4bbe7a8d90616c41ee1b6a228a63988058787a283a787b6b8bd7b831e668c9ed +47eca767ef1806a13b3160dd02a4eab001b21b51b32bd7dc9b49384f168f9354 +edb1f2ee720cf5e8ad49c8e980f82aa291e60ce2f0796a2f9362baad3dc05ab7 +96047183f58322c63eef4f9568a9dd301e73ce14b879f2470d28e83298db2dda +e507c3d2a935b33265e3c93b1801b3189c2d6105fc81993815f137743616acf4 +0990b528af48999f36843e379187dcc6a9b1dd19bb0cf93d8c739bc9aef003e6 +26eee3889d28f7c059fcc72286d576c4a578ecddba3215e09b793728d0809e16 +e4d5068088736e7997c79ac0e6e7db42193f6a5cb890bf09ee770f87413a4106 +65b8d22960e35b4a7282d4d36fb546c16da9504447d91edd2f2d198b3a00a6c9 +ad6536d46446803ff45ebfb20101d3a0b92f870e13678c403a70d630ebacc963 +a124ed903969015f188a871a228ff33fea851f47d9438103548b75e29476de4d +168c200d11549d5dc869e4d6ae1776df9b0c826d7a8c91fa398a46c5c4751230 +30a42d5139de6fba2ae666e9a20c65db45acc68326e67bae3e1271d3184818eb +c8dd475b5dcd5af984b2230be7503a49f7be4a8fda78de538be137db8d9d14bb +e6dadf2527a69744447b12927ca0a3161e819398a7bfa5f08bf3ed20c0b6d014 +4b9a0a7685f4f86ab6dd1a837f082ee7522ec95752908a47245b19bfa040a60e +f09360d3782e0deb53ebcaa8532fa1db2de664c7e2476176fc40bce84481a8c6 +bd22a4f9daff3309dc6d29574767ed3bb9c32d6b9781758a7de3a82e2bf88f80 +fda214eb3db44178f59ce142b6d28aaffb02e7dd474c816acac16e398458a70f +70ea22e2818b39b81ded78b5a89750ff1f7bbe58b12ed762745d45e6b51f684c +0d3ad95c9fbad1d968e62976bafe29bf372a13c82e532bdc9285de7c57f92824 +905c1f304ace286b3eba316810cd3e54ed43aca9162d8ab1749f15fd6305edcc +c41c2a03b284543a54b570bf9569b6900306586a300dcb937924c6c967603798 +9819bf168db06e1a166be928b0fc4c567a6b683558731764a524d25bcb38c1ea +38bd56f192cf929341e20be9bed44c48eca5f8aaeaafea93364bc09a437924d3 +32103cd2adb186b1334ec133321a219006fe06b72ecf0668cbcc93dbbe8b1dd9 +b629fc203878de20b184a6036045ec7872cda4859638c7410e0f758769fa1d61 +097df028f48a7b2f897296385a4c4c84c81c3a33903d4b65058ddf777af805fb +a09c50fde8178aca4bf3065baf8bfe56e7c8b9eebc12458ab6fd3b17e00c56eb +e2864c178680e26ff4467cf616436942b070fca737d344a6b2adf017319582e4 +5d00dd5202fdc7917ffe5a96f0e73b457f01356c051703b25b7f772d4a127754 +01ad7dc7dc1ca50cd21809e59e527dc03c06d77585d99d8f71d0e270859c268d +ef2593ae213d1310e52083d13e59b08fbf57536e512d2bfe378eb4c5695b982d +e7213208762ad15b3d801bb0027c01fcbcb0ae0cd083d2b364f44cbb64b7bc0e +b9ea7324bf46de83ff81f0876dd8e6e8485318165cf9c5073549dd8824c7c2d3 +7e67c6d922bbdbd0b7c9f3ca720dfb0ecf19d77e649c14824159d855ae13de59 +8e9f4091738d5be34fa6d24088859136ab8dd860fee9f329e40d86f2470ee608 +c42a6175ee0cf75e5a3d5e9c6d5a3570f184cd6e09b181460883da288d690a79 +d7ec092d39b738607f1ccc29160e33885b004eab002ce518930ca4f9bc2dc74e +0282c1df92347a6397870d0997f78e7cb6045ce4d692fc7fb73ccdde1ee2a76f +90b928d99accbd618929b56ef904478608ea4edfbe39d6d2b66855dc5d52277d +b0d76684b2bd7f8d1e901e8b4e451f90e2e57bd56b129db14fe6121938176381 +79da1786f50eb8fc41670def568ba1f946e25a4629a1d5717954681fe914d0f6 +63a54c3753afe149f27df7fab1d48d647fdbb687014902c207881bf01c008e11 +df9ac28fd4ffceb40305617dacb13a58e45a20d4cce72ae242873efcf4fb3922 +b795e0774b135700f245f165806180602165188ee68b15054892c60c5c31f555 +c3417d0f4ab3a8b46641111a842b9eef2fef3df9ce6eb126eb1285502a7a8487 +7ccd9daccf569a6016173298dfcf6e74a16b169b68599a853f19e516139daafd +bc265c40a3fdf4db0795ac17646e9119f262ba5ac45d2ef0b1fa3d723070606a +baa024b1df51e31141c2906e082b5a1bb04e1f4353ce4ffdaf7d106bca0c0dfc +d864bb808bace8cedc58f936ce4c15e7959cb2bff5d56fafab420eac83401ea8 +01621278091be7d31c85b5bb3856a89644f52a6df9d7c2b258c2b71fc718cba1 +cb1094b3c10ecf5e39ffed13e6cb3207ccfa19306ab09205fdb2902cbf534ca3 +a8956e278741124dafe3bbc80935050f96dde750a99fac0df2f3f6797b704f78 +682bae293ba9e865e42186455b07c1f9000480cf4f83e9a0639c4327bd557446 +05a7ec394fd6145ab02fc44137462df9accd0ea88526914288d3ceeea5b710f5 +1088dd5677b8658a313c3b4f9dec552cc1e08ba36260de456c4962fb201e16fc +27cc3035e74126bd5762c295a2cb384bf572d296e6936d3941e58e70abfae223 +6f46cda8672a7d660e2f8cec69ea0c344a2e80e9b1dc1520753dd5b248f5e561 +062e970a200653a49b73447a30badc489dd0d54996273222a8ac611089ee1553 +9742a3b2264a7debe84eb60e2ea796b3a19518e7f8d2a5f57389aeb9ff12a855 +93bba324688af65b727f0c9c08f0fb45e2a2389b269939b1414f4818b071c598 +97bb4acc6c2e9f88b2043bbe547f8178def40a4d931122f9f4ddaee1e45f47b3 +1050e3a2d559a9ed57292918255d3e8e5bde36210e524f5fb6bb627246e59c82 +3b363c755d5101f861b8e980b36e5c70a6b4f9329b4132df65e3b72b37ad128f +33208adce88477a8d40953b322d9079be118dabb96e6bd913b36da29657e8eeb +1469a868f957169e33d763d509c6f147af3973927f940f1db81fc54e48a43c56 +d5db2b239dec570de1be8c140aa7d143139203f37030623b4d73df9f0e8e113c +627320e8ddd3f373041722d98af6732ff47cbe0ba9e65c0994128f4f2885d45a +6324bd7b70a0b6b552c17aa397297d5f4ea3cd2765aa476eab903ff02f53ee1f +65922ca1f6ba733cb240e6113e5896d164119b631e6fa9ff0f9759d55043b234 +39cf67812ec7a1101581c280442e72e0d2fffc9d0192e45e24938f8941024d9d +722fcd3407ad005077f53c3a2e5817050839f92bdc6d6a937c6c15643a502bde +491991bde2e974f4894470188e0743448be51f87a0b4566010c859ed57807744 +7a22eb14427e9ea2e1bd93f49084f9895781f066c387de1495feb2844a4287cc +c58e7945c47e03c5c856e2c33c2dbc41e748e9b8fbba53cf187caf375d5e4dfe +69745335c4ba3b4c8b52f2dd68f4a1297cbd787e0e76f1993aca6e5003384a56 +22f2e25cf8776a97a16851277bce97278dfbccad6c28df7cabc3cf8c89bcc3c6 +cb4d13f1974f1d29b917136fc496acb3902f632dd88f8e53f1f5d3e18d220e56 +e8e33d94345512e41a0ea57dc70e66d50afe7740f859e7b0814eaa5d8b050145 +ff57e13ab35ad743d9c3bb4b7a08e5680309fe1e345eb8088c03e102b8374ecf +0e8dfbe308483b9b6397f3e773313be0dfd1533a2fa613bc09af5aaf872fa816 +cddf84e3199014fbad13a3c1c06c3901934ada1afd5c1640bea40d5185e24bc8 +b09e35107e2ba0e49bec515675b144d8870aa7f3e304384ef1356d0aadba40d0 +6863d02cfa5ec2a3ea945f2c5d6dc3bf99e96e9f75ad5e2766409487214d49a9 +583093930eaeaf2119529d11da64a39099ba6d031fc7ad49ffd7418b10a76137 +1534f102dcaceeb0f4f988b5c43dcece858c159acec8234b7477ac696fa135b0 +512a462521cf2221911e06dd58fe975ff1511460cf4d14d441e9a40b4cc203fe +762b6ee85b45ae0656e5c793a74fb582b1c9846a561045dcf472cf679f5f0422 +5f3491ff9569af2f43ab0459b30be6ea7849400527841350ded83f9d1240c7cc +caff29e3a569914c4a73374a55922380f01c138fc78b3fdbc74d85939dd2675d +3a310f7008ebf59cf3f44eabb7cfbae1e71aa2bd2b9b0f2ecd956ece2022ef00 +1703d4da8fdcd31927d898c9f4e32a323f1388f4b12efd043b1f1f56da28c850 +9e7d2044fee9017dedec80a1c29b5c222132fb2e7c90cf783c1a50507983dfbf +95ccbece717ba641c01be95dc3fd6bea33e4d25a61980a03d1da84d9031fc90e +29206b40de3c7f641971509f8581c80d426b450e80792deb110490d3c047a405 +382be757fa40774ff39680eedb1b520833bfbcc779896eae4e2e5bf892451e03 +32a76a70b9fd923d295e5d4a9449d53f1477e2abc4fb3e522d9ce478df66823c +041793f35ee68f0f4e3c381bed41631b9941cf6d1cc01edd5b3e211d0a3104b8 +baf642388f2d53c13cf59175d81ecb7a165bf15feff4ca17868bec727dd04fb6 +f1159e6bf0904cc092d0ddf098dce395ef760f3ccd8a41c475d8ec42080fdbc8 +61c0099b6d5f8911122dba3603ca631cf9358c991cf3eae1f6e494b01aa3a85a +9dc6882e862362bdeeef60fe5dbfd20ec25de0ee273ca77739697273751680fd +1106850d35ef8b6cddf6d95759462eb0ac8442d2b3cf104eebb1956773978005 +8856c8742cb8f53f03bba5cfdbf6a0a871a6d5e5608d73a47aad69ec45ba1631 +edbba55a9aac364dcfd6c8ef59b7e908b4c6d28c9a3d0ce7b65d2dad9edb9f3b +f32b5e08ed44312dd4dc87eebc7b57f1fd43cd74aeddee5424fda8caa4a1b072 +f13eb362c038ee2cd7ec96e50f97ca9a948d087b69fe9da064d2aa519889e976 +0c95e6180f3d708396e8ae45246fe1e6dc0a3dc12a63160ef2f8e526b759e88e +f312f724a6ff2263675e5e81b724e64f29f789192dd46ba0a1c7a97cc1e691d8 +742b962caaa94390863397a28200251a5a2cba4c6baeb029c9457adf3fac47c3 +c175f2bf219fddfab264ae8edfb28944051b2c0c6f8703db56477d54de00933e +658f5832cb37d4ecf4a99ed412cc51cfda06a6a20f75acf5c8042f34471e42aa +7d1e92c898a9b7048b451cb1fdd7b9be6c3d81f9f2845a01e45214752fb41865 +0a3d459d2340c4e33b15fe951f8e4807e7ebeb835dc7c8b9b641690a9bb7dede +f2ac619424dac857cccd16688d6cd2b752e43b65847cca0f8deecc7ef70e6773 +e535a478823cab214e6bdf50984e6fdfc7f21e7e15426314bdef6d469904c856 +02f6cecbfa506f6c5f2e2822e9ae59e2bef44d5e3da9b708c2085932322f1680 +0cd9e6d7361a77c8a8a1efd75a42ab76c93943081f7e0409bba3f035d44b0d0f +348daf413764a22a22c5a9338181784ca6498a40662b273abdea31683c3fd096 +1c8d4957ef656ccad323563daec41544fe6264a1bda714580f3a9d7ba97d21e0 +3511de76972a27609eb0f7099277ff2b8000edd74962a1469eb0add2705665c0 +4489e42172f2bee0a973d59f82fc594f86995028a211f605794568dd66c2c45f +27b75409f34609b4ad986d7c762eafde72f2bcab268891c78064f770286f716d +465eeb4ce45d24c2fcbe8db96af45f65c062efb712eaae31ffb7c8eb5ac968f6 +844f3b206a00629e64993e9d67c0e7b4bba364a8c2646b0f0cfef92e692c38df +cc183b54d325a7919c6d08cfaa203a259b4a691d8d01c94c97b44358f040cf9e +6669b64ddef2594a23dadaa795325938fc48c57a016cfe2a2422fd3e68ef35be +ac8c3b473609bcec27ac20edc391565ee43ecf6da585de417a55f4bdedff6c1e +90ac90affb81cd7d368cc6425498d4e235dcd925f9b56bb04959b2b3f4a9dfdb +bbe05c6d996b1eb7788dc1374e9291571e5a260c2811ee701362b5a069b5fc96 +a2c0d000b692661825fc5675ecb1d21129e48507d85e9d77b78b6570609c13f5 +b2c1d40662356cfc4ce5c6d3d5db286d631b4d6b7e140c25990b364b89ef331b +c968523bae049ccbb1107ed7ec99bd0974581602998ce7f099ea1ab7eb15b0a5 +84a1b53d4e73c607e76165ad9a4d6d1d36729e83ed798dd1a2f7a0e70d53ae3b +6e4b5031cecf29745db54b4719a1a627046b28164edaf8571e722f62c649f111 +1a73b1a7d94b3e65292bb543e9b89889f8113646152219b2d096809a87fc86bc +bf471eaac162f0bf317235a006d4126dc5ee52fb0d96eb1049616a88d198327b +0fa45a4d4c0e916541612d6ad5fa60602581d204e70746a6292961f7939a5b4a +9d85547186a4edcc9f06e5f7c356d972d6d156be94cf5e1bb11e0dcaea4f7e0b +a7885f0b559ef89848795212a6f80eb28c351cabd5f2209c93f473768d38fc80 +7295c57f573ddc5fb21545d9b3b82cdbc973d73d54bdd4bcf974827262026b54 +894d64b630d030d27ed1f01865fb97f2d191adf7eac737d8ff1e1bc245b9a046 +b657832cd260d0f9d494967781b367ad8acf70a3abdd99ff490d7892692cb7ca +8580515e78d47699e827fe4e9f79d0066d8ed9821584be48b19c720027862ecc +4bffd2031c78ca73d8a41d3fac2449918d227074bdd96fcdcd53c5cca5cef8c8 +828494c33dc8328558611acd7f69f86702c2fc1e6ff389ee2a6611611aa65219 +5254df0af0385f267fbdda35528578788499beecf9092e10a8e74a656fc7ca70 +3b613d5533cc0a8f092b7d94f6de89d0888e03c9ae11f4c31dfaa97766c29c7c +eeaa8016aca0c26808e8b409b3c2c04572e121f7fab8f02dcd6b39abc7894fa8 +3f7cf5ca80d86836c05785d41c13f8ced0e417f985d936b91f439f2ba9490a96 +540df4c106bb4f92491dce8739fdefc9a3c9a3659d288e7bda130f27967b57ef +a291e82e9c1329a5278f2ac095d9fe85bf0c7dc163bfbee1c10bc1be2ef56458 +40e1be377760482af67cbf42373d168fd211c0b14920b8d16e10d66f532e6efe +d9b28950af2c4127a079a6f24fb4711325c672611329b2154f9b608d6ce14884 +6feecd025d23b7c4b558a14d7a04f18babbec2b24fe612d9b04dd2c9b8f45446 +188b882612eaade35690d5bf14c67a6b8d825ad88577956a22d687b501ccc280 +c2151d35cc52cda5650f18e7b0a72987f063ca94d8b749c8fc82cc884007853f +e4928e1a5ab2a431e6bd2beda4f06e885e55524b9ff475de7f377663b124510b +c9773ff2f3dd0a53929155cb426d6837fa87c21993f9b4451233f659cb202134 +44db51a701771031bd7e84aae5231a4b256b06a70b2a42238b9f837c7d75543b +d5c65929dded61c70b81339d2ceeb61eacf72f2aad7a6a708f4659d3f4544f98 +a02d78d87954194327be783b13634e3a5353326f7b6cfba7ac6714859e934e4f +89859b53d727aba1309a7ef8b54e5b8b352ad6b02da034e41bf99f30957281f5 +8ca7c70de07c7282176150bdca86ed7c501be6437e9a03e4da022354f1c32edf +64d1dcf47eeaf30b8878c1482b6ea5b7614168b95aef7459ce29ab2e667cf392 +334210b39b4727e993e1fdff34f4dbd985900609ff12d9c9820c45d7630d71eb +73075f0ca16293e46dcc5c28b8fd764790ab98c9ba60c4f11b4c90e13cadc760 +ecf54d6cc0a9d13e004f52b439dcdbe1ce4135ceef597d302946740f69196ad1 +7c27bc0d5f7bb5ce209ece9d9ed6fb7ea45443fa4c505b30055e6046dd4f2d92 +b8242941a8ade2c6da31720118a349870c2681f2cb327d2e9b7f5bf059f9436e +84a8ec26e786866d90c520e188e54fd7a2c92c3261731d7f03ac76d26325812c +3d49c1b7bb44f5ce022d8c617593a26b957cbeace2e9fa3f38c2eccc6abfb6b0 +4b69511abe64ded37921d8f142c6aed005bb014c35773434361afb553f4add66 +975258d2c45a91a46b28f7edb3bcdabc9ce7a5c646f0bc1e76ff66d558df8be2 +d6a3a5a8c6fb82d1bfd28c7d4f09549552aabd0c599022985e1cf6612b80c976 +0f77aa190ff3ab7ec9d563cfd66173fce4850f2a349c6073fd0cd9123755dfe4 +41d088f979c18c112d2613fed5f85c9f4cd4055a23888e67059545c105256fba +ff02f7cf451deda817d588a6a427ee12742c9579fb79c2ae523fde28ae12d7c8 +6d4e4e71a78872cde6ab7565db7acbf89d79df8adaab44f070fad17e31bb813f +4c1065f9f8f548a8ab83f2dd12ea108280bc743fe79beb28a8c530e275129cfa +272c6b9acdf9c14c70e06d2ccc564b9de022f8cec9730546711a5715996574c1 +3ee5fe99b0a07bf297ba7e6ff937343ba69ea88f94a51c56238f72dabcb77b7f +dbca2ff00f47927d5b0a60a733f37e3064d5ae83f350f675477842d5d301bb5e +061b01691efc83a8ac9a6d474953f12e467ea3f8f61296ca8fd7447741c18727 +49717c89152bf8d8a9383d4087a338ecdfc9a2578a88bce15cf01a0f2fee9bd6 +95fc32a967fe9e5a002e20b6e4579fe8d30a63924cfcfd14455a02ed4f149e9d +b86099514fbfaa85015537ac44f35e7d6eea18ae264169941d226372e9fe30a5 +1e9b7d9e8f87306a0c9eaab2b1527de6bd8fb328c25d51cba1ab55d2d5909c38 +fd7a567bc2a68e76f39540ab8e01a9c9bf6668d3d77c5705c195429778f12546 +9477e5d0c988e6ae558e9815ee1f08992619dcf71694e0b09faec7901697ad7d +a822ea37da29c27589161e7866bb2bda80e121f235f469bbfd99d15c5ff289bb +d88f50ef0357e84ef769a7e5dee65b43b6ed2576d99a9576091d6bece77b3ce3 +5d7a0a6484c157beb3a9368adf53e5d84552ea765df9a1cc617bf6add4b56a65 +6effc705a2ffd27c2d7142654daad0865ab21a83602faf10c74131c140231b06 +ab1674aebde0ca57d52df24047cbf25600c6285db9bb31c6abcf8c0820202eb8 +0adef9a74fc5ac5ab6c17499f377c262beb6c2c428635c2cd876d9ba5da147dc +8552656766a0b5d8442206500b4d4e3d35958b9501d3e19764741864b16a4853 +97ce3a31481ae09a6419fc00419e0c5756d00cdcd4343f6dd628b4b16c12c7de +3f52bb5e8aa7479989bd18d242c8390a7556f479730c9dc5a82e908d362fafb5 +77dedbbcae12e53bc1cf70bd381128216c1d8aa8b4f38865863ba940dd332c55 +aea36dc5267a85fc7a05ca2ca533dbc344a785c99b065a104a077cd3d12a9a5d +093b9191e0d8c46978474717260810235f6a6446d1ff1d875c89e236695b4821 +7a87350537842e38df99cb0589cac4e4c3ea36e8048626833f388d944f0a21a0 +87578ff4c3a7291d57b752134e7befdcc448dcaf5d462e30c4d43bfc67e0c9e5 +9fee53e7d1a25c5e6fd098f49cb41d336b5fe212a73b8277651a7ff868fab9d4 +2dd773e71f63d9c03b8e1d9da398d525807d14e1a0f992488e47e686af48856d +2d655ed33b7c702bf578a1c9062fac0b947aaca69b96d79106bce6a3682013f2 +7261a66e7e18481674d9a6c247a0ccc3b716581b94d8842aec8a7e90b8e343f3 +530c85e2b4413d0d1102347326b9bccab495cf59700f0e3eaec04344bb632bf9 +174568b0887e4da8e29ee0865f208791e813b1e5968c922bdaa6c7a49ed36e20 +ec7b6d6d02fbbbebed36ca87f7ccabdda72cd9268bb47250eeb253a2f5873943 +cd12ba33baeb21153b476fc10aa9100f795644e4d378f3151d4424e628d25a08 +b736562196ac80da2ac359dc7bc862ad5a702a51ba0c1e20c462ce7d7fd583d2 +e55f034c664e69dd74629fae18eba0daa29ccff1ead0bb59ebb26d7b4dd1728f +ac53bc2fdbd27e03dca1daaaefda9e512cf52f2676d1e0598e31b83d747809a8 +8bcd754972982d42431a429967c073e722767d0e8863a1a1fc9449fa26017cca +c062bec1e20bb8d362bd817fdc306fe7e0362271494d3447927f32b33740c581 +5f285b59450ec1118b9d4392fc2cc1da7cdcb87a6399e8e02dff98ea4558861e +0715854b6c69d0607e2fd47f5d3bc3c5441a3b76a0cfb4202ec00e12e744e7a9 +b71ebce0f3251587174fe1ead0e8e29c8f4f0ec6fbbc853b09aad8e5f38ede08 +09111a08e65d372bbff427cc08312cdb5f38e29170211d642f4efe657b27a345 +5fe0768a47090dba2ecbb66cf4c8d405f992b0e2a24c03cf480f77a70cd8efd0 +6c4dc995a2df993a5f62f4279138dc6ec4d578c07d9e46dc39a20c2fb52297c5 +1b280fa1b034b298a1e3487d0da13f3d3be3b387547245e2ce78d20ca9c945cc +47215f35b93326f066c3a523c972d8fb6988c9cfb03b3cca44cb3f4fad9a558e +f90dabcb1ec8980623df239c6ad6a325fc79736a572eeba369c61d1e8036d1c8 +f8ca767edccd21e8cf56aa31cf258e40c26060be7e1816719ce13866059b74ca +43c1206686871994f42d1ac390303c9a56406b21568bfe7c8005eb4a1afb21ae +fa74194315b653cc88fd6cafaa488589c3059f3467aae5b9ea5608d645a8fd0c +66299b177b6709e2ff572d06abb742105a9d8d7cc1fe1a01b4374e4f14120ea7 +1985b7c612cbb1e0ef96a07c3618274fa21a43a0813e692a5f95d46864a09b4e +962f24a97903d32c37d901cf86bccffa592d3917922e8868282684cefe819796 +5331ff0a363d439bb81d4dd948dfee657c3c574802b6121688f18a8adf22374f +fb011851aea139806392b4f9f2d6233669111d12987fa9139c21f47f8e1241d9 +80518854fc4b50d1fe2f2589ce5cfa4ecf1c42d8ca0bcd1b872aca5ca0e52cbb +5ac88651c3681fefac99864fa0db4e591942d6afeda35e3409304543d45da26d +44a0b3c6d936fa99beb853fa979eb14f13e8a86969412f614771d389d992c377 +d5e4a2aa0f8a881166d20ad60e6fb4a8cd471aae9869b3991cec63c22f32e255 +c83cfe6b93b0d690a1a97143cf64f69f9ca7316f43b254c6fbc03012158d6d67 +6b06c622528eafb37ba442496ad59bf2113b66b8128be686c8412a267b672ccf +5c07819d89f3132f6216cf62009d95cfda05bc4345330647adee20b3abf343b2 +96199a009886066775754e23d565d952640e6954ae259f0c222841b511a4e821 +bd2703919524d3285dd9fa0bdf3dd506282b61d67caa9fec217a81b3e7cea347 +ff683683b0d0f61a24df7fb771d7c6727e2185c486115bd65b08f353168335ff +3db774ef45be39bad54ce04d356bcb52704c5aac795d6b3a03da53a45f938fa7 +7d22db901c94e209bf46f3ec680b1055cf681fd74025a272990a3416cfc37728 +70f716da7a64617f58126090a3cbd619c851f059728b159b5a4bde0e0e93bcd8 +48b16434770b5a2873c290e50e3bd6f59a90217815523abc5ec8033b5c5004ec +6b97b5d7c07b48bba28090ee1af7a119c123dc8d0af284ce68ad1e7be5154e00 +94ebb121cdaae52da30d0caa2c586d6e9e0bcdfc2161737e4ff35e2997744cac +7dd522ef376dd0e96306abc1be6e1ec4eec4695fa380f30a7ce452cb358f85dc +f848e6676ceba21d1228b9d94675757599498be931c92de90b8ec3be3a694cb7 +e8e634e2b85ced512776e0577b5fc901e3e9a3f28339bc9ded9df7b3c33b8c97 +5159517cfe58976df87b1031e572f6e5f0bb0d4ac1453451375c7949e1092e59 +94122dac0ac444750c48d6980f3b350a639df8c1479f493441c5f294f79e31ff +d26ab68c89c72d2e4e8653b4c41d2a07949969ec707bc3fff1ffbcf74b717b49 +2eb26c12c45001dc956e57cffefa80cf29d3985319b460bd9ed288017d6d7d96 +c7fd24dfce405b792fd01c7897294c37bd4bd4e93286a1f1d9e7eff0f0d07db1 +bc31c241cfe6fb430e10543380b875140c194efc6f895c47be57a3f2f9af01cb +c20de979128acb6246d620475355ef5d3f678a171e81c67defb87a55292d6a32 +b6ee205092cca81398420d74124bd62d493d6514240fabeadece7eeedce133d2 +b7a717a08d7bb771d800c19fda05a4e7ac9bc9acd6c3ebae94156b8bd6166344 +3385275944595e1ce1c1aa1234b92cbe6b1f529015460aacbe2975a03075fd1a +f8454f608795d1d233fe53ed86b361dbd19aa63c591ef9a2684ad2cf8252d104 +14c09cdf4e60a8b5bbf1405c178485b55d345b9f28d77277339f1e37af83e92c +99458a028d3bc747d22976330757a9e25873cad1ff3410030297182b0a55a74a +9a91ae818df762adf0d15ac5af73ce85de3c50d2fadbe7b394049252e28a4b7b +d7bc5cac9c5f87c2094e726d6ca12ca65fd7be8ad560d8960a305d64bf6f340e +63dc2cfe61115832959586019584aabbd7050eb934a9630dc579d0c6b44663a7 +e04deb7afd733211ebaee692afe550fbcb5b612d404b314440bfc7d6c9f51e60 +700017d5f5fb86ace5f72d4411c59c8ae37f70366434857ffe84e524a31b014d +b990b47243d3ab10d3ca2600afcf32f83fb9b6d0a9c170cb3792f75b9e84ebb9 +7b7592aff0fbab3503d45c895c58a279d4f15cc78076fa27f52562096b7e28a8 +edcfb88d8cf82c494e673af0fa740c067639abb90e34a6c206d25c5d4d9ec064 +a3b9a1b7885ba69d6c0501f5a232646fefc3a8e108e82a956fbaca56778c04e5 +143fedad2d2e011040ffcde8c32baa7af50167b63c3a893673aa4c18f17e38df +92d15c970cf7adc0cdf4fb1df38307c93f979bc85ac655d7b07fb874779559f2 +96320c590c0b5965db5e386ea08a4a41b09283ca798fe26afdd92035313ac9c0 +076c1e52bd9ddef4e553688ce99ecaea7a0f89c0e2b651ac839802102931da2a +ea5dea997456f8ecfc98116acbea8941285522120791482583b12d4836f7c8a3 +56b228382870330054babbf20a11f0d13e3ef43fd0b4fe19bab3462c3553a4aa +28a74895e7c837be1b90a2f652df11994bc9eef27d81ad3dcba9e554b7f5fa78 +07b47665c77f8824b7d254ec23f9557cc602ecc448f6f441875f1070e631616f +54a684587e818c815dbc557e8652456ce4805e48a34688ca3b63c9bb62114588 +b8ad0ab6e67b2bc498007a7983282ebd1cb737c29939a9add22133a033d8e74a +bcd9fa09cd7075fef51357b7bb06e83409467d22baf0a8d51277c9138342cd68 +1b01e39790ba05ff0429ba79b632ba44e451a812645b1ab3b117589f587c8cde +ecae889953f3e3dfab71cfa94be824cbe1aa4582455e33e7eb9b762d891a88e8 +d948bf014eba7d2db637a5b7e22123c7597ceee18c7640df8d9acafcbfb7c773 +37c3f862fe36d6b9497664672ed80352d5015e572ab061ce3acc467cb16a0261 +df335381e8b04effac9f6dcc57a09a66c4c9e09b8264d022630634f465b5c35d +0b7edce16e4ba271bd363a15540898ce5e659aee52545fe890a55a6e71a25984 +37b980b4b755193baf6a3e2f34374b7525ca87f2fbb552cdaf5930b334695139 +541983582a3fb39f422a7c9357ebd2cd19806dd5f65cbbd82403e95c2baff8de +9bd10504173663388f7ddffb61ece4f9e02358043de1fdfe461789469766c885 +33f3419df0c07e31190bf104dbf0fb5e26afe4426ac7b07aef8eb511707fdba8 +263bbb12bfae9f0417e4c268999d34f458eee70724231d05e481a93bb4dd73a7 +18fd6863252aaa55eb365c73afe5183e32329c3928bbb7176147c8410c478963 +09b6166c4643184817811d12c0d5993df05786da29993f9b6a7050a14bceb186 +377dfaabfa788335f8bb205f7026ea6435b3832bff4bb24cffbe5d06fe8301ed +a4aab845c099b1285b6a49620d63c54aaf6c7c8683841db00c929c85761d35ee +ec18ec0272757fb66c78b34833346e473c3f8982e8d1e5f11c792647530fdcac +f7d205301b8f350f70ecfa4fc373264552997567ec3a384784ab551fb9e797d8 +fa6ce6472e5b1e17736e3d1d21e31e282cc87c853076f988595583c171f78203 +180fc03657338cc3d16140554d431d0df85407b8e58b479f565bfbf7370c0b23 +56dd96c9de110d282e57702bb0c521132337051ad094fbb0db61f936a0eb1035 +a975f8cafc64c3aa3afbf3d6252497e2ca542e0c972a7b9acafeb276751a9b2c +cf8294c52a29c751358402fa5de834a22365d15d28d089ff08e87e310742e08b +4a930a5f30e9f9d2605df8e9bf446931a9f7535b97c6b5390e30d28c052b180f +37ac36f3845cbbe549baf9e55cc48a7f32ae1428ec24e3a96c1486a81d0588f1 +25df661abc284c400557b489a24295cfb46cc4122b890fdf4b2d7f254e3a1777 +a757f8cd0f37cb51e980f5bc282c9d1ab1c43defc7f3fbb40e5b385a217f9d54 +08faa0ba959ba6c5c724e883b81312532c0e90ff275687a157a27238908e6dbe +e9467294d53fb9ae4b82ab5ea704aed0ec989b7fdaeef89134bcab30ed3bfa8e +3d5a16cf6e5d60b8ec02306c1c94d5364dbff093419a8312bf505c1054d3412b +7f657ffc72bbe5f6bffc9bf31349317a112e6aaf37fa2c7c222bbc76599a89b3 +6e84008da1582804ff67c02031cdd97fd022fa91d62dcfc38569e32cd0687253 +72f895e5fed463a0ffa07b85141b7ca5c7394ee00de6b09c841ea01c39983dc2 +7576449f84212614c8e1ce3d93dda056eb996d1bd77277db89303cec0d2f3cd6 +7d3fb1a2704a337bbc7b96611a5d59c3a1df4f0c0c23f9432c5f50bbc3abe82b +dbb9134b15077990e3b4fcee8555428bfc2f078883e6ddb501efcca6a7af9dfe +affd3b84f4baa6856aaf530adf7576099186064109101df2bb794723e4a608ad +ddf2ed9160bca4243c312c8326977dcfd999608576e0231fd3daaafd9183f2a3 +1bdafc9721476bc3a0ab401b062decb70960ecdda0af8aa56381ba9c18b4d78a +9a2748b5e9a68bae4b941476297d30eefa33c549ff8a9be6a0897751513bc969 +03f5043a2ccf6f9598b3ea1f8c47548c79f97189afe37cdefcadf1bc99ecf00a +50c4dac1c4f00ebf780e3974894ca4d4981e473ab4f57e5974bc71d4e0e94c5d +80f766a92f0d0231ddaf8793f114142d5c7b9e4c0ff83af3d68abee31d7a97bc +a9eb9fca36a1dfa2a6f57afb1277212d4f4b94f848858be1c827240441a7c0eb +973c608326bf0b025dae1c79ffa4ed0c8fea3d2f0031c7d2670f8baa9718c625 +2032684cc9136ea9e179a2fad06809e16f5522b5666fab7adbf0566cee482eee +534c82fadcc43b12e6d3378ae798047cf909000b3017a24b769903f5b7daa807 +52f2754cce304036e7ce4fe00cd5689e0001a6cb0deb08989427c41b7ea13b06 +1b8aa001ece75ec29c44840a5516093d6f0e449ebacb9c85eb0082f6ae0b356f +c3102ee4267e930ca38cc9a266e22aa8637d74bb08af30beb38f324fa9a43bc9 +844295300e1521beaf7288a32d1d57876150a22937f1b167f3f5bd39dc4f8b18 +1ebf246ac44ef2482a18656625005144be425436bed2c033c4b2061ac5e5ae63 +2be8a70eb8c3b0cb472dfc5f9d1422aa1ca681b7623fe410a9f7027a9ca95d69 +ca779b21a33c8500c6a6909bca3e6a073cfa80e561fa788e58e90d53eba40308 +1cb082e64e2de5ca287017c87247313fd6cb4226282ae836c859ff3779532224 +619d03514d3fdf5f9fc11da9bab723cf8a5cbcdb7375a75bd8fa4145af2ebac8 +5834b0b8fa1f0855948902bd9acb59235f08e99c50e669cd21df1b9f800a2854 +0f72a4a65d4e319055313bc032f98f7425beb91affe1c18c9d55f209ae562af7 +69714ee42a6485f35f49ba5370cd8db1b4b760884b9d461c5e0c19244ea6492e +6fdf8a2a5808bea8f76052e5a16beaae469af4f5c8b746a20aaf4b92a79bc0ed +f8365d9c95af851983ac88acfe156a7a7e222e11d83be83137cb8c86659200e9 +8b88e9902112622df0e4b01a51b741c0d81017810d0c981a6cf0651237efabb9 +94cbc33f793c55a52282b934ef6689190b550e67179f2975d45bc0dd11bd7829 +a1b10a3936672468789a594c2180d470aba330f9eec77df6d15cad95991306d9 +7a6f439af3bf16b6b4af6603b27efc7b149bf8854b7b897aabfc072127eac2da +b99984400edbfa07ec94e18c151a03c3a727371a7be544ec72fda38dbe6716ac +770839b3739921fddb892843075623048cf29125e09cc8962d1c8284f159d6bc +cc9135bbe37f7a09a256e92411589c960f80f826185519ecc551090eed32f110 +9c54969821914d00a6b1bbaa887aa505e5b9e198ce805d69893a8c5825713248 +6f7d353fb1c3af2694b54e268e9207ea59d1b699f56fc38fb0b4f410c1b6f36b +b71cbb7e6caafe3484fce6077343cc68ad515e9bf998adaa2470bc8e31712352 +d1c35c4705e9805f1506294742a146ced9df1640d115e32efeb4ab0026bfff8d +779bd37f9c733e8c3db8360081dc5036ed7d45a2c768cf984a0ef3ede67c205f +57b8fcabb61b38cd629195cf9942dad532e8a23b3b43a5efd1ffe446f39accff +1a64a5e51b9dd78c91c57ca0fc99736563dec32d9243e4b0560369006fa5f016 +0b37866d7a31e8a058d64fca597d96616203fd9e0b7763c67a8e6239d49cc6c7 +7d0060e4c98cb562b358daca31f837c090e239742f6dd7fa04afeefd6ac3257e +a15b8655f56da3154c8420bbd3f38b03abae9ca95ae22238b8906bcb85d6e545 +df232b152bcf40d35f5cc78ba3a4199a13de9225979f9935e98b269f67a92cb5 +dc88c021ff75fb9fa6d6bf783262a6f9e48d8701a64958e0ea5c86f5be805731 +81ab2552c405a18b74476c855fda484683ed8d5e7a4665b4518afae7c4aab8fa +a7b4b862a1e920f1f36577f85198558e75c6b7beea6915d6369d1aee5b354a09 +1ebfe0aecbd30ff546cc33081cde3abc7bc8320ba6cf824c189d31a5911b58be +019588081108b5f38ab101e75cb412be6fb88a37f48a7a70a2b34edcc9c54b4e +6beaffbffaa2f826aaaa1ce4f3c99321444ae21f269c04e31ef1264ffd4b75a2 +a4001d94a9233f037712b7f35de31b84134acf74f411059e9188bc8f2cf3b768 +2df1220ca7328d412a5c4bfa6e4393420d88cd8a17ecba308fbe514588505cde +01e42f20850a067823a1094492188bc0c0f72f9cd34bcebec2272f66c75f73f4 +c12b398e0a7e895b3def77e9f0dc32a8ccb1ba6c145f24503b563388ca307980 +4d4c5faf379a48385b5a549568ddf8519546a8de709971e54ac45adffa854f3b +c5ffcfed21367f36ae6e0f2498ad2fbc8b468f3549973edbe9061e73b2591256 +520e95ef798cba4365ec1092e07a7a1c1056b99fe019a2a3f267a36689ecafdd +a7d278a98efe2fe74f359aaaabab203850153ccaad8e36e851ee1898496b46ac +a709db0195c1a72fca9a793249079634975325f5e219bca6527dec1a064e1ba3 +0e225b7b4a92ca1af8e51d0445f57c60275d1592a78af747321edfac8b2379ba +8d4b0c071ded6acfe348b76e44043232a893842416e33431172d041ad4be5690 +ab730d42a6fd3fe234157f7d60c05ac073d086ccc096aaed6f398bb8338f7ddc +39fb5473e9dd678b14535aae8bfe73858d8d3deae4a85f90409fd01e63184e09 +c7053257b8e58e97e15004238014c2ac5c29133469df69fd8895251a131127dc +999ce914759938bba3a53879c0f20ca0942032b55763d4c326d823f93f7232d6 +4612ff1e5bacffbd072c47d4b2d047cbc087591b59702ad18c53b8761b30b897 +4e5b5161da036730dbd6c2a648e13d1cee9f4a683132323197f81d9d5a01d5b2 +e54edbd3902c01f154112001a51ac5c6b1847bc8e80279abb2f51fc1f827c5e6 +2bb98aa0de91ce3d7e99dfaaefd830d26d7bc37cb954bf56b67f58cce9c353f0 +6d285e739a38ad7615e2d478486aca552737d05feab9224878f6e514af47131c +2c0eceac915a1715c183ed69fdd59561dc7a0d475b4077a0b4d63ff84cc3f72c +c4e05aefc9e6796d590124e5aab6e67e38b3f0432333ef266825535cc2d73493 +2bffbaf9362b834f3f4c6530da5347aa7b84177ad8238ab7b4a1905486a322bc +985188879264416ee63df9e29283e2476fff6d2ebe95d16342e6e43f0d60f186 +e5d023499f6206102e6a4e4ab2462f50bb9dc93e67962bbb5c5e5a2f1686f9db +9b40ce147ad05140a8b82d7939e58090b793cb73a53dcf728e4c1df3275cb6d6 +4562c14eee738de09fbddbd313289cd51f11ad81951be73a6c49a7fd2c10fe3e +99a57d71ce63f17ac4bc0822b89fe57bf1d90f9c73368f5cd734eccde24a5f7f +4af214321b0237fdf4eccdcf17e33e316bdc199961794d6d7e531769021b56ee +e7954d745123cf5e6706f09c277de720a0c8aa43aac1802a53d0921aab03848f +282d2ab432809d13bb11d779255b36d3af692d9e50499a4e2d03c9d6afebc545 +dd68f82054de2f7de9ce8e6c1e6b125b4e1ffd4c7d801c8923439025a5d6d3a4 +3ddabf5bafd0799b9cbd932c6134233df96c2ddf04bb60dd11aa4f9faf7a75fa +362769bcb7846c3278c5c3a012a34e73d3ea6b75c16f0eaa43ac2767725c459d +948a54b78666fd2fe4c4879e38aaa030de0d7a5e9a13bc1da318393326abda42 +dab2352f68395c0974aa2470beca4a662b3b1081285133ce6904f398d6fb1ae8 +7f246e0126b79821c8f055ea427acaa37ef150d6bbdb984f6179bba05182a9bd +93b570cb1d91ecd4eeb63db2d7cd76dd9063d5748f20e3e05d9a6c680c066133 +3df9ec618a7ab2c9489f9c9ec2517878633c8b6327050e88277006a6998239ce +81340faf6d6c4cc188b195e617789565d9642e69f9b65c89d1f04c9ba3a5d4cc +c3851d80e6332e77d6f2c5c9a45ccc7483d6ca0e45bb16485ed1a77d7a845a71 +f444ad6d6b25c75c0340173dc7a14d073b3128bbc290560280aa5cdef40fed18 +acf008ff5fd841b0ffb4587153e2f6570f7bcaa47cb6b5751233421a70bf9103 +90ce05c0ebe5f8698c015ba486096f15a63cfc153ad3331ef538fb7f266e1329 +93b81be2eeef5fe7cfa547fbf16829fd729cd7ee1e75b4908cf1e07ad14d4d59 +64c98ea331ed0f68bf69c6501624cbb4dbf7f0dee1e0fd2f8a8ea169bc3d850b +7805fdbfbe9e6f6f20cc9bdf2a35faa76c2bcb37e6dd379a1b1a91b37deb63e0 +713a3edea7e49bd9f8bdb8c356972ebab14d3646db7e9791a213982e91dc91d0 +d05141eeeb467a622d8ad66f4c06137667a000503049acb5632d2bccc6ba7956 +f2a13c19986eb0b926519033942411aad2244706cd01d2ed0b191bda10a79207 +0338fdd3cdab503d68a28e4cd4372abbd93a1ffd413e05b01a6ab98289480c1a +59eafb23fa79b9dab706f1d0f3ed31b7d5018082284e1a53f56af977881e77a9 +3015de26169861a5980b6de97181d58ca656a05aa4f19b5c53f673ac746ec82b +62f6ef8e35629e538f5febdeecddae88b1dd5f165ebd5303d4ae43781d912dd3 +0134826c69b6db8c246fc24dcef8f0bb0754b188abec58d89e7999c478ed4a50 +adc8fc3a27610d7ac94d808eed9b7f815cf594240204e26733d2410f8573f904 +da9b42321867933ea4d5d3829dc6d26e577a11d1c3bb08972b1f036129ce0962 +9b470b7033ec2ef0e9d4d264c5be22c205d0a528d75a6c03ff2af481e0b33107 +0c8beb18af4b5a1a61a26b1e16641a2d3552f3308ffdc8ba07533390a394a6f4 +ac564c8d297fc6035f032e8a4a58b0e7ba5fca3c91e70765703e6708b33d582e +2d995e8e79109d184e85f42d7875676026c2cb25d48dd5abcc90595338fb5149 +429b50c6e459512838b71bb6682b547df3fcf0752f7dfa0d005f40e2582d2bf3 +7c3b0efd27343e1cc131a2171ecf841e665d88b52b6ca2932b7f792bae305f01 +c1aca44f3e3c6d52281dca6743d726d7b851e5556943ed12280ddf865222f75f +cffa83e00069787e0d116f6d219d75e6cb08bcae727789955914828b42cec577 +cebdc4941806c8ecc36c379533379ca8f218bfb62c30db64ceeeb82bdf14fe88 +917312b205bc24c9fbbfeb42dbd55a5395ffe42ff0107eb521b09a95f3376f2b +90229739e9b58541785f51c111148ddf29842281907d1f5365e187bdfcf8c409 +d8accf0140a68afbfd9a31988798775b83219728c085c5971015b82e3cc2d83f +89023584833268c72ef5dd1ef665bbd1e8a0cf69ad02d129ab7b167410ca3d99 +81a838f897e3a2bdcd4629db143bcd7d7f3997b332905a1bebaf225b9aa54441 +429ae4628927a9a5c16230a8fd6f0e5f652396aa94de3ec9b2b67eac09a0a6cc +7c65f762a6ff57cccf6fa22bf064d135f7b6f0c6aac77b7c07911429cc9bbcd4 +7d5e994b2b6bd78a0405feefc8436a1563b968b4ddfeb1408220fabf8cc49c0e +cb85ef47a1c2f4924cc6515a56cdcde23b47981a49d5120577146e8872b94db8 +17d2d60b683a96819f4cd1861b0cbf8d89e80b1ef21ab44df4e771a96b540eb4 +b6bf3d7f8f848e05bcea458fc0f78cebdac333e837aa0edf4a016504468efc20 +8d4415f547b96c3d024c313c0c34be7932d36dae133461921d07d007047efb10 +48517c8f97ac845039d9fa739ea982a47563d134c59cbbd1dfae2d91b78be1a3 +de96dc6d65b10125aa25782f4562226d6d15f4f3273f37c1a99c9621ef2f0f2c +d15bdf5d7798a4d842cd7abc494484c5060ba11aac6dae658232c01d76cf32a5 +fa5daefd99efb53faf9ece241768d6f13c264d9478763626f88fbaeebdeac3ba +ac13a1b348aa4e1bc8078c31f30874e5030560144a09b7a6c2b0eff4b01ddd6d +f039b89e14fc759425e8c3762d48b4fea7f9d01cf4ccaf2275479ac0ab05bef4 +1fef56c29dfa3033ef3f0b44c2572c279aaa25171f9a77923b07f4eb2145a51a +bb141dcdd14d89f9a00cde6002444d5cdb467d63e3b78fbf0bbfeb5dfe2a10c3 +91c6d2c8760a90ac611e770005db3c1cdda6e2be72ec6ca8a6419b28d73b4b56 +8d53024b5340e46556051eae86e17f94f80283692ab6ecf7a21e485133736e9b +18646b57ceb8c3f7f57fa6bb95ab7d70e4b3265d60ec4bf5ee9a870ae3d91cb2 +66a9a530fec3883cd464158c51418e8ce0ddb8332f17733515c877ac896583f6 +f261ca135c20e9a10dc4959e52e36dcd44e9bfab35a66eb0ddeb7d493e35d2c3 +c6f95309ae789aa0acfdddcdbb9dd967e6ae59fee45025e726957534649e9a61 +cc0bd316966fc89f56fdb2e189b6291c32d283b37753f98543ea1adf1e42395b +72c36e036e34dbfbbaaef1ec1a6ee029fe2384b491a030474a14a0241872d7b3 +3989993a95ce9a49383a5e4b1a568a1153d0c6863f086335884796f82aaeb37a +4b50faedbd2da5db18f31278e514212a9b0c2f4afb6fcf49c33810f4b4cf712d +f12a5a9cb7cfbb24de1951d225b6e052cc926361ef890eb28d6a986416398e45 +a254fa3cf6f2a002fcd7c8d7b7d42b073aceca69e969a5b0467d4d294d8c6ab5 +f5ba1c0bc4941433c992bfb928588e7d104edaf9e80b216ccd10fec231baec28 +4c3dbb76801abc0cc744a5938752c81050f96cc5a52695722f82ff6aea6aed40 +a79108e6371f378c52bbd930eaa5f594669d31abcf58176ab582bcf0d74ab652 +cfe35b7eb9a0f0c94267d7e206f538b850ba365097a93cf1e41044e30a0d601f +1573ec0d9c693d68a43c66dd144d89faa036364a9dd96c565bab73e909b3aff1 +f67e904fff8901b685fd3884b1c9c5e214bdf7cb903464556a1aab8d14577270 +4c961e363ec78cf19c98b3fed46508f5643361fccc5682623fc143caf668ae45 +f6dbb611a227aa8c976dae2e65ac8c8a8881fcdba2d30351e9209fc3c947f617 +25769fc7a569aed0a28781f9b21e4a672e414a4fc56282363a784b9c6911c4c2 +93ccef3a07aaa0b6896c9b9230d2451cba537525fcc9a7964cb546c4b1cf7f19 +32333244950be2f4b97411aa2e521ca2f2026825527189114ffada72d856b6e7 +ac2ddde6ed8962c249e4355335ce8ec1481c1447813f2e18ccebe52ea774be09 +3444ed237a27bbe8b13689b3cd6ff8f758cdf45bd1f930db115153da54adc49c +066ecb7173be7b0504b13655960a6e548572647c3c2a016ca484f198dbd3b55f +bec5e63a06f890773de1db6b7ce388d588dcd42d1701f2edd0346a0bb0c6fd5f +d6bceafcdbce5963e84b4daeb34f96f130ac08af40ae3a3f4da8ea451998a4f6 +ea636ce4d7bcf68a9781e1dc0da61f3f1d54424ebc78eab1437f85300c0c9b3e +e53ee61f7ff01098ebfa819ebd06d0cc76c8f7b79b0dc1382fdb35375b5f2be6 +ee4c7f9350d3afc86a8393067845f7a0121f8e2726caed7add44b7d981289287 +359dfa5ef6d68cc4b9793c66164fab87b0d966758209486e4656dedbfa0401e6 +da92a57ccec105a5371db14422ca22646396c428f4548c4676c2934bf8ce6bf7 +ded4aae22b045bfa6deeb82da843efe78b28ffa06d936e174acd3980c9722ae7 +c79fd5f8d4b01262ac673160cb32f6f637776682c83854af7059946f1815390f +e532475b7f95c8cf5e25dad22baa2b987d89c43a5de50f7793fdcf19030e564c +3f1fb816b5aeab8c283402e2a2557944041daedc3cf56858e21b12c1b3ed1bdc +acc3b0bac7b5caee72177e82ef621e0f6bf43f17710bb740cf8cb58c054cfcd0 +17e8a80951f5b676fff4f496c6688905c4151002ef6cdcb4c195ca1ffe696604 +5799dfe8f070bfc1f9e34b9d96626d34246bf6b0bf5bab3a785747607c257621 +a771fef338522b82d43e04209c33eb7fca2c334215f34612d41cc3c94577c01c +5a619554c8c43fa7522766829bd6f0d22dd82b87a767f2de84283ef25384051a +385e81e1ce9261c3a12ff8b8e7dd10db7fe0090809a4761541e59a8af1942a8e +a6c9247a760bdd862af1157fef8e4dcac2e33d7f60e394b149f4e22620486d80 +0d15be900e19b304b3a036428ef9abc5e5ca72666d08f6f67d3b38b41118b695 +9679c2b2e05b13e317709a23bfd5429762b3ef92f6d3d84217444dd1a99b8217 +5bc5992d69d5842e119fe594c175638081880bc4631796379bd2ccd3b7124675 +1373183a485d08103392c00712d0b96627fc2ff23ac52fc5979ce27be2e4262b +f4393dd7bb31a5c249d87b0cc93ca5d0fc9c6b28597bdfaeea8cb116386ba424 +95745c745ffa8fb927dd0e8db59f22874e87e97857c6aa8a5ed22a063a645fa1 +071a6345eae0212ab342367bc187e64868f47c4618fcd6432931f6aaaad106c9 +2615452f9bc4efa77ad62ea21c7ca25fe387f18f290747ccfb9ebc876f208e5c +be109c11056ad49cc686bc9e581a926942a51912f71c817fc0dff0451c3acc4e +a739d1bb89c0af34ab5538ada2d0c74a086fb18f370a3b6103782ef715ce73f8 +e6afba12b462921cbe7c47280a7058bef1b510e2714bbd2108ab4d371a035da6 +42e0de9c8549b60740834a9e70762958527810eb07dd91f35a0edea14865bb13 +2c5c3ab08e620d56c47ad8c1e815c556fd85b51389ba55d7787bcfa860ee9a56 +20a892c9e456b4a15e8fcd621cebe3fe0cee5985b9f17d75c92de6d69bb1107c +a0f0172e9a9003259b87fac6a5f364ba6e183fb5c606606458c632912021fa90 +6141c7b3b3a3eee139e608a0bad9ada5f122610fe5bcefc0aaa6a6aa54d368f5 +86bc1cf5454368b92ce2a667adf2d0296fff7fc9350c11bd1c56c893efd6384b +7762396939fda1cffc72c273232794bb9e8fea91af248528c408ec39a1436be6 +17f8ff08714a92852a199283e23de6afa58470e4039a2ac3a4e73189d6e03577 +e22b0b6b09b53e1180a3a65cfc910ac05bea47370337ffb40547e191be4b7eae +81d3d2ceb730d93f2117e2f6f5a80f16dbd18e753acf200a9b90d782901c00d4 +0f38c3747879991e918ef90a978eb3e266e9d20fbe54fc9f4fe75237a75c9f1c +2034c6bb3c243296a40d47dabeacb28dcdee77e99b2f4a18edb273aa87bfd658 +ece239ce85e1173f39960b09751b00bbe089d59353ae7f3666f5ea22879dfc1a +f92e67b6c3f112fd0dfecb85e9833121e57240c04f864c40a0c5e86eeccb1c56 +ab6e37a7502da756e41d25a8b2a4981d082277286e1847d5c1c493df2fed18fa +0308d1cc3a7d206064280f3d1fb6ee833e9c02d9a7667c6749025d6032ef5935 +0f17db0f0057ae61cf7ea6096d8a50f8f873dcbcd943091e2d5afdea02978041 +0136934215c61edd7c1d698d2467152c288551103977a9139c858136554bd9db +062a618c336ab945580126a14aa4d07ec336ccc8f3e32339cb1cd0a0d1dd7674 +be5cf487b33fc52bd59f03f2bc4eac4d817c356ad67dcd11f77114d1ee651e46 +d62452a654340aac6f62cd9302bc8007a370aa332153e29b4df6ceb58b2d6fee +5e06a4087bfdfc9f9bbb94972a37f82d3acc52a5daa93e22172a1f2ffde28cea +de8e9646fcfe4f5a8deaf5e4a893ccef06c87c791631f04728285f701a0c6285 +5c2aca136c5efec8ad1cdfa3d34a11a6fd12dbed930872b936ece7993d32cba7 +ff62d5f62620cf352bf58ac18af4051dff9d20d796c3a906f446bd307830d447 +0c67d43ee7ad19c26a3d24c8267c37bb698f884c6a7fbc91713e5610a88882ef +d62329aa2dcada088cd7f7f81f7dd6af8d2d6e507154674e9e0ac0b42c63eeb1 +382ccc97cb3b301af8f77c21953f2e9d3bc627cd7f3e56b661368e825f375f45 +9c0a2646970cd04dfef6b08018d14bd6d35d30ccb1dc3995b723ae956dd547ec +994d6c7b74cc585b2e168678362e5357f49bcd4c8d054bde8f4d444552617733 +6c7c2d84cd618bc265e01477211bf6e7a93868e48acbb0f69870612b3971ee81 +48da683177c7dd82392d21f0a13ad7556ae07eb9fd03870df7968412be97f37f +03798eae0c99aaadd0504d9964339f486d447f5efc82a95e2d2a72377772efd9 +251ed73de6e73758a85cc8b6b3bf11fbeb220e9e12cdd314c3878557b32b55f6 +0aa93f6345036b0ee9e58f30106a531dae0a385a42e9a9d57551aaee5d228fab +6fdd1ebe2fde94875747641fd73e5d120675f45f8656a4768d76d8071108b799 +095fba50cf84cdd4f4d3dba6ae9ec9b06148eb46bef19029e1471a19a0459984 +f35573f978c9b65fb0e0dfae9e93d48e5b931afb492da21d2bc5a38c2330450c +654292a190d016beac84ff2d287b046ce2810ed5617568896c1302e255a8b68f +a8a715af3adfbab012abf3a899e3040296c21b90b36572646e6d95c0b4c9632e +413b1f0f3dd7a4e73e7cc8031780de1bb3456a270e7be673d1ff54142d6eda81 +9a18d2facd7a9d1187516d0ad7ae7c8ac4df60cce82868edb8d728a3fdbc8b0e +53d512987f7992387f8b29a6b4636948641542b38b0f8be161d7e8e99f636a8e +d770bbf2ee6a120cdbe2c40ded56a58054f4e335d9274f64126a00ed936fe6d5 +a78893790d028f88634b1ada5beccd99c48cf02174ba417baa29361f74fb4be4 +a82505e7ae52e50a737b37ffbfbee34d060c2af17087334e9995eabe91502de6 +9ac8b8d081c5f3e696b957d4444ec5718562984cf3b051916a042c584c2a5713 +36c38abb403a17c054db8e41f4ce0da11175bf5003dcb7e9c314824118ac92fe +d272e643fd93b298e4bf387de53b6f01db1870060f87b68660648f2f060b4c9e +ba46bff34d438e1a086e32d996069be3151e15ed116a894b59a915580c82ccf1 +2a3d2a896b82b2e058056f273e4a0dc1233a7369dd89d55509a4799fe993e26a +4bf775e63669aa919b49cea40588033c1b83520cf0b417580b102fc58052f1fc +42b566a26aa07764c9b1c5d20b51a8b2213ff4c5976bd4080f603ff03f0831e6 +d3d6d41a49ea4235fb2d381a5c176bbc90820136ea813069a0cf45ffe7c14c6b +9439909b5870e894f40bb73f77152f831e64a7b83c63152824a5b0bd00025e4e +de5fd62a2998fd691f57c1c0ff7bfa691611c443283314df0d2220d3e1961db7 +d308506c751775ee7b5d2d217656153457fcad5ecc22fdad49ae3cccb48d98bd +8f40974a37ce9865fb4560d2be69716014c511403b8d6c16096f5982200d80bc +9dd8225a7b801eceb465b2129fac7bffaf4903d8e0a115604876b74d54fdd7fb +671a182b83120722530b56d525e0d8bb04d28396dd252cd2d0bfb43f0bbaf077 +aef0b594cbd2922ff54dd151d293bbdd002b222dbf1bf04b5ef27c50aaa79743 +df4fdcfb0b552a9d82a5f06dcd765e3e9fcf928468d642410543728b3c5cb75b +6b7835f814a3822e47e9414eb746405dcc06451d955592bec2d23c4585ec48c5 +2983d81de17147d288466652bb37f4e8828f19f2d79605eeae626955f700f310 +5686b63857f901f8c0d39194e6d4cc1decdc9f0c58efdc5ec8bfc61ff500e973 +f46df0ff842bac970fc821478578c83a8fee0f0c1cc08ca9bcf050595dec0305 +1f8fcdaf407627592cc49c95cd70762f2343bc123f6e82572a63c7f0bc7bc0bf +30218d208db065c6e34400ae5fff1c644dc209f9a8f1017186b84fa4f7d5dbd5 +aa7c5a84bd25fea1c58eb671e89eceaf8066fae2dd26f6e20af0ba8da3cceb40 +6c354d1c4a23c8e947e7ab849942de1c4ec0cc158d14801487d857fce67dafdd +d29d21fae667e384ea891c751b42487777c44df2c8f1eba14f5b2704db92c06b +8111eaa217065485499e6dbbd2a6387e360e84947e891c9146395ba4494af0d9 +997e1901d513430719c2c96643cfbf5cd32e80eba160122cab14f36c7cc0e4dc +a8f76e26aa307bbbff3c70927cee68fa7e9311a28a52c09ae4e1ac45ccb7193e +c08573554a6e1a3a7c8c41a8a3f5f90a99fc2061af8d515048562a8dc0409c13 +485f43bdb5c7a297f6e38c25b41202903662d8d94140c56423f39fff21b3d617 +be05212b699116a23743768bc986984f0b345ad60805f20bd5fd51ebb5756707 +9d12b4c27c1476326e4101fb5c888710d0ba1927d85cb2395fd94ac9b7988942 +6bb240ce52d08ba33d8f6e033bac5ea89d58aadadcceb9d7299ae04b21f16cfc +e6994ca594a1fc4a2cd1052cde7558b0f7211f251f2ca4db34a79e3649e3f7f7 +d35da651b66d557e1fc4540329d2ee2ae5c929202cfc4aecb81087e81f074b7e +27d930caeb6c4b854959673fce7f0722a69b12b862476577fbe21c143bf51dc2 +56d8ff8caede8645437d2bde4f732340d265d7192120700bddd8cd1c62940a7e +7c13e6548467a8d892d890f944bead7cc60823059213005706a0f5af1fd4fb72 +87b443922d7d60bb7f0253071619d1e31912c44aa70f50875e962484b33c55ba +1785a6f4c9a4b06a1bc60f31ae38528d64c1394bc0a9663d6d5b195f82fe937a +7cacc89280b539ef236077903455fdacf72d9f32021867522d2ed4bc22370fba +cba25c4eaa6cd851ed89f2bafb07197887d62d8a88ca360b72f3d50f75e671ac +3bbdfdc97a8cd5d11f48fedd9751af994f1a8c0cb965b4143d94bd8cd6ee1bfc +3b6d05446036a99cd9467893dc2041ce7c31c7977753fa0ce9372ff2e94e5db8 +a6e29228dd2332906303f863d6b3fda19db1a3c8a282c0800148d87bf5e19cfe +d602b052b18af6e2d715d5ab39fbba754a02fa138f7059139caea465ab37fbad +b4c84381e21e3c92542fcd12f78a0f8256d7c872db269cb263d7afb9ace6eb31 +7e92ec8befc0680e150c340fcbd9a4402cd31ce7effa8c7a43c4bacb2039636a +3f3644ded2186553edb1e031a4ebd6ee27f570a8d78f8820be60667467fd377e +cd122821ac60bfbe6414372ea6222629aea084875647824ef2962fbe9e9e008c +82f7ec6dfbf9bb507d69da194bc0f1c77cfff635f3fc96cd9be3f155907459a7 +d5f107067034a696854dd2ddad3e6e67015717c15153c343c21d567f98f43acc +a9b291759ccc9dfe70af766569141c00ea4c9d8a74a4527ccd073fc5c3739645 +f7c12e1155b546ace8c60a216549a244600347454d4dae4a5fdc119276c4bda4 +961c9e2f015e4bcf06e6cf2e04b3deeb495cc92776085af8760d77ba61af452d +1ab9a612073725423d6626148e72b4b69d00088bdd12340889a7efddbc8620d5 +65df43b14305f2a7de1846e988d4a197d493e98444e396031ed2fc7efcbd2137 +a931c3ea165e5ff36b2aa55befb144943b9a4dd51fd1509f6763fced48572065 +479c1133b5aaac5d490bf31f80258ed0dd6bdd46d893ed9012610eb34739cff4 +8fd183b7d03d39d857bd78ee60f7c9b8f06bcea59e90b2b9e328692e18aa8c8c +f9681766393202342b75ba5ad585411bf0b20edcbd7a161814d24871630c735d +07aa11bdcc83c04d7e5607c67bc8c9049fd86a29b4a92787c03501b4f16913f5 +607e4f56d43f87ad7d505fe36506bb95f2114946cbdabf5df41b70930673285a +c9da35738b4336623de4f3edca6a0e1380a18f8bd91179ec6e82c8d165f430e7 +7c00b0a48b98a763b7461ca9add215f32c48a03fb04d87788354eea4938dda3e +41f47f0caa2fb5f85cbea544e1d9517d9acab02ad06fea3dbc50bfe59f973abb +143a4c141a9d1a44c03625e031b769a03725a7edf5bdd629965975cfed341662 +e8d0a97c507d251e14581f56f820d9f0bb3d0ad5bb7f8f4431deb023912f9fa4 +696b1ef20e0d1a3c772caa39e4e26efe123d84ad032c8fec193b55b25a7ebd1a +70e5d8c79a75430bc68c228f2345cfcd8e882871cf54c42243028066f9b415b0 +2e5dce6430ff0854db8fa0cb34c8eb4daf12ca8e51da8de9a14a02711ec96c92 +0c33fbd02b4ab8141772c350600961765848e72069e4c0a9397e55b4beb9a1a6 +b8c5da06edea54f81a2afe3263f3c22b0aedefc1e342c0188695ce2f45daf614 +333460e67a985a8e352f6d95d01dedba8e555e621072883660a7ae954d0d90f8 +271ad913edbd9b72f70b19cc03acca245969ab09e0a5d9f122d7a0f11d2595c3 +1bcae69ef7bc9d3b637a2ea4080f01a9932e6d20290adb87596d49531a6b223b +35b76ece18e1c4d156d970f86877f34ebdae6592c60b8df068881d841bed1be6 +f4cbe9e3ea34801d098ffe73de049bb5ddfa41087dadc91b938aad25578c66c0 +b0a843e543e8120de7b2590dbcfb18f391bd53d40b64d9a4868265c615304c4b +1f8abcfeb56ae49d6d893de20e722e6aab7fc5dd9208e8a52792af13685dc02a +01b9735631df4c2953541fc16906217f26a1af5c23e7a2eef345e516a756bf34 +2bbb0da3493a7f541593a8678fcf807622809ba6fc08cc6828e526d1bfa11764 +77e3f135a992585dabf6208d1664d40e725b5bb0e591ca4885df36ce43a63b6e +b86bf2d7648253e0d2ded5e81843f190064fd37aced6ead38f8611498faab564 +c9fef5d273a99298a4a4a6551691ea33afb6698d028af6dd08b22a013b1d8c81 +e2d72004f8f59fd296b6f915f850206a2bd6d0f520559a39961833a8cf7022fd +7d7be68058fb28b05626b6ac91b5c40e12f7fb36ebce6ce5fc3fd787f97e8fed +2495ca7686ac191a02aadbdeb4901982126d859600fa3be9d0c22edb1b6a846a +e221c764530fd802a3c3923104b6c453386591c771233e037be70640c049acf5 +40bc5a05eeddb5c94c36259691bb84faef976b9388da999113ddb505e18bb949 +631391da68f3054973cbbc970d0b9f303e0b697241e2653f3bddcdf81d92a044 +bf0877b0b5cd8c434347cb54af95df00b677ffba6dc77641552a1edf7cfb93b9 +27a4868e920a81de9bfe71f5c0dd625e1542098c2f35269b13a310362372465c +52a95ed95ab08025506427a516088ad925cdb2832a920a0b89409f1ce8e2affd +e28d980c5e2b70e373ae0dc6016738d13e2dcd62275c5814d040fa8bd3c75519 +3e47c8baddf83d9b56559e7740f2c765fe88d7edd7e17cf96c10bc670c832553 +6ee3df8af2c0f3641ac5b73f89660e8048678621b77efd0e36f849e610e3d543 +41a407203331ce35f1f61c1ef3f885e8b8e6eb7c2e044b7d58c214dfce100871 +4cdda6f7076a26b5dd63f5eb3542fe151b7266a8f79620d968b33a1a2694e2f3 +30c57c6d43ebc35341624552437562ace4b86cab7ba45e1bc36f0af32792cc73 +31e4606b9275007ebbee6bccf4c94d75392ead4a24d574385d72410db6cd2bef +00a774ae7d8526f3d055e916b8a336a6f37f7d5ebfbcf60955658d7a7a5161b6 +b8fe4fa529796a4180c3c2f5520f12898938f6fe29d41f6b5051dd47f7d937fc +b195fcd096c5ea15d17956cb36aa9bd8196c4cbcf7e2582681ca953147fd1458 +81f7a995c91656eef2e125c83bd92467fe53bfa56008296ac6214acce6adfb13 +cb54cd5bc4513e2b21afc21d445315b1211b0e329fe645d932e9a0edd66a22b0 +3d8a2b299b2137c12c5f3fd2be200ea7b666adef80bcea350ab8168a6eed70cd +6cf710f89856dfa735b21fba7beb95be18ff73a71f11064c48f16b8e4e69d07b +58dddb56aac9a1afaf34e570f74a3a4b67ef07b43604b976e9b94652f325119c +b4fd7cb58329bf5a95fec2b7a9cc81f909b644b02102e5f69ee94bb1c4721e77 +40a69e4b7f733b8e60cdbe2e7a5b1fbeef864f1ad7dc821ac9210414fbd6c485 +01c3f0d92f56bad0fa7297e363253ce91afd9fea6f4b220431b1a3006541882c +abec8a200db7e423df225cc8837f3de9bd5d5a4713958de796d1dc34bcf6f5bc +ffc7dcd86f34c56205296b41a02e29cfe8ab520ce8b1d430b98431f8642f4f61 +cf5d48ddf3576981a23ef8800f573bf1ea59c3fc0a094d68005e0f78929e14f2 +371ffc72b9d103e6be0835075059f2246a5e360130fcb374917f2ceb53f86e8c +c41f3826d749c13b906bb99b28914185fe596ee62a1acceace8abc134564a8b8 +a60bf888d26553072158cead9c12e1ae52943ea178cc2221ff164502dd580c63 +ab58465bbfc39244fe92a6b13ef6479fcdeec6cb118038659af63a8218a53cd1 +f7717f073832af434ed4623dc485c325694e53a1eb443739276518c78ad32248 +105fdca940fd8da347b538e2bae21160953bec0833c76e8ea58ad208927cdc92 +8c2371bbe86b475b781d99302dece9305b546ad411d6522b233de74f55c760d6 +1b033479aee1b28f7d7af5ff029d9d7f8f2b81cc9fa3459c489bc0ec215b0e6a +c52e5c7afd23d74de3a860f57c60cbe4f63c75cc388db596364492be46297f24 +6335370d643d0f002fe28bd102a404936650671e312ad59268477368b0452413 +a9f1f2279711f476c30e2e1382a0e99358c662a48fbd1774d7e5e7f8ea27d98d +09830c9a1b0779420f9eddcaa0c9a03303ea47612716e1d2bdcf9982915f3dca +6f27338a30abe4dd888d4df232014612ba94900e9795715a2eb989e1216a9af5 +06567a37846cadd42c2abc9269f0565933bfe83b3af21a8355719a5aa3336ae8 +3f119e27900f4648319011d21d0ce403317ddd16c73b7c7df0d2c3bca31ccd8e +749721101012951998888e444d7424a8231cae2ea75bbf5784c95bdfd12810f2 +0f7b545a2d34c345ffc02b40b4a188ebfbbe286237d0d6a5dab7bfa4d368a4f3 +9a4ae0eb8c3ec350068989cade49b369f6dc1274623ef6a297cc5d94bb662cf8 +b4a2d2fdb5071f5de093f84a76e22c65382c4d3fa053511da9013ba2fe7efa40 +87a039f6ea334b61bac246684c75fe201746851a72d944ba75757b7f858bffc6 +d68bd08a99e269525734001e181e9e48f4f0dfec105ad341dd22111bae6b754c +930d10248d6ef210cecbcc5e26358f1c1eb44a7e56c7514472c548427c239b84 +75d81280564d4e691410457a46379f6d1219c41cf7862cf9019d6d9889caa6d8 +de7a0328bcb32ffadfeda9edbe576ec37cd2b8ba64d88f7190f061475f360901 +2107bc0caa700c8c1d42a31f3b3bc34eec1785bf6043de2615ca436dc26bd39d +2bd47d0c8947dc44d858226c4def55dda2e2b3475e0969d34cbdf3830f1a9eda +ddd24fb80dcb3012150fc7395a32df3422b64e2e0e6463e3bc97001cff04008f +0ee403e1573c2c6566df09638c2e3b753b6756cfa89b60dafbb0ab85280baabc +cc1c07e470f3277b2eb3e3cfb2b899e0583a28c1436b1297224e2e7792f285cc +2c7e4ecc9ad44f86f71cd6d5c0d9aa6a10d8470e81c8b3f404ba650daa355c7c +2e73f335705e3aacd04d78e220612b54f7a815975cc5f3315a13ccbde8345948 +a8b277ce1965ab563f7d330f59ab57351d10c192b2393cae30e435a4f9ff067b +4f3836c12fd1c9fd8d4522b0916dde72494b89cd0908dd4d91bdebecd98c1be2 +5c6c6337984a961e21770ae4fb3c3d60cdd6c44749f66fa913cc0afba0bcfee6 +4a0f2bee9de5229fa2d4cfb36577a38146eb1ca3dc0744ff22c4c26ec3fceb50 +25210853c824cb8a26f3cad66c473d4f8a38b2975f0c167b2f7f56f938f69c82 +d91fcd5be049bbbca831c9ada8515ac1cefeed7972a8d0a919f561c3f13d5d6e +86af58bf6dbf37da19ee089635c0a841b659689a0ec64e77a26534a9954e8f97 +1ad85e67ca2e0d36ce4d44cc442a73a8c067aac3bc15a1aa339b081f78bdb1dc +0a44ec3c4847fe94c14129b74f978753a2757565bb3bd583f10bacd5b2f2a96a +6b72c32e64104eccc67e3071708798f73f8cc05b336840fa3f4cbc7395d8202e +03f5bd0a3da53ea4de7a317e4f05af850daa05345927a1433120acfaf85b2e24 +fd5b4a0788d30aaa3b96687ff694dd96e87a2538cd2f097a4a40bee1b8106427 +7039d8c207981a4d315b8ad821aeaa8b26ce2b6ab0c5c3e1bff8538928986374 +6c4358ef84bd789ae8ab0cdaab2125df4e0d9508b732a36e4ab64bcf645a30be +cc9f6cf8ac765e680de75ef675706f617b3f55f449cea8ce7f1d3e7156d56e2b +285b3940945e27957a0a4908248c7811edad770f81244aca667826cae47150ea +2f8f55f4f5922146fcd74d16b4f59fd5b6d294af58fd472345d751d4124ad70a +c6f00b54b8d523c69e7b84c34d1538f11c2ef2a7f8acf9ea715850f339357bc7 +f3ba1728d320bef6dfaa1cf383bc2f9684fa3944fa0827cffbecd1d1c8019bda +c1da39d750b6deba3f4a062a736ae34d9d035ebd8ee66e9ad3a7edbc1ff5fa3f +9207dae1e01abe402e5e0b37b418bbd46fac4c047aa1c0ec712e3a548681913f +b51132fd2d4b2bb2d5f74773431b0b47418c7b1658783bbed7369195b46156af +1b6dd3b5c41b2d5d99fdb2d0fbf7dcb9f29142f37f79399f585935da05c5de55 +70e28057c83920ad5535b0020f167c8871ee118910e3591568833c73023ea714 +5f6f13ec777a59b17791c8292a15a8b0ac48865206eb224c6edcad18b198bea7 +e24f4c83a83ae873e010947e53c25e2435c5543db85c444150df4d6e81d561da +ba45936a67c752a9e5eaa26fe974e0cd661b31226140a6fc4a932aedf4c149d8 +5be1e8040e796d1eed386eb3269596a00b93b39c37754105e108b861d66c45f6 +9eaa5046c4614fb6a93a8ade544f20fe63e627642a85b3c4ecb938bf31cfe87b +16109de89780244733fd991c3ffac164dc2cc57f0c8c9387bf6fad5a8773d5e5 +14bc088716ada47bf6504bf4ace6554da7c642b2dba3de2c38914f2534468026 +e8deb84ee2cd32d09fbe5c7a921de89550763fb7ec5f03a24f3fbc91e8bc49b6 +d24ab60da59846b69b88344fb976625f96d0c10c6c0c924b8c9fd4aaf01b32b2 +fd791d73340ea96b303db2309d37091973a15acb75185901de5cd3998ae73085 +107c20cabee5663797edda4695aec83bb01609c0977b5a59ad40720a3d99d5b9 +f0e5c494e741482d5fda5eee920f93c5a9d3adb33cf2c11a0ba4759196a4dea8 +4e3e235645095375b07ddb912745f1a674a51d9d7d83db61c7d07cec47ebee81 +d26ec68f4a8de230c260a37f6e0ca64d37003e7463f3394ad6856f2f532fcba6 +3774e9bcd930e166d1131939f623b3f5178948b14e5fa53dc5dbce7e1c6b3258 +09489613d4dee865265da94f0c652a7aca4e9369fcb70ed5d8e73845a0d03fd9 +3248ac02c1d9f2787d2101424399eace8d0ad08aaec3d2be7d1fcde3a25b63eb +fe6d83bd87053e0ddab426b5f1035dbf1daa40f29639d3e35e678513706077b7 +5c396fea426678021575ba600d4a91f1fda1b916e116f1af3ff0b0b716c302e0 +908d538ee1cb3b270730965424c15ee1085c16fc8b54d1e92e91d870a8184ade +7ab188a5b0c7741fbb3e2c0b1f8afdaccddd93e11e3fb1ecde1c8783ccc84933 +0a930f20c108af089e864f6290a3c9b8db6dab092b3a77583b34bb4ff4b35c1e +0d24f819e32fc965da611cb3196f49d3184f722086a397c6d1fba1a7f7a2e65c +53bef52f912acf44b45bd8d0ade66755f325cd7ac00e72795ec9dfbab4c1b896 +3c7aecd4d799a79df76a9b0db90550762e7d6a57f500d992d7aecf4757fc05a1 +9be36ae28afbeb3d896958f72f9c02fd3460e96dc525cc22466ef40bdd4d8d8e +2ec94f116f46ac31b75644a9f2a146f370442c25342b256759bd732999dc565d +442f0a0d6f857ff5eb31f6e71794000d20ec84c4edbcabd43b2aba9d06ace8f7 +e56f84a239477a1c74e5980f1076d86d47609447354dddebd8ca00337b2e5cc7 +72f6bf66f69cd8e44aeabba9c0a328002e8e10f78ae4387e0f71363c04c75bc0 +22cd12a8c47e310d4a8180412a53273f222c7aa1bf6042e49b4ecf491b5072aa +3d8fd84ddcf56602623a0a633b6ab8ed4c6c9584faf9aea9bde5817367533cb9 +6d3a46ee9cda6b3390fbb4b754f5206abe4cc008080dc4e9d2c3c3726b03c857 +a27dd55dbd463c35d42c94c3fe19ea0990e6bded6bfbafe13131c8b44fca20f4 +e09d104e718c87e434ddd69f348027b4cca04a464948a467daf1d3a2820ef525 +bfba6117cb85a5fbf4d5acc2e1f4033e9a062906b9e081a1e0f075d28d54770b +a44de9f3762171be541e68effc8dd58cef9c18ced0414c21d7633b2a22af3921 +e3c0fc5c04d6dd20a50e1ce0339296dff9d50c639e76e1fddd5adf002956b565 +e76fc7b7910c28fd8d9e354d8fe60fef8008e34ec617c2ca0ebd58b6c844ab3d +fc56257a21eec97161f442b1c3a71e3e0e5514bc511e0a5cc2f01573f3bf2b05 +04ec682351566181c37baaf2084820dc35b4622e47df28cf2382572ca22757aa +dcaf25c1d6a99fdecb771a6751a53177a89b37a09a7ca2c8fc5362eda3150381 +6f080d9992ec6c76ebe5f624853f64cc3c616d0b10fe20fdb6780c8fe47192ba +d2ffa451687b3da05eb0bd2dc2e5d44849e9f76a76763bbe7801054c4a6e4e21 +a3ca6eac9fcdaa02a95fcfd2814ceb70f52de8bdb9c608b7ac71d6da87bb61f8 +58ef8b9c98a90dbfb748342fd3591ab6bacadbcba941118f5c3e5f9c976f953f +f9eee10b06baec791480b42f74ad28053ae3cc34b4e0db319d18edfde13825cd +cfcf171ba5f1209b7d081bc5aeaa77c84fdef58715cf2b7e935879220fa1c88d +be6bbf79c48261a3c54c47d19003601598f219a985ab2ff54020efb18d53323a +2303588dfaab0e6e3c5d916b5ecb9ae48577b89fe0647dd34cb360d5d4c961e5 +201519f518e33e1dc9c6aa4cc07fb18544b51cb98670b89af182059998524f12 +5b007c0789fa9c076dcba9b7f21ea284eb1ad9119184c29babb4bee605a27f57 +8cd2e9f1726cf923447878ba18c0fbdd1c1698710acdfe5b6dd052cc3bd193fe +996a3002ec6d59a20bc8be46f078107e9c7047f7237cf3b64e9b877c8aa5b276 +868e263a7fc0bb0b98f009ac13ebc0ed6d4d1a91837fdc7bd1f40a64d5764ad8 +9bcd070fded5d3b8a1f44d1b11af7c7c6a7f63f06fd84018f46158c2e07b9ce4 +66222e5924f15dcd202f8f1323ee76fc088ef3073351f7d72bb8e24dd4df90a1 +46ef4f691cf3d9191a10873662683002974810af0b2ae065dccf84d8496219e4 +394eb6a12f3c23e030bad0f66cafc2935ba71fbe8988ae16caeb23ed401ea750 +be899c34cdd26133f829ada10f1ec4d87544991e649e923fc40917f9d91c5d06 +386467b33c0f3148f7531e0ae7348e68b26840be0bed4c51f4ec9a7bf73efcf9 +082c545a2c78588c3fa89819c223a151dcb9c19b05ea0736945781cfc506cb19 +497b8e5c841531203de45e56a8a6a3df4158d75e6530cb700f61ab9fb8dbaf97 +448dfab8487cc1b41fde048cc85bb6eaf5039f0ab52616e1ca67346594b31797 +32b34bf06121fd0546809d7d061aa8d157fc9c47b9908c45da54b0a555fe4144 +52877b88e301846c75239ae993dd50b87014dd56309feda15dcb7f42e6ebf472 +58d3f7e1e6a188456640c063c50d2be90afe2a07457e5288724d718d56f52d0b +11cc05625f4645043575deab804056c7a517475eabf6719193d7f4d9000a6f47 +6aa04316582f1adf0b965f0fd8abb4c5c06f5d3dbda2e3a7b017a0d810fcc061 +e574b31645ff906cdd8c7231b52ae3b7c57ef35fb42b00d6efc454b898945322 +47386fd5d68abffed7116ad1426a515e30ce38ef022950f34f1fec130c613f0a +a42d55dbc476be0a02b09e0f7a1c1d149b67b4dfeaac0965990da1cdaa8d0deb +45f39c7dbeb0486af2ca4e1c3310f6a52fc340d4f2739c1d794119b47249f970 +be80cad661628511f178dd4f0617a6b8ace86b45ae9801b3db9c18b6dde80b37 +892ac99de9efffb8790ee35510fcc34f87980d445eb4ac1cddd42dc01f93effa +afb56bfc7554c61e6f72e65c4528f4b9ef6d93266ef1f29ed8cb5a25043f268d +570fd7be2192b237e9f1a8b9a82e7f48b5f81fbe8c971d7a5b063b550f8eb77f +3aed4d5c352758c9ee43fda617923c7a06a3c9a23936505d88325727c4ead7ef +b12afba595f181efa46b9dcf029ddc9fbbaf364c421bd782af5d8b7c463296ee +2683acb13bbdf8d1cc0a97f75a8a50118e004efc7f226d181f798e240597cf9b +55a0cc076db97ae2c41de5b43899c3e315ab9228cf030128a35b2e1ed20b9349 +917945d603efdcb0c708802ec4567fe503434a758347254b653b8c7d4827f60b +29c563858081e8b6c8143ce5611fdd6ea5a523bea18cff94f8bc3c0310c8707a +0ed73856b5da22b2f0dd71763e15e232b85abbf98822b22643bb66f0f8134476 +dcc9d1893280cb656eff20f3725e7cb8bb787b5f94abbac07c067913eef011f2 +7d25666f06424e2d9bf6a108d65400a4581fb9e9be54c92f71baf6c2d2f90f55 +1fd049ca0dd0678ac5a944470686453e3e46c8bdb6dc45c89e845b0347a082af +72fdc391efa4f4d94c7125c3578dce48024050832eba3dbeb6bf143f9a494ce4 +c1a46710b7a46e288fe4320da8203285b3bc86bcc2611c0066c916272b324a05 +88522ba2f65a5510672a5710b91b6bdbc6d1b9cf776d6f7091fbc538fb6226fd +8369c7f07b9e909c1152679abc04e036e59a26eb7312b1f6fe2b8a050555d087 +99806ff619938f6910df77d8861169fb71c676466f21fc697b37613d513c5665 +921a3d090a8f862d0071602ea7bcfac8aade06bb4d682d312133cff1a027327d +0309b100056950f36b19f90dc647a27e65083bba6a92a1155e1e26c0a4f86bda +a5deda9df42f0b10233d7bb88827df41b63b275eac9ad37df77b42a5f875f068 +299ba984ce97a846192061b4ac52147f3dd5a71a91483df0ad04e37ffdbb4b8b +59b4dacaea63b77f53faecbecf57d19d296bfe9ee61cb90221e2605d5c7f0c7d +0b16db8e6c08f5ca6f2cd5de51281320bf5bb910ac6a1238f271152b9ade85e4 +e344457a5b6c0f0dc4bfe08575ce068ec628d60163c5209e3059c3a60a176f6f +b3c16e152733eca6f7052b51376fa3be6f6a97aaaafff75f2609cfd651a9a594 +e4da38d2727bb850780d37ae7967bf70794099f3e98a53f8f41c7267b68e8e6a +02b5b24943bf3ab0659643c1eaad8a905aa8b60c74b717bb07993dbc0275a6e3 +c0834ac98980e228860ef954e563f21b749a407594c4c165e27fd2264e6dc8be +da1489c42411d7e46e813c6498f082279c6d76eee66684e2f3e47d2af5ca2f73 +6ee23c818d67787c3e7c8ef85853b5a993dbd85c30a55e7778cb872c4ecbe5da +2552b834a020946f43f4cb85530e79aca258a4aa1d30897a04ac817665e030f4 +70c44c5e7d3229d900b74729c9e5d390ef86357773df55d31e1ef70795616403 +89a3698e7a773840e0ea749bc429101a3a3706277b9d5282722cfd05278c4a65 +56e1cb2c0df95384a7077ecd6cfcd896a2fda34f4c65cfc080ef4669b69ebdb3 +11efe081ba7a9b427f1385c783783afce90adc901a8d7545628fb19907b8e5e5 +7eb18862afa358b74bb004473a46db2098290aeb616bbc42e57e45f8e65a4e29 +4c6e1339f6ec894ac4685f4ef934180d376b1870005fb5c6298261353d46cbac +49e4a02544dce995dbbe1279685bd14ce709ecb2e18579ca7e00a4bd47d2c1cb +4293449d34fdd8ecd449a35f09e9a49a29f81b0a67eedce32219cb1cdb9d5a45 +6bd9271fb8a7bd49a7efa90f373bd394e2e4594ed6c9053f8346d8169f4897af +862937f5972ad3bbd69475cc04dc7c9eedc5bf48dd225ebb513fbf697b4d1b5f +76ed01d9d0522d1eff45b9971419667f443beff4a5aa003c890938f1c11915a5 +dbfc078e49a3d0fffb8f5c0154d67e61df267083df69800aeef10c7ffa470959 +51938cbaecae5085f08b88a3856c8798f92e964fbf917ca22061e48fe54cbaaa +36d6c6ebe044e5bd11c9d4968b412ddc7f7b3861091089418e41a3ae4294c4c7 +bc510629bec5411ffc3c7093492e24f954cba0539a9546ce19aa4e382c68fc2e +817909d72da92a8b27e9a3aeccf697cfe0eb2c0521173f19ee05ca2f21b34356 +7710dc7e96e4ab0ace1d56ced45f568f3bfb46f735f2bd6e34d2f286640014f0 +8b1e85f304b485451d76fb227ef09a4b2d921926ca34b2e8159a453a1fb50c16 +3d97c183e1b32341ec2d93fad38fa0409b7606ce794cdb68adc6cbb53c2ed73f +b79b09fb572d54eddcbaf25894462b79499fac515adee8d810df922895b35227 +cc4d47024e9564d4838be1e428ae2bd6cdaee5ee9cc39757c9f4a0722563d8b9 +0cd6bd62e97825f463e21cf2cac000f58e3544368300d796ee08f7a6de957dd4 +15242d7a16fae68fc9878f21470f7c4e12eb0f8b074ca3bf8e453feae6ce09ca +f4637d5fe3c34c72cca183e3850bf96d2bc0d69bc2943c70dbd9eac34f039b24 +e99cf8bba2fdf8ce10eae783454521c68c5265525f9c77bc40de4a690820486d +44d974b070ae95c982f4a9cdc72639afe6b876b7b72ebdd33dd58c8080166665 +eeba0bc4b12d6cecbb4854f89eb3822477e72690682267f85aa2afd34f13dccc +bcdbefd6d3f9bb6c3b6af601d1214facf3610e99a18b703403d1ebfd14aff2f4 +e820ffe1e2a62a41c1dfc0bbd53a644e72eb2c5a70626e51e2487b9bac893646 +d88c1ed915ddef523210e81f065e497c043d032dbc137cb64db8b69cd6ac73c8 +a68efc2d414d79d0a9b6109cbbaccf508a14fa3fab9a1f41ec637e5124ee9ea6 +3fbb7c4e1aefc516b20d5332458d3c8acce8d43de26d84ee7bdd0d8a3c0f1004 +4211fd895ea240a1a3c0ecbfd591b2b4627459d34c358fb3b10284eddf8e605f +8db8d1a0db74f1b6688942f8fd627e8072bba21cf4848648b8e3efb238baaf86 +25436f14237738a6be29bc161d7d6c077ac08c39f28686817e3867e1bd4ce765 +c778381db5e7b73b2ab3b7fa09325ced469943334ae4e84cd3e561c1afb72ae4 +5825095ea9ca0a8436e2a7ac856c79278677bc0e9cfffb55c33b5bccb31af272 +8f8fd499397a381b3e8c53f6073c80e1600143d189528e774d222dfeb871aacb +d49b262b12e089d00ae2ced3c5674ae9df1c5067e35404239ba5a73fcc4ca1ae +caa02641010f59fb8593e2ed0f770a89e3387029f63c0a9c4ead1af94fdeefe4 +91a6443f6a736190951fca5f8b9f3e53a5a43155aa46a9ffa876eb19e6e81769 +ca87e98d9e805bfff3dc6551e69e3053ed19fa7210356edfb92e575a2af5a501 +b8e31441067386e00dcf4228c349d0b330b0a763229ffe12967783c3e7fa68f7 +446d0001b6ddeacea49c54e00d736faf87563da2ed2f138d47069ad62ea60ae9 +4d5a445575df12be0fe25e5ddf5fabf17eac6fdcfbd717cf0a6938d6064b5783 +55df19ac1383afdd44259c75d24c9110096df6296c25c324a182c7cd8245bfa2 +a71c80c2b4a5d6d1915e353c15f15010819f7d3ecca93df6d4cb0716cc527335 +1e6a9729a9afa2bebc71992dc2a75cc86333a7d64282f225ee9249bf7c97d483 +40d41a5ae61f72bca0c87c5c54212b2af41bbdcdc300a768fa551135eca38cf0 +0fbfd137a64949ea2564fdc4dd3fcdedcf5fd35566495cf6a3d88b4e56a0b992 +ceb86ea9ec7653638ba175b0fedc9290463fa65aefb7e776eb79de1ea44b4eb8 +d41d8cd8e328447860e92ae464e47a58f51c8ea2d8bdaa238d91b64fc88f3946 +1b8e645157cc0a244667ee97c0132758e9127fcd9ccf89b22d3cc538b48b746c +60e41a641ccb5ea996b71d29c6c4ed83646050c3c5f2cd9dcf7dfeb50ba6ce96 +12a32fbb6f1d8785290fca81c9479b351fa8df646dfa4072fc91e5cec339de14 +4bc8a6bbd9f01b7c20fb59adc81352ff087ff7dbc52efb3cc55f188edebf47d8 +944982f057f7a2c88b54ebf5956f587388d1350ae295b4df480ac39cc2aef7e3 +1f2b56a48220656aaa156a87b8b8b563aa13fe8d241a36fcc0285b4474cb7dc9 +8189f519ddf096e77bcb5f864956514dd8e5d13c222d2f81655eff239dba0c47 +3e69b5bde7413bccdfebf6f0062d25d1dae155121945b172324ad520ac0018f6 +2689762a4abf0cfbb31949d364eed11bb0151c37e1f13be424b70bc0ccbeb664 +1ddae77e15cdeb0eb24af3eb70eb0e4e69ef93f16ed26869b716569b6adf1b8c +59e09faf72796071cef91de471f2b1e50945b798ee76085efc08aad6834b9265 +8a38113689cff9e66c956ca46c15116e328797f826235ffdf6fab271b223059f +0982735aa361b160387df91ce0616bc8507e56bc435d17c7faa03aa47bd69c26 +068980d8db8c5d9db05926148567c49aad76b6a45c21f65c6bbb258d2b5c0dd2 +fc5841a8db23fc57de8918aebe5680a4850f92249a837855c2d72c14767f929f +7a4c4d3611c1b2d8cc0fcf5b889c64fd95d2d63d65e3ee8786ccd7988cd8d5a7 +507290b2ef0a2a0eb91cd35d6d7900a3aed5d079679853238e1c41526ddbae5b +490d513d1f9a0af03cf0645239ed7be9809ba80c1648716aa1a632d0b30106e1 +386e88189bcd8bb3baa7cb96b82ae5f7684b8ca4ae28fc81cf285f27516696a8 +3c9ff8b37e02d2adefacad4c319231203f6549f75dd0bd605d432a6a47d57922 +fd2a9951cff7a9c0c80433c2c420a19eb628ae7bf1d63a13a00c64798144473a +a4321af89ddc566b7c0cfdd53202ad13795809ceada3a7df41dc1f45d912517e +e0cdef7bf9e2f2903dd780aa53e54ca3004cdcee04dabe063be898ce8f541fea +6b6f00940c2048e1087a6e3954929f5bae1be3e74af0e4d67c16c7645a588105 +a86b42c3fe5e4d6a2e35d49a64d539edab0b85aa327c66928ed6a6b902167ee8 +9d83a58b181ba41c9c448ce17f3523faf7aae2e4d7903256190ee6d0d8e49ac5 +147e170895a5ec27a86805396aaf3d02b2854217910236a5954293a6e87bb807 +7781e60cf6ad4305526988985fc285e0d7b1d3fc2b77245ecdde0b9348574acf +45c480318996cfd129644fd6273ead5366176269e947087e18473b7336c2f2e7 +afd30f1579fd265793c001559136529ff3e446c7a2d608b135baa48eacb81765 +f594229743e1ca9d4a8b2a876957cf8f01e0f371efd614f13c310fb678c15c3e +8084ec51fa72904745f19e3f6b71486980b8ebb020085e064cf74eb21a294b12 +5186b4db47ef4f531dad78d576b9fe473379346f280deffdf9989fc0b579a806 +eab8db7690440407dc4799da14c672db575b138b74f7f398a52a6d84ea6d94d5 +1c4642de078f3c01134e4abe61feb606c06dc112322521c830b88d251144dee1 +cfb81afba1a930f28497bf4b6168868612c220b0eb356ef17243f0f522af2b8d +b2360a80475d21fe84ee6f25489fc57176a5f39c562468a7dc1984a51e7a3a1a +9e1ed016a3b9fb8be4ac50b5a43cb8d13116140d634121595a926a9279f819df +9a740dad1f480f954c7f9e0f06763aa9a2ede00e155f671c44c5ae0e2766b1ed +110671802652146bf22e441f041496b2a73501baadb4e3f422aa12d0d9cd03f7 +09f0e624f1642ac6fe58cb277e7abeb5d12d48f28e6c24accdf31962334d872d +0c261e98d0e2ad51db8153439d58fccc34cb886cf462262407c56b2596925f27 +6e8cc4b84cab9df3ab3e1a1a0412b7f2e8a41cf7637961a2c33edbcfead9d894 +7e4343f658dccd29fcd344bacd9a1d4668daa639bae1bdcfd59329fb0443ae6f +748d8d69757b70d87904b1744e8885b38f48729547cfcdacce5a649533ddf281 +bc26dbc2db2983ff0f029be6e3e1015ef4c56081bcf4d4162cd1c17c7a2521ca +da48379cb93e3cf8695259e32de3627841c5658e470b543f016758d27a351fe7 +8db84f0ea2a6ebf26bbab4bbe685cd6bea94cb0d03af70dea5486ef8bfa40d5c +f37a9bb55b4da0a3cf77c36c860d750513091e3367263149f433686fe9e12211 +f29d78e11b4969f83bc4c3d1a9242187a38afd67181fcabcf366bae5d464c6e6 +b5ba4a68fccf0d5cd7e33baeb1a973d353318e534bb55c39f50455bebe7e7222 +d590fc1919a3cc857b7d5ced21ff20f22dde9dc996cbb33d404d69bd8b26405c +6fe869b17d83953157671c1a6025ee6f326d7f8f3f7f1bb08140c6a018ec6c60 +770f8d4518c1adb99b37fec51bd5eded8413fc69dd55edee09d5269400ff424b +a703509738df47bf2ebd5b0d202ba6fd1f33ebad70458c49d2f9b9a293d525c6 +3cea4868138144a2f12b22c46ccaa6c83225bf3aaf54ec671aa5cbd3c33a7916 +144b35b0c2efb1532f3d05b39c2a11faaef5983210c9b233be3459bd6c03ac45 +4d28f92400c2ea1b5d126774c239312cd7e0710b8868bda36478149bedd6b643 +249cc5a1c6053da8586dc854dd4d9adde1b0a7186991022bff7fd1867dfde4ec +8a160b1143d1c2d99e83c011c8faf7474e6b796a7fd3d28544ec086ac9d29854 +1a5885d1f39100f12e8646d096426335f8e76fed0cf63dad0f95642f9b3cbbb1 +37dab158f1d82a096b37a38698be42466cf4c342a9bd336a3399b19e591c7878 +474b515155bdd00ef7f4e4e025825e6f9fb8fc0a26e24d1043d071807a847a68 +9cf4b1c1f59ee56148b5254972270169a10855bf63d6db39e784777541ce700a +6ab48f7739ebf60ecc92f9e794b19afde47e11e4148fe324fdec500d8d397c8c +2dade2b3345262bf1acca853438039d2bc114e9f65b3105cc2255a29a912c9cc +0286c685b75e4a1baa2b73e10af3b4d347c74c348c582ec347928c207f2cdaa4 +1d2a7c42e84b846d2126d2959676c5b65b276cc20c1bd12577e2004eb612912b +3af7826badec8313fdfbde8db845f94197a117b63c61bc083db1889cf512d375 +1f59003c3ddbc2fb750592acce2ee0a1cad08007603899b32e77afa21167e071 +a157a3732db62961e517cea0ac66d06dc22361d0ed59b281ca89f4e96eb28407 +8678fcd81de7971e0daa5bf9e4bb772a22d545f936d5cbf2d337521372f7aa1e +5589729fb1e1643a57acdee33090e0dc320bdc66939d0ee001532cc4f0e5bf15 +d131366b958ffdeb318f2c394068d6e8ae648fb2b996192f15caea73cad9ec4d +0b6dc730ba740591cbbdf981a1d8784b7da041773785dd31fe2fcdedf0f10121 +65c18d1223a4bb411c77ce269850f1ed098737f4327fa196a2755e6c00ab6de5 +c4dd77115e93f4c078de197d0e60f8d9897978e0484fbab085ba3e9720ad2f94 +c8a181085631b7452a36ccdf88980cd581d83ecf1575e5307544ac8f4032836c +93cfbd9bf9359b18fa3b3ec84c33f08704998a205530275f903e1c5eb9f5f590 +c2c976499bf3dff472da9252e990e49a00c66d2277eaeda63d792d4d24863638 +849081d472ea95c0b3891a8ec3ab3191e25b595807b272092a324b09c26fb0da +7bae7db2d1505e63a559802cfac67142e16301a1b4c1f928a00545217357a5eb +98425e9199000c0dfb177e3f0d718923763ebd4807e772eca895bc6768b72a35 +95a91c1545c8511270f94ba25a3dc83a1dc96fe30e8aeb431d035dc37172d01f +2a271beb48576c7348841af80e8e112bc91cc4c2e2dca1366d1220a50ff1fff6 +af1694a1d915072daf69e306dca383f40dd1e999a258952301b1990d19802f5a +29b4f96ec2fb7f49cb1f234994f924906bee5d551d353c3612ff306e880a67c8 +f109b76ac6f1ef52e2d0ca3e1c408a74643729cf41e294f87f1bbb285055296b +f7a6a7d4f3cca0b9a28de0745d567c0f76a0d197e6e0e9cf038f88d894dbc42e +929ef760d05cc55d231f090a215e1a783abc00a9115d1708e3fd1aacccd5d5be +e01ae1fe04a583d8d32589980d1e40414a1c402faa905bc9b493749df4c07d1a +99d1c8f1a8c0f0d45fea4c457ee07241c4d7f62e1c96113873ecb1ddda7f8523 +1a24dac1b6a224ded2160bc2fa2d6fcf6bc939b6733456bcc533cd7379205eb0 +6fd2ce6b7c4040b52535a22d19ef2cc0ae21ddadce8c80b50fb1fac3ef58115b +47190569c5c18bc7851f52af1634de62110c9bf7d9d00aa32f9ccb749d96b427 +fff86c6e887363c826f0902ec5695a5fd9474bf3a094f75f3f2f63c35bfbc7dc +93cf1f595ac8d95cbb7924e7999268e25fde9125e6e3b2cb547c9b74b121bceb +e804db92ce9e9b3ab55b0baa9190ebfd73156d970111fc592e90cb76282d5486 +4411171fbbe196094e60c70589d8212c8a03f8678d1283d428f0ec49bacd9233 +6bcb51e4a8c45b1a599ffd146f7ca52c54578be305dad1ff96efe4503433fdf0 +88a3f0651f201d9e9973a45412f002b085e499c52c5394710ae8591ea1ca21c2 +e4fe49fac453c21f32723fcc5ac066553ceb0d6eccc03794a0985e3ac31c5702 +1240d97948c9949909704dbb5386b5086dc7e25b72116ca698c9d11afbb4ea1a +02d976efd3f618743e82e3415d68de1e6578cf4a152f9f53d1bce5911c92f98d +63ea086abb5c4fc1ce57eab83104666136b29f41c7b454d12fe6eadaa365371a +f8783e0f2d218808b07c3ddb3e1f2408511cbd13e4135c2a0f09cfcd6c14fb42 +64a63faef6e151450f59aa47a30a8730a9570ac95705b980e3eae79e884e415a +d6b5c60b4283893dc26fe1f38acf5eff2c9bdafdfdfa9513f1ae468707a9ecd2 +f7cf41b782135d80d7add1da44b87a2f05f7b8c3122748bcd4db8a4abbe8c191 +e9c3e6b0047d3cd30819dbb495a29603cca9ffb00edfd960550efdb913282bd0 +e7862066cab5d95d5353146a2a7b0ea692ca9f27c661dcc4034fec17b72587f0 +f17e20b36bc5ce9720d0d939c4362743ba45c4555833b23f035063acdcd096be +e0826a868c5d4bd27d7f408748f41a0a3418866b71a3d1b7b2fb01093a1934f4 +7778c8aad8abce70ceb03aea8b2964a9436c1aa04d0157e2324e579fe25175b6 +73b6bf9db002cfd702a7c796dc91bb1384aec9f89c89d5180317e06f80056164 +4e4b14c044f82356e5255cf403e06d94180546af8606e5bd1af4738d3f0ab7c3 +e01d7bad8125f1a466528e407302551465f8bc50693f609645047afdc5700e78 +34c982d6d53d4d56f5491e9ea3902a0e2294ce117a3f8d4bbc89c219aea39af7 +88d8be2a4a31a7912d67db02fe61abf8cf800d7a01837f9b9c8836baeda2119b +026868242be9f0f172ab5035877b16ed769026be654bc9f157a739dd974c55d2 +fdb32dcbdbcf7306bc0c7b8b8c35ecdbfbfbddf5236fb740e7a6fd540dba4504 +5e4264597a493f7a3ec9c66f6d4f820d0b3b5d645f7a6d94051c72266db25bae +0ad7544dc295ccb5691ce57cf65270113f051f63b28524a0c2b48f7b1aabca61 +c8a0ff2a90148713a01f45b568befc40925be949135db70d745a8fa00caaac6e +3e134001f1061aa4ff4c2ced44683ea97fe1d5fa9f092fa57b2b001ae2291ca1 +189874bda89ed07d465fba5564c1365d859219f7491beb207d728b28519e0b0e +07aef103947cdc614bc11b138fffb9dcae3f20329a465734f5ab55e7441b245b +7fd9b54a030996c69367f9f7f10e398b583da4edcfc8a4e9318fbd4dd38f08b3 +0b93c2ed78f1dbf73f81785e7f038fca002485ed514adabfcb7ee77464125fe5 +b7b6a764b613f988859564876ee937b80b0f7745ea0622ef9de9c114c7e18c24 +d8b337d44fb9ea37edf9eacd121ecc3926edb3d9818328d771433cba4e143088 +5f46924f2e23189d6f9481e1f718bb72376577e3746509c9405f49a37497493b +4d5ce2c71e1cb7ae27a5c55822d9a29958fe6b6b45718394ec039b34469eb237 +09477bacf10f3e97d2cacda589866dcb82bdad25ff8f3285d3980fbed3455fa9 +86799ca49832bff5fffde205f976160115c9c5b8d8de9f53309565a02421ae41 +1a01d7ed8ef48f1aca8a48404d6dfaaf1e5a6c5bdab4cab7f34a2d8279132a28 +44e965ae771ad762dfe5aa9ed1b08a3e251ad1da7cec58c5388708637c13fbf2 +f5e70e311d6e2e0157e887f52ddff6fb507f0940f6a4453a10a6a5fdf62aac53 +2c84b04e33252926973923a8e4ee6cc7164a2752f391b32eac744ce36fb72e4b +981a0499b6befed33c25c5923614a78588c96d0dd0d0c772357ca188862656fb +1fd68cf514a437e11facb3b452ff45e5849881698a81ad9c24424c8a20ed48f1 +b5982a49c140f41d4734209385d1c5bfd892c5567af814663c311f3077455697 +b1d7afe54877157c952ec152fbc2089cf9219aab504d7b60f952280a149fd3a2 +dafff17ea8fc52bf2c4a2e56557e881ca6ca9ed52be64c3f1c8b7fa6ff50d9ff +ed932efa0f7d26437f8dddfcb70e940cd9c42cdc4e852de4d7c12800128c7aa1 +4c40f67d061c8401dae61babb0253d1435797c36ba278a8ce56d1cedbff8f557 +4288427bcfbc9d2e5fed3b53c3d3322bef1e42652d7cf68f24a5a66bbe201728 +fc9a3e445f2e8b1de2e0dcb45970be22b1b3e47203a86367e35eaa08f6a717da +2bffaf82efe6cb217c85174f22429ee1204449802bf7e190cb93bdb52509191b +7392268b291489de1b32569f4180960fae7099549e605b087f5cd64e026f2ee0 +b2aefa3c6fe457b56881232ecdd6a8c13dfc04bd3d578c6fb76360ba106feb67 +8bfb5bd6a1da16e1be0f8089c9d524ea3b5c8507c82b80242ee129d35a14658d +ab621def2efc9bc8f434edbb12d3ad0a791f72de8ef6d5edcfc67699d4fdc3ec +87dfe5e41a920bf420e7c5a1be09f054ce246b80336ad29b4ecc69c0d8dba8ce +8f607366d3e27a776f5b8db05c5381d909fc979bc823ca772d8a0549ac796ef8 +61a73732df9777893a609f874bd1eda318f740312a60e49cc39abce30a996d82 +04a81673915824914d8cfe04a6e6b3089f7a5ee7cfe1ec22da44f0d5370985f8 +ab9e776ed77f60367eb4d7d522e15f2725e35032e1dc101763e29a9a639099aa +a20943d92f63d1b1dd9332d14acf8372e859477eaa1d9afd84552906edd4f3ac +cdb48ce1419f0ff4844347c039219f81ee34c21a8e14f5d1d8b2459114c11ea1 +b76d3edf32e3871747d039bc486a6a8fbdd05aaaff427818905b74f6f5cd9be7 +27cd79255fb2b51a07f435b836a15f15dbcc866055893047c1d7114ec371c181 +2a5778f11542cefda2022698c4cb3c57274dba93510beff2cfe08b0e1bce88dc +223331431eca79c16086dedca44d6b40bf07dae7f92213599b1be57b4c59fc60 +733132833664924471599ddeaf84aeab175f30ecc02c4b84413e09513f8ce8e7 +df0246dca43cfe7a60c113e160d75cd93cb17a4abca3ce0eeb9f40048ec1d536 +f9a4ca1fe666995e38a11042f528b963f349935d96c650785a0007ac9eb0dbd7 +e6964bd21c9013c19ac29bfc3331334d3be4acb9758b9023ba1f672f6d9760cc +ba2ba50158a358a6d360e3f362846e0a6c871c026f5ffbe658bb1f0b83e0630b +716eafabafcb12efac2e273a98d150686bed49a733ad8ea635a1961165199ceb +d2ef411b8867a0c06ec1ee1fc060468072068b66cad2f37d07e1f19d3bbfc774 +1d158dbfce3f931d82dc844bf0d5cbe44fce6ce61676990f45e143e53a684e95 +f2c4aeb351f5329685c54e80796cde77547d0eb0b500e07a02477b6219b026fe +ae1a8abe5890fc0080bd4d3f53cc329bb58b73ee896ec5d2719f237cbec366dd +cc1abd47dde8a8e01490a0dd0f3a1ac57710c479c5f8aafd26d42c4f3f57d81c +7cfa5734a9409676983d459e2fd089585748e1b06d3b2576c6c69b5eb327b113 +ca7ed12cb676e78db3a67a02991e3214d3df66e41e9d54124644d6485c5c93fa +1fe3d42c34d70c8f6192be9f4771061c6fde1d55a086e18184a1f8af5c0fd87c +b8c09ea6f8a25268b17c71c286d0325f72a63662dd71089f8819888291d4f35d +063739d777ae4310d036c6461c1a5b3d731c78b26dba37c06c0c6c5fa3cfde63 +2d2975ed4c87f3cf63c27f27925560606f198a8c1a05b78ed50379134b0b37d4 +bbed24a6ed41fe7bf757dda08b5691a851b833785dfaba2b869b448322040fb5 +d7a2823d5ce0ab893b34ef1114ff55dcb7dd5cb77f634a4bbbdacd23dbbd0009 +89c7aec3ae309dc64a7cafdfc32f31bd94ff3e43193f880b1582bfe1cc4cb402 +4741b89ac8e91fd42bb655b2372f13752cb12a1c200c2366b552563f25c17c76 +da62a7c0a5f6d2d7388e55381c6c923c01b690c5c77895022127c0426e2cc14e +44efce3d64e036498b309f08f7559999cee8f31dbf4a6d7f3b21dac914b69c59 +ca1a207ee4fecac84d454812d858cede75ded3bb5c534a5d10d6cd22c68176cd +4ae8009bf4a46e119f91820f418940fe19fdcf7cc1bd2ab5569d08be57a8284b +c9a782f519986779d6f37be3d9309bda4cdfe58e12709c1b4da5e60217ea18a8 +7618a3869f63171e9a5bd0faf2220385dedd10925e643bd5b59844cee442e169 +4ce7091f3009fbb79ae6a32f6c800b775967dfc6098e6d7791d623a9deef46d5 +7a110c127b59d231154223725d537ee13e658ff7141ef113a0b223cc675de9d9 +86e4d5734655fd4b1935f6bb42111f64abdc89f1aac862db984d70ee0a42f8e3 +a2114fbcffe22e5d81286e5c658cee6138c663cbb367fea74b44d18cc6afa790 +772b3d57c5cca6c978150ab6309aae771bc322646272ae0a2d302c69a72c0fc8 +3e6b78c462057f8c89c2e54adbc2be1f11536e5889d97b15d465a2b708cf7033 +aae960e7625d443cbdf267ad62806ba2658d957b12fbc2a24ee9e29f87f07984 +36792858fdc8f06037ff810c96d9b27de081bf5b2586bed7490ef219b04f265e +a2288459fb85ba2b6c78eafb25d788b1bcf5f0602efdbb61190ddcd5b59aa603 +abaddf4531d841f38543b3acc2ff01b48f9f5d00b4ec9c2d253349570d09ac72 +ce2e97ae718f6c6b5cb537b99b511ab73f5577cb52ace204ed57c5971143b441 +51591ff11f01bce595cc72478a78306b38371cb4875c6b0d33686ca1a8a52393 +6cfc2c7b402ec7b9ece8568cec36478f1ae1c0ce43d279ff1b4b9b1365a912d6 +0ebfb0ad538f162558c170e95e72f357dc1c8f4b90ba12b54ffe397fb7124c9c +dec9cdc6785865de9f5cb82e625f6601af7440d5f539443791d661bc2cc3e5bd +b90b1cccb6996b94c914e9aaa672393aa17309a77b77b1500fe92487b6c0da77 +7431a716e6eefd4b4d5a3b7536f1208974e64bc785c8e66a8de6b7db5e13df8c +f080e33f6c5dae599da7e10b8b2c14b328a9af3f1092f2699373c8de5a4c0e74 +68890c9b9dc3efedb81b3d2c5d4a66c8adfc0348bc567c724a3a27d34505af72 +7a5c400f951400f682eafec5246ea878f7a806d4102e3e20839ebc819a63db1e +3130e1de4fa47a4bb3976388e0ad80d5d2e95ebf1de12a2a3484322aaed6df20 +133e0b0bd67b98f2ef8fc13d36b768ccc316fb51710a9c8f8cdebd709d4db09e +385dc1a11ac5736563c40767d983e5402e2ea85032def60def5e9d0476d48b00 +687778870d188a7974bb41feb6bd0c670e5340b2e2f6378395c1b9ed92e174a9 +73a53b0b15aba77f314eee518590c837559a0393b52a09389124f1e0fa6b2e5f +099c975d37f1c9422e16db0e4d2903717508a550bc54ad27be2713d8d74e276a +64b0cc9f781ce022975807938574ebcf9e45d07074b1c93e3b29e1261c5ab6f7 +e97e360e51cfa9ee2fa7720c8d660faf5246a4fbb0c8d52c84e0af76b7be02c6 +729e77471e3a745d5bc1b956b26f7c2e678823154997325d97d4d76ca1deea04 +47fc0375071435f151eafa8db1fcba9c8fcf2e1aba49c9d2a1395f157d7f2d32 +3a8106f7e4d9a3d49a07a8263766db69393f8041e380c584fae19aae653b3e5c +96cd7e92a6d6ff3011a2147c17562cd2de5c98e619e2b08865ac8df3b7d4180d +18849eb8b958a693100b2a6821b4f82b8b341c38c9f8d5cd50fa2a3f89ea7bcf +c780011059e6fdaa7e892bcdfbb0cfb884e704d549c3394eb1cb2b753240df92 +3adb97c9c9ec4c8acf7d9c0991716c7eb25437b9cafbfb5b57feecb400ed7c23 +45f41535db18427bbebbab0a124ccaa4b688f553052001d08945ed8738b03039 +ac96b39e1dc5f539032ab0ea137f9c9f923a9f04d4195174171e692a9d3222a0 +aeb6933a55bb73b1624ef6bec607e3c393b12649b549b20cd19718c6ce11e940 +7872988f8759f214d9f93e81b7b6b494247637a325a0d891625c3947cac82013 +f45b7b27d906506410048cdd20fc71cd0abf45af739bb3986ee7dc28e2cd1502 +aaad6079a9ebc6031ce76d89f170c8fc76a825d9a318696dcd028d5070929db3 +d23c777d0f6b79ea799659345844b116cb06f43b5d8b88af330a10cccb374417 +701cca74fa1dae963b0949dc583e4970bad2f27f5618a09e700b763f822819bc +f6184c9f5a5c919956ac39d4ad05118dd219446b056dd0dcae51730bbaacff4e +34af8bd6d8ffcce846918a73f22f497645eadb0ac708b912c8a189ea959693a2 +7dce1fa10c9300c532ad61a3b7c57f5193220f2559215e2fc5b7f64c77503d27 +c21efd335d60ee1ca25cb7fc5581e77c2a595310af20beffda320daec33049c1 +67d917f49aeb5d9985b9f7bbb376185e9f208e3062b77cc9a2545ed27f779b75 +16eff1df7b6dbbf538c53d2862c848da9693fe29166753ea1f32d25544b6d86b +01ce47fddb3920c763b17ab0a5aa17ba4ead8d651a4b97461dfbc5afa0958c3a +82fa56e84ec7e396f97806f0617814b2da473e6475be1d93cefa5216ebb473bf +10e5640392798bc6067e23dde163b66165b9e22663886bb43e295fa2886e60b6 +3494abd89d456a9228a060af6b000ef9591b473b895c83929b1088915dc42423 +1ec0199e01f4d9300bc2367af6f1a2bfe40f32239239842bfb98970f5fb599d5 +9871c8c4b70f3c8a2167a56dcb70132d058f77460bcf36d8cd154622c92cc38d +82fb88c35a2dee24d11109ecfa65b9e3ad6e10d37f16538c3a87cecf69d5ceec +a12847e0b9ac4476d831d8fe2d93af81c832eb575ca3b24634748156a3275fd0 +c6a868e0d22945bf46ecb0df469c11a30485da01f6dea71669adbb95a9b3f8bc +423c61f1ec58b9400a373e1052c26debdd5d55eb0057900c3dcf41f54762486c +7cb995e1362c2bb21640c886e36641c907c14ccafd84f307a1f74b0eeb782870 +79c0002d3dfbce81fa061affe5bca46e46768bc26ea3b8d1249107cee4669148 +f74f06dd877c776a93b14c325337f439279b23ecb982581b0ce6c7c7a294daad +af3df9868b72a494f47c4160447f86cedcc7c184d75b0744e222c242d73fc595 +3b87cba950eb3c08a2e394aaecbb304f4bee5125bd2fb099449c0803629be6b7 +abf53391b8ae0772fa91df7a3889b853db596251518ef385632332bbcefad1a8 +41137f7c7f91592ee93d10da9ed9b848ed65e4f5ac948fc2d806f915bde5cc4f +5eb86e817c483c9cab68faf228a9cb1d7566ba573f32d5d5e01b2f2ca42d79cb +1b8effb13d53c175ba61d78f654c503580690904e429f29c4a8f1df551c86a90 +424f6d87bfb338e5f0f6a26cc7c025c2b7b2eb4dd5f8d7c46d4a7b3b5251c60a +710ffa6ffd1296d9957c943841616cd3eab1dbd853c4727b01e5c1b0c2049bf6 +20559e49311ac1bc57e553216cb7ee53865d4b2f4416d26e29cad3914d54100f +02c5e3497f4e7139a92581f1f81c10b76cef2d2f0b5afaafb4503ae3a876964f +ba4a2aee3d05fb1ee2977a42f6fddc33c9e39f1fcb442226ccea1e6ca561b617 +aaa1909229eaf73f7c2be639c7d3e8c774664c64a61a2741bf327bf00c81bcd3 +2b7222221ef1f49b38f0b7d5ce04a312dd4c56492ef4ca45fdaa57f419a4f2d4 +69c001ce6b95e03c09ccb66016c70d8113439d1c0cf5b1d478591146ca454255 +e1a9432c61579368568dc1b476cb15c0f57f2605b53ddaad08240190553ae1a9 +53cec7ba2dc54fa46466c0d6c55a5869851b5a8617e4e21962f219b1956c0faa +f1e96c924b3fec22370fff308268f39acd3a0891cbb1d6f810a28d7e583e0f8a +b4d5b929508742e1b1d80ec182af65aa9b3f824752c719f7f868ed122ce9a671 +ff781f5ccdff7a07a3edb2bb9cc262183bd0750b67d1cbd8233425299482974c +a7f3c0d3c532c18eca2aa656b6046703db5e81e8ec3983ef955877b7345374a5 +a932b9c6eb071075818a2dd348f426a1d5250c1543094dbde1359da691bdf401 +abed83079f3d5a5477d8156a4d94ea455cdb64eddbd93059658c42befa801079 +8f145561b188d4c37ff7f3d9fed9469dba2db8fd2d0f47589f2134c5270fbf25 +364b923ec382e6623c6fc551ac6a1f26d5078dfc4324805a981e3d788ef6ed03 +cb67ec18fb933e4d9887fcfa1944fc7aab5cbf0b6d67326432569810418b9ee1 +9f0ef938670382983762a7859641054243c89257fdb5b2307784ff8e9bb55451 +b73076a77722f691ef78f4198061dddf6572f95eaf2c45b7a724b1808bd307bd +089badebf6c4de916f77b325f5ed8adbe11617bf6cf635cfffdb6a25de264e1a +037da7389842040606552c1047bd8f583b426010206cfe52b8f1ba70e4885b7c +c47454f6454564cf2c6e624cec87e29f75666a814712dcaaad1d723702b1dec1 +df8b130db7321fbb9346d6b53d7b58e9f6d06afc77de17b3757a1471ea507869 +8c0078b31019640c61301a28dbe9931d449ae82cdc72b16f25afe678d0f7fa35 +e19b7b9958d675ad8c6cc33d0f5c2705f6817486c17896637fa499d413ce0233 +1efcbe883b2eeb422c2312211214fcceb255ab0e3db174399706930871e32029 +9d82a47201944ba330a23ca21d957d6bfab1bdb9b2b7e3d1e239c3811cd0c5c6 +ffaa91cb56f697e8c9b9134405a1f9f8d6b430617fd05ce436318ed362becbda +9e37bbcf0fe8b4c94fdfbffd4c2f2ff6b6978279edb71fa37d024193bdd320f1 +3c5a36956bfcc019c207fafa4aaad947e796d932793ed299e7f4d0b5452eed9f +ecb4f12693b70c6f51d52ab3d4642308462da25786bcbb3d0fa44ad759dd47e4 +7aa52b08c68ca8da95b9936bdc052619e283a834622c2eee9e6cb0aa8e97f511 +170aa3a9bde72a9a8bf76578bc5734df72a6899667ba324ba546214057ccb8ec +655edd87e978182f97c9db2386d48aa284c08c17833255e1716eb0229ee953e7 +dc630a523ddd69bba1395e8ae1e5c5e09b3bb840b732ec77cb58e0a409f723ba +7d10f22d2c115ac2962dd092cf457a40025c75be8fb7116c9b6a95fd46ee9342 +1b70043130898636433abd0135213ed6891a07ebdcf405b093bfbab82bdcb692 +ee6845b61efdf0e7cb4aef0998b116e57e294c72b72c4629846b743311d3030c +a951b9a6204acb6a254c328942688b006c73daf7af93d7dfe1d4c784f8a9a51e +3d125f02d0fdc7ed4ba7ebfa4a4133604afb514eda30723ef39a3ec9b156b69d +99bf982da397947eb0999292e909c17c78d884e18d3e59ac03fd2aa7f0241c7e +c10a00c78a678ee38cf9a38060533d7a732ce4f31565f1b211800e7e58aa0f76 +ed6773eb8f90c35ef22f222299edfa17bcfce15a88c48f63884adf771ae49431 +2c4cbc2059a994f5a18ba3625e5100783fcefc82b5216fe338df0c2068487d62 +3376310225ef4ceb43aad95add0a3b5cb88685b58062394ae6bd9b0501b3abe3 +7ecc08350e64b4952b562f739f3e33e50b13b457ddecf1615322a3a191afcba9 +c1afa9a8884350ec36f1ad5ebb15201ccddea7f17c0240d75e49e301cef3ced0 +da34d3d8cc0268ab89016204d52f4d556510de4f136f27e67b01cd9f782adb47 +ad01dab939f86eda47450d5e9c751ae42d3c4b07d66edfe21c86bb0c72aab84b +379e4ea29f4caabcf6f4337936f1c6ec8bc87109ffa667ea4eaad64db0930a2f +39a3a148ecf1fb9d02871e03e747f24dbe7da6b48297f6cb2729753250a5311f +2ec22427d614f7ecedd1dd2efdd02591e2e8e4dfcae8e213f77dac362707bae7 +c2d781875f992eac370ce0faa9e606d8d0a53930f3d30e5ebc6edda406597d4e +7c5d8479a6c7924c1ca56ed949fcd1943a83c100bbf1282690997008779fbfb5 +e6168843909f84c7a2cf6dccdc0a0b4670a83d6b440a5434d2e01aa9af41f61d +2c53b059277db425a59d965ced106eab6de497c3a341d3e3fad9e5e335ca7d30 +4e2c428da8f881ba91fb99ac07a39cf5cc407ab52a91e541e0f1ea6c26518235 +cc064089fac9282110c20ff014c0bb53299c77a5c93456c3d6671882cc7af79b +e5dded042c0ecab6afeefcfc21246ddf69fbe3a6ece49dec01f3f047f03c77a2 +89d5d85cc435eac6031a322cdf84c458bbc958ed4a911b062027a1d26be5f545 +7fa7ebfbfe8d79a68ff506c27ded2f25d329e1edc76b2cf6cab5ae8b23a8fb56 +5f3e7071a8423a64f19af140991705a8284eab5ee0f514bc006f823a49d9b5f5 +45de832c5d70ec379aada04dd165d1c548ec9279d3fc35c5af4bbd22e78935c3 +8755051ccdeb2ff4a537fbc1c37dadcc949777914f7f14af2ea9f2c5abaa6cf8 +e2cd5463424e7c2ffd6cec1c81621d20bfe1b7741ff1548574ac299b2e65eba1 +a7375c8bd1d8b89f4b7cf6587583358f2f94e865d05e5ba61fce8e00af28dd5b +ffd7a21be0788abde5889973eb6239cc6d1c9fcb2ac34cf0143c5d758e67648b +a03704213d72284c28655a19d4613f8eb7b49d5ddda331a934eef5db3ce9f3f4 +2debbf2d7ebdbe2714509a0b18f37c5265e8dde83ffc3b3b7d541467b87647b2 +a3db1339103bfc259b5ef11a65b5670b8954b2d6d8196c4e6f1804575e39d976 +13e501ef704a398aef7fd17163161176b18b647a36515f02d7e71b90f39403ad +3451a4ede401fbd7e6a506362d7160802e28bd923a05efd1cb46c888e526e789 +e04bbaeef4aa47a881cfaa7bb09f1d7258481bb001098737da165b47855932b4 +12a933900479150cf873a4e77ed04015b46aad15374c4882d070914739d92639 +f67d7fba4561f0a4e49260a060d37b1e10d10a6440e463c0c2d69cbc7ec0ecb9 +c36406c941e49729f9be80bc68bbb34a61c9d7d5f341b8102e1f006c94f25891 +2375de1b55ac345a646e0658802a72455225aa7b400e09b93b2201a77fe3a40e +96b15da74aa86e8993ed0512bcc901d58ad37f961691131e910c129770699cdc +b800170293eb62d82d971a494d723b31615691d78b379a35589feeafeed9f095 +39901295aed33206b618dbe15dc11caa383b499ef9b8144c69f393558ec81744 +d77af91f7c5fd7197936600222767c29ce3a09d72882e0feff808a01095adafb +ae9e7012bea2ea6fdcd7baa078503fd03afd96d22dcb7f0a88c1655f8288ac7d +7ce54050e29c43204e51b0660a21668683024edabcc8e0ccd985d9373a70d3aa +14cdb7403bdab40b2512c5b43f510b763fee6d8efc3f3d52efbd43f862c3e270 +94f50caa126f720468e8e0d3b6058451c9924c80a6b3169e4923d974c511ea79 +58c2fcbb1ca9582314979cec3d314e5acb494e49ea42c55967121f85ef009252 +5aeba9bb553914e2d06b4f5230354b04167c01c41129c4dbdcb23f5643277ffa +b1d5739621347e154c3b42144cf92353b6589079ba94d8d2ef2afe23d9832c89 +92df326014cd3bbe09b8394ba3cd6ba71b96826610015a9df9080dca02519bce +a26cbd16957e7bf2003f239c83c1a0446b65d05d1f4b537f062839887e424403 +c2e57ba56c8760b7c53a839ed577635c21f3de2d601a48d034d6b55240b7ab26 +2f7a3c0554e22377ef66013dd3e2617b04ce637c94e22ddf203657e7d0ba6640 +fb1759f381a5f5eebcd590efc472e47c63b72ad8e5a503a9eb5fa47e6edf1aa3 +6998f17a59cd28354f6822549a49bd4ffd884b0164165a38a796b167edbfbaaf +80603641d4ecc43492a3195dc8d2efc326c9d2eb52718bc012246c5f139ac214 +e072d1dc0f1806d2149fd9e567e4ba6e0408b76e62e46b38b3543bdc3880c438 +a3f2555889532f1bf1e71502605a316895b2f51314ec7ed6606df58cb1d2b5e5 +c85059dcf5e45271e991ef2150c08e202cfe5d27a6bb81ab74f40dbd2e041af8 +09c19bc4d2bdaf8014eda9d5f4f688a1ec7945337028c3553238a8f20c0a5650 +ccaa887b0b663a0e894790351accbfa84c871a5bc66c2fd8e4d87813baa7694c +d1eb125bda8110e193cfc13d2d25bbe9eb65dbc425b43c9a46fc6cb2b287d577 +adb31aedf7ae61b6ca25b063495951bd41b40039fc521bff42df6611dd906123 +38d5382e11cf3ad6476b7451cd501ebee0539453d2808f0cb76f5637da63074f +e4446d9caa9e7af06db07bf83e81b96c8e783c48d0653664f47043e0387c8be2 +f54cd625fc692556a3f8aee08387a851c153b8aad36ac440eb54e7c62d9ab800 +b96b044f5062673e302050fd72f7a6e18bddfb71997dc5ee39a2d7af72afe432 +98fa8d47107c569cc19af8bba0c6d8addb0b86489a8b6feb4b4e71906d0400d3 +7f053ea1bb326fcb1a68239eb0d312feef64dd166cfb75928ed8abb294827d9e +0c41adc94ce8806bbbb55b7461f9fa56de5895d32e0572dc9a5818d0984d1b18 +5c87e98e4541a0b55e3d0f1cb1831b93e003cf35063ade92502e6379b53c0fbd +71af4d6665d594e5892fedea2be709001cb1998ab8363c81c0ab01638690f8ec +f4b01f3682ac126ad39c860d4841452140261bd810b18409cecde695aa96ad79 +4b03ad1a2c54efd56f1f3e9dda5196d9f4ed39b79a4284aa2218c569744ab9c8 +595a068b62fc00db961dfec8e2d6f7d4c7f5aeb4820db83ccb13636a6ff01f14 +61bc90870c0728011212fea446e91e3da2cec07f896190c10751856344e947ca +64b1c8c202afc370fe9098d1f355e180afdcd39eebcdd6ffa10f16b31fca19d2 +0f5a9d1884ad420d48bd6a767c5a5cf16db2431d2c728407191a02d4ea466017 +ebde966b607e1781e6009096cee041690482c3cbf99c775a1eb06fd31c52914b +a216f4df946f27235301b2e3febbdc989dd99071da245026f359bd7e4b59cdc9 +7a23806a80c95a7b346dbef40b9426409b2ccc7b6ef300cd2e21fb2681b136b8 +88156ea569541b40115cd983d9f4f91ee92160f8d5e7c33aee0179f8a84649cb +28a93e8d422c208429df5443c3f3f536c44cc93677a99759b55c03a62711e468 +9b03b5b3a641f8c91bd69ef384abbd7fc1ea758f6cf7971dce353ed8240326e7 +9db249840dbb2fdd6c569003737bd907d36b84a5c900484d16801fe467998ca7 +70ed9ccbc66071fbf415b88502031577444c7aa7d53e82a0361dafb776112062 +7cdf92543f32609c353e765b012db568d05c31144d6c4964b4a713ad7bd0dcd8 +fbc12bc889fb3d8150fde2311baabfee3fb762da5021d852fcffe7081899f0e0 +0b77740efdea460b07d98b09d54593f1f51742d7e089287d10fbd28f99cebe61 +1972942d34f8ab8bcf1cf32ef039a82f3c64c3705d88839b1407584d203c2b49 +26ca0a1968cf4bc9bb0c0283bab5e8555f3f8d2a537302f24b48db8615ece41d +ed47d52b140e51ff466a4a193d4d4d27946479b48f9db2802c3840709108d0dc +e7bf7acbc5660c6c536de5420c69f61c47390046cface36c0b2fead9e5758aec +874debde408dd3f22bfb8145218c00df618dc688bef3f5c04348a0fb9d86b357 +019d8f9a54caf54df2d99cd42daf2a003a6306d2800b50ac23d6ba7caadfa41b +8d1efe2d4cb191ebd7f08a76188772d230fefc009344eaf7844309f980dc491b +52a86c7cf89fed89735ce8da32b0c6048e263b9d5ae5a85ab19a5a00c77ba59b +df189bace51935d6b17035b1dc15ced2354cee565fdcdb115c6acfe479b491de +e9f45c66b784104a90ae947dfc601c6551eee7809138ae9c937d682c6195e8cf +f02383579b57ed826bf8c736bcf9580ddaf8bcda9ffed72854aa4d46ab6301c5 +17200dad22c5279c536776f132ea3048d7cde91f74e35fdef34a6598d10ddbb5 +32ead108423bcd3a15975e0627f73a67f8c69136cfd89ca3f7776b17252b1c2c +53dfc35d305bb6de8636daf26cf167f1fa83649d4ff7da82ad68b3e137284818 +d1842809448df121ea395159cff656150fdff6dd54428874f7712eda13897523 +908f32cf4db4d6f57cb4648627b7e0fdd2fb570ee190febb1e55c5dc951ec361 +5cb021f8f6d2d87b5adbcda6b7a94f94384c24430503f54e6e93eee092d5d78d +acc8330415fc7314119d5f3e968db4ba1cd94fd7621baf51903d4bcbbbafa8df +35600d15fb97f1b630ca7b5edd18169949458ff4d2b7b5eb092e6fcb0ee5833f +16718bad1cbc744cf045988158af9e2ffa02eded398cc08b201f6a39a8a20f51 +4028ec7f3f6e3757d7a34ebed1900a1ffbfe037cb412584d8b9fda6885839933 +46249da9e3ad06203a617269ee6ef4713629e9b46a8c004a576b761952735a99 +d2c58b18c7f724fc92031de6869dfce8787b6330e4ebc9c01856b977eb6f33c5 +6d30507262e858d7416dd6378d8af77c1948f0b411bce1a268dcd34300f4ec9b +31f78a8895f98e876174a147a55c5414eacf7cd3bfd890d414e1d1da8c27fb39 +0824dab906585a541bdf7c03c89216cf208f294fccfadef718940be43e8fdb61 +f69f1db424770f4c0968895cce8e672dc25e987d402138ba4887b083ab821247 +29ec90f84476b22068b63ed8d82f2ee8597b9d59af418f52ed137fca3a606807 +a440afab4e4bc037b920bc3b3d63814413a0bdbaf2632a90abdb1d3b34d857e1 +b166435d26e62b7e9bdaa5ba45c861e810d8db63cbafd9f2ac02514dea4d426d +46af53efeedb1cd366dc8d031baef9f893b2f27745ba12096c8d18288486ffeb +934c00af139fb1784417969cb765dd19640814270530dd0a21ed88450f041fb0 +2aa77a090823b10d01e1023ff86c2577f23af0b4e10abf5a2baf069a606707c7 +1dc51c1c5b11d02c49077676cf1f40d5574add32f18ba6aa4b5fbcda80a25fd0 +767b85b0f4eb55d9a6d1459a63dfbe482298357d737b9e295b18b22f05951948 +e516e6427dcaab45ccc0704c8046b878a11bdc0a86967ee40f01320b161a8d58 +cb21bea98d630ebf1f192080d22727136237b009f85c86b9bb6f69d982e34fac +463618c9d140a1647c57ec49ede38ba1f9448c766c4311f4de107460e2876e74 +033286c61659610c3606af2b93007e24ede4d64e0b0f5df99ef836b5d75d6426 +00b0f05b0b9fa3dea1e7c74211c95467975ae8efa02f03eeecef8febdac9ad0e +5636a91797234a50a1c7c0c7c650a58d4f9e249020ca552a0adc3df7f1d457ae +5ef398cc07eb167b28560e33409c7c5492069ac38f5b0c6ae24331e74c26bda0 +a109a097255499032f8e99b116ddd7ca99f096996446baf129f6c2df2f3f751c +4ab3cc4296fee78153d60d2884323a84260157db0b83a72309272f109ad9dd32 +6d9938d20e092e89c5c8b10d9cd9f1eae24a3b780babeb0492ff0f7fc63fdac9 +1b45f9cf8a3b4964102558caddc29ba967b36d1b13c22dd89d1f37bbf7b836c2 +ba6406526c97985b6e4de37ddc8ab5469297ebc782aaacbca90421e5008673d0 +2697c2453e57e8373bf02c4761e948c720858356ec239a22f28f58474de6e462 +a64b0b9c82cb7c840d9dbf9a9d3689e566f737b09c2404eca0fcf25de22d6953 +9909e06dc6a732e3cde85c291081a1c5957b2c9a39859abb6ca9d68c1e00ecf3 +b125909927019728e88900edd66d5e1bdc83917b127452b64647e5d996f19a55 +bf1090e4a89922d92224e63b2ed79b24268cf92be4a21bc6ea7bc3afc180926f +39b85cfc73a13323d33eb4f68685ab96143792cda55ed6aa7cdc60651da24d1e +f9fd902fda72b31b3a87ae2c62e4b6ee5d64aa1a1c827d78497eeb3c2e2ac8e4 +bb3b87601a2cb367d2c70ce98552d7ceae1190be52de6afa7d6fca55d1a98e63 +706a96519fb104c3f86b25f0f121d2e25a2c1b449dda83d91dfa32bbe6b58262 +4985beca84b1272a67b7c2785a2d7441cece3fd6c3d7fc934e45bdf3f9825e70 +922f47d7498e7bab40d66731e843bce78a8bbb212299a4f9b7fb8bdbc51f7d36 +693d2d94c81610cc0811148dc54c9261ba75a5bde19a33339caf987d2106d1d2 +7ed8b2c5407db118cd72363d965c5a09d194c3de5787a94f57d2bdb493e6cab0 +fc392a3a4373eb3b5ec958c0638d48947cf11b0c4632da94e0e36a8993fd5dd9 +f52db83345c2e8bbb2b5f448c574cc8e3da572f59981cafce3310b16c3c4e9ef +f4513f9d2e051a3dee4e4da71034c72b99d62a81a97d46aadd9be63b41ac6098 +babb445aeb40a48ad9ce7c1e6fdf741c3d5f8ebcc5695a66d9ffeb1db4a505a8 +8d1db6e351895158ab17d482ce481af1230b851648088dd4d291e5f0a3a7c5ac +dde0c65a169213903d064ca3615f34b0fe92dbf1e380d54c80e2c52df69c9c9b +a25fcfd3660c494a147ad37247ebf38f51ef48dcbc9fc0e60cec35256d899ba5 +92848ce1d291c3c1aa0e5aa5b21491bfb2c5af723c176b29af98b4e4a57391ad +82a82a6b4dba188469243523e4a6fd024e405d66d0aeb4589663081e40a5993f +6ac22da0da8430f8aabe88cd3af122c4272279a792bf50bcae6961f2eb687b5d +afba1f2ecb46524d0dd327765370cc48131eb91803ca3e902f979fc09f5e8532 +50506e2ebe7aa2375ac714360d57e55a16b734ec4340d738effffd641710de32 +55e2c2b106554f42fbe3ab601b74fac94d6ea07834df928426233bbf558713b7 +9152662afc5b0b642a85dbfddec329585aa5e65c32ff9e4e9b25e98bf241819f +d5cac87f4c205e89c74b3f28c0d88537a676b23bb8b32523dc58807d1446ff5f +abd99216aa8777cf7315f35a658adb310f4b08d46fe20571a70bc0ee91935cf0 +1e9ede7d8d7a3d03fd080758fe7f95d77199018874f730e9f4362b262ca51855 +15b5d8e2df59db952a68379e922a3e4c13226bbdf3266d86501a22fed8c2b1e9 +42a334097eb92c051963ac3cf3e6e10ce648f5b2b7d3c5cb99af759ce6db8f0e +11aa75f3659d8dfab95b5edfc753604392f4e66772da292c681118092938ad01 +8242101f9bd09461d4a8071eff46bd5c2853e3e4b1109ebc295703fc45bce8d3 +0988270f512a54e00c7978aa2fe33b512f6db3744433a19e21d22c95fa7d29e6 +5f5f41a7d1783b044ca77c75d8c6a9104a83c9c9aa9d1c5f59af8f27f3427208 +1b892e00e8ad5ebdab8793540bd4d1dac3133b0f092e789110cc1b4b99d9ab4f +3678f3a22111b83e3b1fe13f3549d633c4a11042822fd741d81f0a539b892048 +ff9358cee4f94e0b4813c4424b7fd7e5931dbe4fb447623d5a0416f01b013b52 +2af29573578cf55118c8c6d3bac95887ef7c20a0cd0b782e608b80b92aff0ea1 +aa86e65a5cc711b52f383d0ad8bf9bcd32d229018267c32fe82393c55a72cf64 +d44cdde2e095ccc3d4aeb91baf6bd1cd9a9a7c0b8405f9ef036c8a7290a79af2 +45d0b9fff907dbb9193e876877c836ea2d04ecf358f9b2ce920c78d6bd0ad61d +70744b835941646b11baee6a2eee6fc8d7679081647860804f51166eb9081942 +6c82dfa980ef40dbfb69ea169f040cc084beef3a159214e2bc667b4b868b2d5f +ea491e45833a13d64e46e6d210e32d8a7f115e0fe8d1b2e387e6586d3048c1e4 +c01d1cbe86db25b5b1dfcf66b1e3147485b0bb1ab0e256bf40a642469943160e +04032ea20e8b0594766e67a6517a247e059bcd547748fd6cd2b659c59a138f66 +95bf117e466468f468056019c713372d9696d4e9f3a08fa75f3c1a285cf40991 +4a29f904c36e1c49c937ffb0c5f1505488b88aa9735044406c3758165691ea48 +92196088312828a405f41d398b7e49afaf42f46a93b710fdb83fb50d52ce70ff +3e6aba35d3e71b8ba7185968d56d53b60bd58159dc138b45a662aa748047eb9e +1f450281b03acac4ccd60e98d9573abcb4176f3126b9503a6de7706421700da3 +1c7c9b58f1fd2dd2e3b0b7c383d8c855a479279ee0dcb52397dbf842eb4fdc52 +d9292752dc327b8d445d4079a43ed7eacaaf3cda2cfe41ae22ef1ccfc1d1abb3 +cf5defde2692c92c514841b231404db099ab1278e1a3b6bca120917f6a99c323 +4d0f0aebc30fe56f95e4c601e35c4a97c190f22ae0f79b8d487e6007e0fa654c +3e9751556d4d6409c8ff03b3e580b5bb476feb83d9016bf6fdc9965489753c1c +78dcba76bcdd404f4597c6be549455acb4b12597f0675427f4da3a48ac5d7af2 +533f8756be1d7b4e7b5a0eebb7a07b7a629a67a05b7427185cb3669a2d4078d3 +c47329c7d747c80316273bb726e6ee87bae8fa3cf0ad22dcfd6d9af692459d07 +9c184685c9b3dd107f3da78a1e45f11aedfa94c260ecfd6a72149ddc84973de1 +c9e3b833d06ea15893a606063aed85321d878535f3c9587d94bbe53d7444fa1c +9ae37daf14317e36669e2f0cfaa1ff789c70b64f5b076c208a8bf692495bb5d9 +44c7a128bf929ce2e116ae56559cde6ce518afef6096da9274493b2ca1e6b056 +5950cc57aaaa91a79644f7f510339823a4b602bfba9c6640b8ff07c1a41fc06b +b5e9ec2fac3a9dd54e2dab079130f16466ac56cbedaa3a5e0340ae6cbdddb034 +6c964e0afc359e9c0e71fc0805e9ab587b6b3db8773aebf1b42be95a5f9c277d +eb641b987daad963c609477823bed50abe7b3f5ab3ba5c4110f3f4651f300fb6 +7a126647dc3d67f451311abbadc653150d192284744132903c0391cdf53fb775 +91488054f9ad3cb2298acb82f976ef320e14bd14c84a882f87c3e8a84923683f +ffa24ec2bdb879d45b685b1e8f53d814c802cd7711d003a25b9b4628669896b8 +77023fe69dbb31dae4bdbafde42f92d45a26370becdcd5701522928b6a73f5fe +346d1a913ee4cbcb0311fa8dc77e65baaf8d95249b77581e707cf6cb18e46c55 +37afac7462c2baf81f8c19a65d5b79a8fdfda9e0e701e7a8b40e2721f1fd3e07 +7ec2c819b82041ecee3ec8d1202e252febbca1906c8e56a90b5bc8168cd28f4f +2d56ac5bc7b4e799049eaa473b2dd1d07a84fa9f3b10a19f6084d8d61742145e +a78878c08a2d9b3ec98c3f24a6f12f006ddd218631b08b2ca87efcfb709725d3 +e81eb803a8cb7d1cfe577b1171da0ad82ed088df7f5c8345c2d0dc2704ed14d1 +61c9985c654a31f24ed4c1d1c7a46ec48ffe1abe40bced8ad31ff2c52f45baff +23042ae487a09a15e2ec0ae325959f5a302a30ceb1f998dfd82c4b7426ba0c6b +4268e9770e9de360f66a68388e3ad516b99fb6ecb0398fcddb404f2314c93852 +2f7ac1a687be06cb1f44770fc24089fb71cccfd89d600324cccdee8185bb35ec +905323ac61e70424d7817d6eec5ed5bc69eb0c4864fcea199bfe43e92dfde7cd +e32529bef02a6380aff567bbbcb923f8bb39a351b541987695b03b04a3f20149 +1fced5a3187b2195eb80c815082a7543aaff483f45b142261b9f3a2a73122911 +6ffd58c67b58ca264e778fda96dac9eedf2c0ae7dc23f185d036f8ccb8e3624d +29af127ae40c801dbd710e50ad16ea39133ca90ddfa5a858bdc8c816f8c838b3 +f6c8576710217355c82c12e3a474a0ab7424845fd97461db0a865a763e7bcdd5 +356e79b2aa8bc3449492dca3e3767d52abe4493c93fa378740de2ec43ea077e2 +5e6d959647e53ca568496ab87e3fea7b9a364f3544379f5b360dd6d9b575d9f3 +f0153f258f594281958e6b0425f94ac679f55d1fd30ec729f9544dfe176bec84 +d3c0bfd35cabdc4492bbca066b8e398e9da274d148848391346a8d87d2f1e2b3 +cbe50735372af177bf2a1c866cee5426271e7bc77eda37e11a6521b05e5f6865 +8a9748e009f07add863ddc530e2b9b6714ce11f5a889c4a3ccf517125b34a16d +4398b9f050eee9aab47ae58c18b0c2e0b9e3a9a64466a2953131f797b688a328 +9ddd6f17198fbab21a0ba18c7cf2b5f37511a2193b018c9746618dadc42cbf90 +32935c77dbb2bb8fa90923b3500e6cce6a8ea765a51e4deef1da48f03c90d14f +ded2e8a42e0043841b46b598628ee65a04121860fa485c0f64b915d5fa128073 +c3d23b0ee4ea1b74d55784e9c3dabe3e9d50c5bc9d132bee4b01638c11e592b1 +e9eea491bc7cf722cf834d2adf26c51b2228940280a52aba8c5a604b8cf03ad1 +878e637261705bc7591c6aab1f8146879e79fa7b3e64ded2f0d9a804bb09cf7c +1074b8e6979809cc6cc15a4d5436400b0e81ea3830b0bf74c7cf7c5ac7663580 +0cae13058bca582af29dcd1812ae3d396e08155c89553316f25f59f8705636ac +4efc6f1069d00fc91bb47e7f88aa0479ecdf5817c9cbe4571f98b3081bb83472 +9faaa6c7b786c447618edfb1c76a8e36600cc3621dfc1ba1fb63a9c8330db494 +27290145da0856d54f5852cdc3b37ef6b6a9f5de2b199dae290fdd0dd3356e3c +29a21710d9356373534d3b5ac515c0f52a2d9ae58068781fecbeee46bd5e3130 +10267868aef4a7cf60ac5da26c132d7faa38346273b4f87279f02df87c52116c +e41776595c5c14ab808f4815ea159be866830fa2d3cbd757104dfd642bee62a0 +f96c824868d59d38926f486bc58ea697f57e73a694294e304397000bca02e08e +3979a345203a44098bd71d93255d3d4de64b89c48e27c80bcd5568746ff5c93e +01982254408efcdeb6f632f3bd79589e62e450950facdb43648172fc8938dafb +cae34367610dc3eb123c6140d31ad2ec7799189906f7dd7b7e4905dc5afcf2eb +33230041003c45c856ae716ffd13401bb29d282db14e1f235166ac245ddad050 +23ce8649e9e38fd125b51774ea90513317920715a5e6e7c3272da7c500bf0add +43720b29d4ac9786351c6c6dbf88baaf671e28fa7da379d36b257dd5f9ac5069 +51ea531c1bb769bdd220aeada686bb321a8fbe646fb1f22662c7352c3220e3da +eef1f2d9d4f7cbaec3e50738aa2f25f75bcbe7ae80885129cf26dd61fcd2d684 +d7cb93fd68148815dba1ff84bc5e9dd32d4f61fce2798842e4755ac338458c94 +22e45dbd549f8f8d94068210c562e9165f53a8842ed2a4957fcbab7b45be9968 +a20e63183a8375d5b785fc02e41c870d278f778be1c8adcce7e595f8fedf7e9a +f2f986f2cea963a6073ef33a4f72f8fdd751b3e5e7c8281e951966adf89c82ed +19266c77718d86cd45872a374537a63c047d323f8cc28e7b0c492e4e7499af7e +2d34e1c97c3de2aa059c6a2d22ca918e20e263d35bfa39b96e9cb098f556ecd1 +cdf24342c9dd0f2672adc780816350025db36560f4e5aa9226a993627ea46261 +bd2f7cafba11581f495da52700e5eef89b7136337e466ab06ee9572320c03a74 +e6e4f77c612c736e2ddb7312b14bd86fcf867c9d8e822d27160ff152f5555005 +384e3d84c43f538e3f28b747f3386d7e4e3591ac5e94746f0857d6566d885bf1 +b26fb04a5fa4f8b18a8e283a15a471230e1e5fffd30fd8dc4f57a0a1722cb270 +b4b6a0b149383bd9077463c2b4bc98360f81dd8f964b330f32637c86d0076140 +61b708d7322033b5847791aaad03324de04d662f34bcc1f969e2ddddbe9bbe9b +09ffc2f53d494ac1c4547e3d5339755df7e939702184656f89abe509e8349dc0 +b662580d2161840218833e521143aa2bf181c842fa37f98be1bc5b9137068e3b +630eec291e2f1c8503652dbf174d8dfc44e714c1bf8651ff39f74dded66568f1 +f5c76bdac030ae05b9946295c24ad60e22e0f608f0b9114cc7e60a282725ab9c +5c864253d4db2ed14814ac43d6b7849f32d392129618bdeb25258251c34cf608 +0e80eecc36770e23196cab5bf081313c03d86581de394871627a6ae67c2870a8 +2454b8f91192d97c939e99e0b1cd37e90a64f56fabaccbd198253eedba3e1091 +75726c9e06ff5229870b213f5c422d3696a468829755369e5069470b9277b1e9 +efc8be24a1dcd56c7056f6c0467745a061a562f504afd54cbf5834c988df37e0 +dfa7c5036f4e1ae08af93144585ec6b2923e8fcd01b070e516c57bb771b92798 +eb4b8630b93508da9c197c0fffde911f871b8a2c4abcca773c96878686de8300 +8661b0fc561f587920a4c2992644901276f61c2ac42c161ff26bec3eb80b957c +aa45fa3b7621aaaa3e40ddc3c88eab869155a457055c7f9094c3d9ed60472483 +c3bfa8e25e9f13b4a18c51bbc503c68bfe6aabde36526e4f53e65cc48f73ca4c +b365eb897a61f17d275b315cebe5cf4e21f45f14c608f74592a9f2ad98b5a6e6 +77f58446b8391f5384a9ce81ca896012cdc3a8ea32cb6c86fbcb0e475a451543 +95c8dd4f9d2ccfc2396a9452fb60758da5f4cfe120b3047407073b96fc770eb6 +fb879acfc730286f50655151ac90157edd412e5f3b906bd58751923cdee29a10 +d7d0f5173c2d2cd92f96978b6e462baf37b55e24bc46cf485497e3b5b8c59811 +49410cccc13d6a0dbab3fde04836773d185bb4d46a332d32965eedd2ab8eebaa +e4df7987e36658d0584a8abcbea9602f5365952057dbd6d7f6684eb391e37a2f +6078eef3848c81892832f2951993372d3ad12c9e35dc53de6b373878ec3fcaf5 +229079a281a5e0e51b805d92bad8b4b7a4473d7c39c784835d051ef45b61ec06 +ee4ad1afce4141b78ac310def0f73e2c0320332e5475fccb9ddf5adb1170204a +fe96a9c818802897b59fe9068c14f2e0b8a090adac46f02a49b2583b0940415b +24a5c4e494dab9094973184060483c1a2b85d1ce1b9a82a626eac5c34b121d8f +83b976a2e4bb1c5950d6c9eddc8383e4b6a07fc920590c6a3d2421f6b21ba2df +898317d173769f2943bac260acabc0b66bcb802bd92d2efe8c5728f81a646e3d +c0fee26f156fc702d9942c4e5fe236b20d0e51d51458c75c60c4b19e5e724602 +2450b64e81119ab6f2367db9b69a532d9d89fcedcd7b3e46300a48eb09454300 +7c8017c2accb94036cbd125f954ca4a7f79fac0a2eff6bcb30ddcf04551a03df +ef9640ac8e20476b1c639c96712db7b10a3dd21a2bc2c5a6e4fe6b030b730698 +e8d99b12954b784440332038f60eea00d25a4295e04fb3c42c4648406db356a5 +1770d01208c35e476911614aed6a6edb5e1d33a8195c804dd64a742cb5ed275e +95f6023b7272b594abc5470d97aafeb2acac563ff67a2bf172d6ce3a8aaa09da +1c80a0993cc265142aafff11cef0ea2f556fb8b4ba5f17e2146250f7456d91dc +3ea6c4f518e19828a0ae70f6e0f4b70d420f16826f32e50804173d5e1091409e +f68836385548e317c07cd7a6119d3035bc62623bb94556bd63fc603db50ba5f5 +d8b802b8d6bebe65b61029903df259dfa474d9fab2b8cf5d22ace39fa4f41380 +8d9af69afd20f913bb8ff0757e7d7b7de2e46919694d64fec0b8520fa6978f24 +62491dd0196237a61927cb70a00e4a920dacbf38ddc00065dade6ea0fd275384 +6365364f4c4e357c5ba1b2a4d9c23868991580807cfdc76e032fe476e4213874 +0206e2b63011ec2d38f063d7e0e0e158e2d40539e7d791bf2f00edbd3442157e +5dc659b609edcab9535f1c186163417e536aef43dd6136cc648b51468c523cce +3355ef857f72f53f5082f9d0b31a5e0e809818d4669b6c60d4af3366f1b1b3be +e4f76b7583b35ff2eddd640dcf9a84320248cad0775b538f06d813170a21d128 +bff686f7200e3bf74ccf2475c22ea87ef78cf0cd7d99a88ebe59bed7ef749cf4 +097e9143da8b6e2a53da13a68a496750cb3bc103275e81bb758af55c3dee6737 +354e5019378c4e7d96cd82ad379398911d92650d69e13ca7982dd821a1c5655e +73f4557fe7b4a9304d2ff22c7cfcb18cadb318c4e045add1cda991b7740ff1db +bc827de503e7ef42c38cfce0dc5a2ee6e0436f66f2ec94f0712da1917d7e6c81 +8653e98db1837b29bc6f84fa0fe01f1e9428b5c140f359b1d14a5726aedd9690 +c4fd1b966a3c46a7cc5b80675449f27d4de132a7a3bda5c822703477778e409e +2626f494d095b5653ccbd6b7f521a3d564441082de5dd1bd8d92ea35af0c9fa8 +fa4ed7dbe4fdf792f8dcca408c0297a573af541806fedb9dd746e35de0e31fde +b88e1c041fc748ca4fafcf5f31e00c20467f1130462d9e01604d20d00aa6fe8a +c252664af40915dc4a2d7c2bd13f05c28a58b2d93e13ef373f673765330c8060 +7d42eefa6d5fc90c3009d953d78f80714730bf3f894f37277aa5b0788e73019a +a3f469286adc9e2496736c4f2e9dcd444c691e4d365b8b40eecbf848c0507c04 +70cce59c04b588cbb607edd684e154a4e6b9511d869821a8128fd7709793d2cc +ca45c36ac8c33c37e89f3e8aa9f68b957f14a7a68a1a79831fa3972ff00d73eb +581df4ce752444482c85e1ced92256b86e81fc8ccce8350ee1115f091033832a +5413fc25a705ac68cc05dd7add5b553d923b2ec3210637c804cb13e2e6e3c464 +2d28b45d3a0450088195ef3f52abc9c00e728564ba07c43e18f07461ab487082 +26ff3f3d32f48fb9bc841fad5460bdb5ce1e909dd95c8b1fdad4a63a37fc802e +2a71c8abbe494f5a7b77e210023aba67ee3ad72873d80d9106a6a904cb08110d +72eb92c9bbb160cd2b5bdb9254476d3ee17847fed304be1d5ea53546c7485cbe +fe67f3c3c7bd2b518863e241d77de378b0c6577ef9492284a5948db8370d7053 +2fe0f577c3bdd5ae66ab06b97092334b740ab016de1fb629e2c7a94d4d967e32 +768c2cb98f4b4bcecaedf48cf9f53e88e8194cb8e05fb0743be5b95ae88aaa13 +3ce143ff5efb9178304f7bbb821bcc3754e850c72c636802b0ef94cc738959c6 +cd3b5c55b63d01e4068f26347162a787584f8d0fecfa90eda0949d6c86fbb19a +e8a88097a902212e22fe263e797ccdf899247600dfa1cf69ace87ca22bd494dc +3e2165c8477f786601591cdb1c222505ebf3c1438e6090888c170607129ef816 +71cfec4e1b3e4b85b9edd0394aaf76a2411653b83d6c8acdaac93520e2ca1243 +c68f558f43992d5d342c75ab608b1d7aac0c54c906dd8f828082b78dbbb87512 +77f0adf765cd00039dfef63f8133c9cbedc92c2fca3e300758bb994cc0aa4e51 +3a36ede5152f03d80f70f27bb36d61989c33194c3c391f49483cdf07030391ee +d33374cb9f0cff75b590e0ea05525668e79195e414becdb3bfc1b32912950c02 +5b87326e1ea8b276f28801c5f4d249ca530fcc257c6944305f68e4e052972643 +7086214d6aee284e4e81e3274851dc4199911dc9c2c345f3f283605a0e3d6d1a +49b0f32284b4ea19cf6cad9f0d3336fa09faaf2ccb056465c83a22e1d054caa3 +a06d6161b35fa45a83ba870e8a28e50f073c1daab08f8df4be360fd1f75062ce +93fea2b08da012d3d82e96662143485a6bfd3f882cc188f9b8a0e16ae06281f0 +7ca940f27837c872dd94a4223ca3c1e0e0b30e9c3db060b3ea0595120b43050e +ea0ebe99a38de98b49d75c9c438797157a9d0da29f88e6a2c4a7e014a8d74c97 +a6c0c0743ca6b9304d2dc9772a77638d8ca66d9a594fea27fd1bdcc430ab51aa +b69f49ad2a4904de92d1ed2b1eaadf09b036f7215d9a4886ecbcb3d2a994c8fb +f0acd6ef6f67edd92fd0e560919e44661216d8c961fd201c50ea512ca102635e +37db30b8a78dd4309a27d25a319635721f55a7ec178d8df7d3d6f26a5f8f534f +95afe2c38a58919edd94178ede7b3be71146b831563c69f9dc0809bfc9bfdc72 +0ae7d4b1e807c7333a8a16ececf9de80380961a4c9140f7d00ad5e5987b97b2f +10d44e89c9be89253117ce4af117c8b2b2cf3bb228f964e2b23e84c51044da5a +4c1969f11518cd2bfa7e245222ed4d32cdd571e57eac2cf4cd8111ac1ce325f6 +80b7f682d606dd3f60d93c608f9e7f0c0f568e2cab763ed32527b6e57bf4d067 +cbff0c62306d38365f89069a578b553738597922a25d1929bff075db37d5247e +e27d1189d0d90cc8cfb3720f1465695df6d7a5ce9b79502ce0778c1fb98c7e02 +43961438ddd1335268fe9cff04aa1615d06ab9239f931006d4ff8d4a9f17bc12 +b86320ac11f1170996deb6ea4613a1bb9854e6ee49d58c1e1d147124c6e77545 +4eda172383e35a0de992d0e91084fbd92b037b24064d88eb97a2347e9a5c50aa +5c504b123c666712553a5a8095b30f0d0bb463606e1b773652f70d1cc18b3d87 +e7cf4cf38ff099de63b7ddd97ed77fefd48703a11b18dee3ea145bc774e8b9c5 +cfd473c7b1f15e558e1f4d211811e718174dd57a6d9a045342579bb707a52b94 +93b14882192a4c7a1a3a98118255807c1b7831433401651df0a911c64115fdbe +7a71af3b5fc9aab773eb25cde6e21b344608ec7f1db3664e1aec04e97ca440e7 +4d1fd6e8f8537006c8e379a233c565d5960b3150888c3a2617a1d3abc2a7ea65 +a20cbe51cc30a0cf6a82ce091b031b2107147194e8279dc548882c4b44364027 +a59a2b64cfffdced679d468fe7e1e38013af804c6b17db6a70f0ec611801d30e +74fc49394859d048de8882baa5fe42f2189dfcea97e3f13141e90fbc05757f36 +387942cfd03c71cab6eec51712cab7bd8a452b9e9ee7fcef78cc433f37077a8b +2698348edb6de2e5a14eda6ac5c051b8fd064d3752b821adef9aab961898d98b +86e09de1f4dd2ff1d164cef1ec22720421bb4d8d1a6554ecf163cd5eef84cab1 +a5808462a492b788aef259096ff34b63b69def56023c5936672f1dbdc73ba3f3 +4bc6f49f40314e2f88ede985192da44b350be2b64d2e43c24e2e3cda706624a6 +21d92895ac2007bd647bdbf3f08ed66e35568ff25c8380130536fbd5cd2cb79a +374e30a2442f67cbfd4959a96c4d450e2c5f9bd2365301b5a6178ff7ac3827b9 +92b0dfd5ac0b6e2e8dae164570747de735b1c26b71b5afb813e26ba22622fda8 +c02ff35e2440c9f92aa8d983d4de5dea804498a2116364d0b402262559466cf5 +17427404bfa931cc830db15bfdf6496afaaef006683394f811ea29a78857c5f5 +a4a2b60ec629521a023e7a8c1648373380dad586c854c2dd6c8a66b2f8d3838c +31590d929b8f7bd494cf3068dea0b7d0a8c8df13d39094ec965f6cb14cb0fb31 +192e00b2397d2e856339a6b37c522eaad4119e533ba9b1b50539b5cb763deb21 +5432adbd56ecec66d599b346ca20778be6e89ace49e3a94f40daa87779bd2230 +4084bcc49ec9335e8b062ce102dd164c74a5820f0611d285a64fa0ee3067cd0f +4831babf5696b40d935cc552991af295b6de7a51e4d85419695e28dec0739bd5 +0b036dfcd54233607db0240ed765315133c90befe834949c470a7e132897c7e8 +6a4dc9e10cf26b35569340a2cdca7fc1401b020dfa47e32fa2ec6cb384302b11 +928c3fdd312b2b4a51da8d78d4c6e47b5b166a2bd97d6b148b12c182a5b8aee1 +b8157616e0b5b009a95ec2a85a740ca336305de3fbef75382c09fc9030cd9795 +c518a9a62e12ba2a04aef6f077f7c5a3211c7edf6d5a76d786428a4787ab5df5 +011abd80c6e20e22a97a34a36e913fc4f2e80978d07ee2ce587aff0ea9203349 +32ff541b5cad18e87fa6b62d792e6f55dd6193f1c95e9cb22c7a49bbe7eca5b5 +5ed717a11d4656c86249585e246fb9a6e3d762b7d8dd56a4a729a6265323a3c3 +0729bc59480d6e9bd3b3fc4fce30234932f82c5859b6cd871a866d4ae2e127df +2c14ea7eff9f3824b0d6903b70163da82019b3b7d73b21c869cb7b2e1e66de59 +815d4153654775ff14bfb50fdc22845f5f50b037d2e536f8d79fed1718892981 +0dfd89c14925f44746f457b8486ce1c01ad9112f6fbe959e17df36ff563f863b +d7083af4f38bc44bfd06f3d354c483a1a0056218c8cf5492e7bd511f3604c9c3 +5a1f2617b4c599f5c6754331b30448e8c40e6a4678a7db59c265a0e6e6b33604 +0d32c3235e1fadb089955e003501b5df491ef54f1a4ca7320ba9f119b48b370e +25ee6f16e57aaf5d7c617846f2f9586675a9a7fd468c8d3fa74e696f60a6d066 +b5c00d9c573736609f09ff5ad1b68ccdb90f53eae4a43e26f003cceebba47b3b +40177bb93d21ebde38ec26fb248fcc8bef7d7bdaf763c9ee610a04594fe90173 +4cbc0cb8dd6754b15e545a0a26a770b31464bc9f070db643adf743f7de00164b +4d3af0600554a229bd5e12822640d2a266ec6e73148289d671c5550c04f1c56e +895c8f042dca11dcf99b12efbd80b5d0ffef494702ff8d5555e1feaacac91626 +ec36f3792241887786a9bfbe8980870436d88ad76a64bf40b67a08ec36076153 +ebdf7587f0ef85c72731b5a38bd4dbccdc2eef1166c707caefb9e6859490753f +b935bfbd37db1723740fe6f076e5bbf53ebbc1052888396723dfe8e2543dea40 +3246885e3589c52413fcb2844e78ecfaad3ec8a491a3e87934cb9122c0e22877 +1112bbde51a5828ea530c58411a6954b6a2b8bb1c5cf2dfba39c8e93626f69ff +9f85769c112dad02f50eac20b2c55c2a0d654e6e56d38ca12df576f2992b1329 +83ed96125a09c6a44132a796bde8b7cfd1b7a2570115afba3b8ee9bb1b5311c7 +2e08967288c1f09f6fa3ba88bbcea7251b3a2e4ed5819b54685d04014cf76b8e +1b29e19a7bc81304c6ff0f8bee5ed79a6158e80cbd2ccd68e53d2df5db4849f0 +d30090121e6c301a90f225114654138d79218ba1f98cdd93cb447aad1429cb84 +8cad0ce4183bec55417bf7ffd6e88b288fb47dfce7a47594244481b6fa38474c +697ba73c8b5a8ed9f2a723ad53eb070500d85f41014cb97106634f980ee0b12d +bf051b3ea14377e615cfdc1a293df10f9a9b3d131c4b434bdeba2275976a806d +81b211124200c2b3ec489e6289ba20b3cc928cddf80451b8dfd902a03cbe5058 +1e26608b6cc7749fae2791b52252232945aff710136f606d4c1851a7006aea1b +ac0b2b65014f3cb7184733f4a4c89d4e8753cd44ed511939e2834049a511e795 +7dd0a92492df72707965c4ac76c4fb8f64f3b030bb9cedca3c72f242b67eb3bf +0f6da3c7964ab0d9ba023662bf8acd04012f7e543d3adc1caed9c8693c45f324 +aac7a68805f394eed203b8448cab5ac384782bae222820ff8e83c3212e429121 +a323e9cce1a9c4a3951dfcf9e525bdb78a77a34d73e2353beb79009c8f8a73d3 +e81f8eedd954de2f654c3f52711eb14ef1cd595c3d242eea6c3bb4b8e5518137 +0724eba43a43adce51563a908b2b99b7712283696fe4685fbeac1f9ab75dcb06 +1d523aefb412d6c1923c4520cf514dd4187bccf78e2b10bcc9432456be782c51 +34f0d4ac9f4630b64c9c9a04cd62b1b2a3d6c14e293c096c905336aba5107e2c +a9d0df2cdcf1e2cba59e46767732ef987a4698a500b9034a87e31682dd993e73 +b37d59f476fec2ccaa6faf64478848f232e23e1beaba6151e972d882feb8b996 +21aed7f6bdad94bf02279e528f53c49a401a1528f79291cb7f8da7d544d1e97d +0649494921eb68598debff5e4e012bab49a6df6849e8ed41a1a458a77c1df8f5 +6f39f32d889e918be8be7a598e7ead09562e29ffabe476fc4a745d922977de11 +8bac05ffebb73c22a2f65a8f7e7aaec07ddc16ae3cd0a7b74e4a34e04ea25832 +3a584822f78a74c1a5267e20299adfa8de86caf85e5050e469868c4b11bf1301 +b74e33932733bdf0738930c2b37fcb582baf5ea99b999ee4edac8127e980eea3 +1f475e98718bad1ef571f381c9104adb1c7b0bf35934dc66a8cefc7a6228ea6e +2bc241bae962f35ad45d7edbd69b18b7da0608223ac8223af03e83c51816c494 +c46bf1f9866e899e1728e8cd6a19adb5ddc76de33696ccf288cc2a68988c19b2 +bc8abf0c2ad8027211595630f6e65eeea97145ebae192fdd5a47c9d5693cc7c0 +2b4da2c9c6518efb5d8a5f1f51590f53454d103c0f7e85219007563cc6e247f8 +ee1bf839d511d2b0de3b07a837d1175316a7f94ef0528d3f8dc94cca211620db +d5d43779b9d3a18d24cbb59b250bfc954a5ba8f45a251914c03c899bd2695158 +106043e339be2d1875f53e1510712c7773c6d85bacb806034bf4ee0d6d7b6b76 +7f66c3bee096d46b8ea320ddc416143aed6667dcb6c13dbf491444a052834f40 +4891006444918b895fd13ec97d4b7d671141e0675b9d64fc751b5b1be7d4c3cd +aa102c4df0cf41da6fdda24cdf6186a236f8821c2fcff496dc7ddc63b81b5897 +d2dda4c08eb46128861dbd7888afdc57f54e91525017dfca043206965686fb81 +4ba80148b21f6314b3b322648ace69f967fecb3dbf66845c9d7cf7097c73a7bc +3266ba11cef272e1b67481c00099a6e0983743ee0f8497440585ddd167406899 +f19b49bb97dd273f3ee09ccdc1acc4af8b512356dc9a2a6290a84b4b6296471f +271fbe461d06449bafa50f4a23be3cc1563b0fbe4894ec211ce7c637a68d0b98 +ce3d9d5cf8f47b944e30e0f039b42da1f42ad72bd678542b28f446101e27579c +9d3324c6247091a7a0e9bd19fb1bb0e4084caa75cdf2ade6471068ef3dcbca3f +9cf11a5b54e4f05de724adb58ae5eb693a310fdfb47e1b906ace009b30391aa0 +738841d2e2e86dd471d7f8dc50757825643dc546bc1180f8a7c37daf96a01763 +1f835e16d0428d234a24cdb83f59c553a09328a4a07abaabd4e78eeea0f6bcaa +e69f50be8108372585f78c066336d52dc072b7cfac5be6193baaeefaf5916356 +8972ab46c5c794bfb7db4e00f68584fba246e838b431a4a84f10e7c136ab3d16 +c212dc36443128a8a0953331a8eb655d5d8fc9213781550db63ab7443cbaf1e2 +c6ee231a1bafe97969c02b0025d777ac496c0afef31b6d3e5c42bff099b32e86 +2734ec0d4290f62de2948d8c081befb55ec5d4cfa2b5e34a2016ffc521e64d35 +eb3e29a4eb5af8387f80fb1bfd7799af6418c1243d8f637035cc6278a35d2747 +0bdbed6184cd0ab994bdb191929f075d2bfff1c2bb2541a7f6aa2939c8f4adf6 +271171197888fe90a37ef2fc0f579faf60e1d1075710780be8dbebdc2977c4ee +f1c71232d5b2b67f1f02c74d7027c3b63a84b02cae8bc738fdf627da1e4aa277 +627a256c14b4d4841e67e79a2cc48d9f16934e77c304da17c52c230e7421153f +40042faf7f045b8a20165281feea6e116626cd9bf6150f8775b7c801050668e8 +ffd77a505db6e6ebdccda183dd8590d5ae8bad19445c5377d2b3765a518938de +f2cced87d8f89e172229c1bab7650b87124a8f7493eb9d76c7ea49d056a07b2d +01321499b5bac27b175405193197f9c463182e8084176704dff84680390242d3 +ab5775873b697ca3178b24d672faec5e84b1283a6da7e61391837a75605dd27d +fe7a7c15b26203eebfb146daefbd85c60f7afc83689a9f1bbc3002cc574aee48 +5376f51ca5512fbae641a8fca0590d4f8fc4d0968c0956e3fab8243a849d2c98 +5a8453e7be387d77dee25d95092e1bcbeb2ca752446b72089e51e2a263779439 +6802224d6bc1c75b4229406e599a07ae4eb6d51edb699b184982b5eb4d381bb9 +6f9554affddc3ae995ac0821a500494c00ff3dd07a076483cbede6b39b71f5da +7961456a8b5b93771bcfda66d1f7e694288ad65e3592ede99f3c5a0ac3c31795 +872f81136150bcee7853b80b5af56742e4f1b3b5f149510ca9e96c0f5e707b58 +2cf42abe1ece9b07360cd8839f9e0a8146608b3df9f63e0791af5d49d855a6e4 +65d04046c2e30fb9748d3fe663210ae2291c63d7424ae311fefe2dd94f182c2a +3a5a537ef02cf705f04b1487dd5ad6f7686d5c27a7ad8c90678cb413c27c5808 +ace7ee2696f4233eb9b7be1fc1161367ba18e7732a6126c761f826a021020bcb +0cc9e2bcbf970ab3811ffba16199ea2d65c4699964f1017419f21fd277652cc8 +1502d382f1eab0e392f46e4621f933dced55db023528f605e5681a6159bb9bce +21f1250d9b600da8bcee76202636ea7c639c50c01d974b77531e55426c0fce8a +f9842cdc443cf7b45089bb89ec94ff118144361693a55f79e1becf66a76dbe17 +5a439b172c629a06a9845e08ae4b189c70e1a247558696794dd84fbd2570108b +639ffa11da36ffcc3e9d896ef3d663eff48ba4c6d82d8fae2c67c6a4f9b001de +ba0ab4da907eb987c1f05a7ac49e2874849e4659aec28658a3fa8cae79ef225d +ee0bbce2db37397dfb71b032eb74ba3d3d11e08c87250bdf422014d7643ed4dc +dba982ad16ab90b15580fd26cc2de70440a237b7fe2596bcf6a8715e16c04e67 +1e112179d6783031f1811071808bdf9406a8ad2cc99745465a79293d67ee994b +f1b800d77abd1d48ff1a091494dab0cefdbcf8f839b81b15787db45e78af5d4a +eeb00996a4acc6d9e23d49272da68a6a0089e98e0503e4b383616ec40df8c152 +667af161e4ae2cab5c4b20fdfdf4f960443ef61c53564b6e44a310e557f5b52e +a28a2316ae84fcd8e2292bd53c76b1906d242acc791ab64be4c97d8335720679 +f2acf0f4fa4c8b6c22b94033a45f3ce8160ec9aa1eaecfd86cf9a414525659ba +dd878673e6873015866cb99eb86c3dd0672a8cc946e5b3d1ea0ddb6e523e7720 +82004e7a7b50d39c92b1cee4344b662c075e22816fb20399580212a66d122d9c +42f07c778a4aa7d5a4b1cd136ea68fef5fa2897dae0a8a9336de303ea869dbc7 +f517f52f1703e44e4aabb501023bcb662ed64c7bf6ef144ae388e6090750d713 +a52a484dcdbf7ae2c80118dfc72f36116d51b2e2cffd2f4fb327caaf80acf4e7 +59023709e537b5ea015c196a98f7381fe2687fba887267552390794dcb18dcd7 +7d11ba00dd419b6c810613c35150aff7c8842023f76f5aea496ee897b24df273 +9b09ee41fbc37d041a35a7155f2bb063bf221be1770e55c036fb1d0db7990cd1 +f91cce555b5fb90e14c7b3597073c23f972721bc93047c5bb0dabe57cb9aeff0 +76b1733bb1f928f53c7dddd834a32c27d761d739086b5e00cdda57fbebf43b07 +367737e345bb728e43b4a25070af42247ed782f987574bdeb6abea42f1517fcc +3dd26b1f5145df21fbdc23b4a5a1f7b897cd9871d209ee317801dd5713943096 +4e1b57752fed92d5ea8c35fa95aa30a2e946a7e605474ef16c1462da2dc88d74 +c26a9e9f55ebac4dba8323dbbcc398319c4e02ded2bcdbad5f7ab227a218f172 +c9015c6ad7f7e59f8fcfb19f87347c9c9285a0356b668f969f3cfd31fd7dfd20 +4d7efd5e88080862a687f3ebfc6d59ed481261f27bffe0d14cb3e50c94bfbe92 +6a42f443f90a7ef3620b1f5c07529b40229b4a4e0bd8dee86e661182ab056e0d +480689a2cc9c577fc8c23fec07445d68df3e82b5d41ae0e2445270a232114d5a +f781ec4f3210fe0abe5e3036c6b6fbb68c5af853f7805abcf60e614ab8470a07 +6473b797f5b28987987577a4b46acb49f5d154c6ee402ed9114f7db3bf939377 +e141d8355689f92037da304b20386368bd04f707bbabc072ff9c61c19b186dff +fddb94f850d234d4870150812d16bfc2ae715ff67498c1b33acaa894b6e5a924 +80e54effc477bcf6e47aef64558fa92dd475199f82f49a984e7bb448533a0697 +5807d58c5b2ef18fea133b1a226dacd8d9d994998c531f441262aa6e3020d85d +482fd9128a281f14c6213b8065189a7f9a3b68da0716936b23b45976a4364c0f +a198135074353cd2fe81f55f62f2da556d6943f9faea0b616640bec36fc559fa +53dce74fdd2e3aff6c5738ed520c01530e3c239693eccd248675002765aaa721 +f05102037b2e7be3de67e04e6e87909b7397761261ea819ecb0a9968bee5c92c +364ae47222dc0f386a8ac92450c00e661ba2d9f5f3716c7f2537d48df6cf016a +02935d56b9ecf8c7198c593b515411a1def3707e3c864443b9c3d56f42f087ca +2c4056c6d56ca5a1327030470af9cf11049b45fce1ceebfe395a042c48a70ef0 +82c6cebacafe85c4099958f2f9a7bf6389d954c68037a3f84f9da23f22b0404f +7c13b3ac0d8d033eda4f82a9af0cf10da854d197e3d86e653b22c78367ee4c8e +6e36349972a9832f6b21ef94c6c3324bb39ccaba91c5c18a9e2d86fa1a72052b +d0c4cb2f0085b7ee58c12cd775826e1dc03f0ef36ddc61fbc775444e299188c5 +5b5bbfd1ab139b2db868c8ab6fab3eff224cf4b60fb9b95100656f1eefae665d +db93fe55623be236ac7eb26868162a2cbd582891834553fb6ad22afd50a2b8ad +9b61b3a9aa37e79a344ba5a8dba9f48d3093b46282a73c3e1f550703c3eaadf1 +f5c31d6cd985bfa0751b50cb32a9ade2f42e62188cbf57a9fca22096c2c475ad +7dca003d51c42ded7feb497a7bd163adacd9f8274302dfaa4f889c9507c667fc +5ea528b6479a8d92140661d2907dc101fd1a8bbca74d49d9df6c65cc68cfb9b0 +549428a7c4f3f473e3d6bea478e499056a17b25d8006b37e08d77d9036a54caf +d5c9816ec7b5e4d73f00b56ed54f34a0961c13e2d2f44de472b195b41e7eb8a3 +b6123d40feedc844647759588ef09b766064ac4d1dff5e64742791c4d4421d79 +cb57c94eba2d5c02591a7dd4fb1785e725c82b37010bfac359f2f1e033889670 +ab5d04c0efa5085cf881fda097f7c84ea08eec0306f0831e2efa12a0480fbb4c +cfa41109c8fc3db13c866c6a471c8aeebc02754d9630be274a2b5ebcf0e5bbf6 +7f5584729c3cc91cd810bfef032900c2126d98897b03cf89d115916bb56751ba +89a311aa7fd60268c5c9b4144d88a4e8d210db42283ab2cb09271ee34bd7796e +f972613d7f1e92e7a0134efff684e3d91fe8dd36cc196b60cd449db52142c216 +d2c2a7e374b6a7003a934f28b8300eb5b5c7b72fb342b115cfe415ebe798d55d +a24deb27115094d610054aa97a3913742363fca01df49f71bdb5239ce8b914ab +499d95a7df62fe6ac8e48ac40b76bb4c40866260d223ba9cc3a7bb2749560db5 +d7bb952d73cce047178953c242bf0445362188901f2c855ea47cb9736589c848 +5c57e90e80b5793f52c21bd8bff6524031d424c389fe1cbcdfd2b609b522e64e +977fbdbf4c84386f36a04025938fb937d5ad51a49f0e18b37f8db3aa44e2748c +d03a4c0196ed88ac144d2284b21c5478cbaf5c1a96319f18bf3ae76e0f5893b0 +39e625579fbeb26c6a6feb83382f784fee7c69292f9d97136e82d70698544bd1 +184b4d3f39af257a1d6542689094fa7159e71f4668eb8150e880b0b71ae3907b +df64016f1686c2d2ca0db01cc59e9d2d70c449633199f0c9a9ded4603a22a2f8 +7c8cb4ffa06e6d6d26b7b557467fb646c3ab8c7c8c9aa6cb7a7fb69b72fea619 +5ef25a5251e875714c49b0dde0aeb1ea9cd0de8b7c8a66927b9c8c2c59898d00 +c15e4441def3f7f922461a12ee277afeda15979e344df421b89cec87536ed596 +7fbd9853cdbde7cb5e11eafc0597b1cab3a895a39a05fc9c5b5981a0122bc144 +344c2beb08d11433144fccc2772fc6f6199a312f13f59308a4b6711647aa7395 +d766a88a811e504f11a240466641b16f79e22913e36c693b45551203303b8689 +cfa6ecda43589d02e8b3637db9e2cd60ad5b00681e5f7ddf92743dea6576b02b +4b39c3652fda6453558dd999ad496c4ada603bd3bc77c9742fcda72a31ceacda +4863f946330aa9d3dac343600940b81656e6c59a879beb715a5db248f442c7b6 +9382be2c14a31a042a6489369a118270b903d00301664d646fa2b3e5aa3969b8 +8b27a452d1c0d29f04406f3ea1be8e3c3ad23138e84cf07d68e1e30d474fa284 +bdc97186f7034b71bb38d03390afa087472a5d68295d66d5d53ff2fa1b6a322f +67b5315dd933ddc61ed5995088bfd7e2cb856ef815ca5fb034b37ea45875ae68 +44ef3b2bffab078fe430407ae1810e6eb4f71e09a9d4973606ac79ea7ef096d4 +d704e7779c22682a7ec6772cd0dde2ee311381b633f7eecc5ee66f06f1b53852 +666f67427e0255853044f0afcca07310c91853d7df33832c6a786a8493f72165 +d5a17e719a5519f459d7969efeea4fcc85d372f22f353a64844601889739aea1 +5bc8bebe7854329a8c85d7aa6d8936010267ea8ceee4217bb047ea6bca7d5fb6 +c3eca74e820cf051171a008e87fd3dd35926d1dbf83d92cacbecb6a5c46eba70 +bfe0b8b5efcec8652e78040172b838b8b3a2904731591edba7e90d915ec3eb4e +bcfbe08160c723651915066e1838345254222b2f4fb7bd9461e9b7ecef66f4ff +92cb1a0676c7228afdd1203a467b785976f485c98d8d37474fb95894d6d08d62 +91c44bbc37eb54b1717e2b2e4362746ceed5a25910fe3f4014556d1683de4855 +58f4ab89c3f24bd539d3f3ba5931e0792f0316a0569c6e5bb7a4150b07aea87b +ba61babda725bb53aa37475a778988c3e84029081852f0f2be114a98f46385f6 +a72e8cbadec2b9dfc5ba8fa553620dd13e597c1094dbae865b122453cfd27c94 +e7d60a722a19dfde71d86727ab8208ad8f3cb0c9596e981950d04cc94d309242 +e24360a549a9859b0d8fbd98f25dff341962d74891f5e68130989c5903343d88 +17f1914e5c8e1243dd6ce054aee9cc603902291eabfc188108da0737beb66891 +f20a77c40a58949dd8e270d9e74b94c81357d5913fa7cb9948a654f7aa2b47b3 +684729bc71675324826e7cdeacd85c75b98c447dd2dcd2903a8d6c3c9d793ef8 +c84e878b9011308f15642618bb87e682206ff1af3c7e1cabf48c40963e29d6fc +71a6ae73cf7fbb1690bdb3b9488fa508e4c92ce705cd292daa2968801773a4fc +84d34071f00c189ded6b7b7bdeb6c4d48c5085733efc51cf3a49013ac083f3bd +134ca7beec62027b08450788784b2e15f0c322d7d62f3ed831a7ef9e743fb23b +2f9d9071cd9ecba463f5c636faf106afa0a09ff87ac24497290a36d718cb70b5 +e4cb79d5604c8e65d3e7b79969c269203e3b75ba3fc2d0d8216fd0acfa76b3c4 +a666b714eceecaedc0c089441afbc014b996b4288f2bea69146aa42fdd01e5f5 +3d90d0c89cc5dea2e84c35082618ba10266e222475d5ee2197824a7d59523857 +8d1924f73f5c9260da8858d78db4693325944c330e4cda82791e3c8d632d4855 +e9ad7580e9c06c3376aa72aa47c101dbea978d333fb1c85ac923d54e779edab5 +ace9740a9f9ded55e044c49947b6594247a72034e1773faa820fdfaac84d1444 +8156a3a8fbaa654b1e8b87fd5d059eb6a7aac9c758640527df295252bbe9a5fd +b3b4b5b931253a87de003d18b39ebd49982afda4021c4c7de7f4082bab04d188 +d48d63d902ffff32b0bccec3b87bfa52af12afef7ee57e8c28856e5a97c2d35b +37bdf7aedfb7a4dd239a252cc7044bd83d0935a3874d514112c16fb47ddeab00 +4fe9915b0802c2cbb9dd8bd87bb81712340c337cda8a02b9f0a86bceaeb941e9 +18e689caf0d26182e5d901db06016671c4b261dc5dc328f85aec98e510c892fc +67dc7132bf41f578623328dc925781cb7d8de1ba9ae66f4bdfa5d53bf4492e5b +13ca31126b43525b3ae84aca0ee5cc888a799478e1aadc757937245e2988ae54 +e4a1b1f85ba8d3bdb77397e30055fc02f18a4affa2f8baeb7c377ccc0cc2d244 +23f407caadbd3b37baa95729db1ac07b11cb800de1c4e6471be7e69d7d821ec7 +03cbf0dce5e14978781de36c97e7aa6f96e99f463bb8460bc2b2b04edaa576e9 +d444b0e123078cc470214a7459f7cec22828e55ac0ef4ffae430780ba095d2cd +066288ab8a07a8ed4e2e99fe81dc2935b72302a2f74d29290d2b7a4d90b7bed3 +14a2ac2eba70d3528144629054a800eca86b70b2788a73072016b5b4f7d74ac8 +0b76c0fd4b90994ed02ba9fabe22a434aa99baa8723b1472d260a611fe82c93b +c4237fac6bb6881247866621b8441a08f3b41212cb25cd8f9d24f4b635065b5d +b127ee87a3bf6d2b4a655873a33488ad0948feef7ad1e5f17673e1335d25fe31 +a23ee46829eb96259f556459694fc70f2f53b56a79ab32197b13d7b551235554 +1a3d8bf7369b05c9e2662b8f38e58d7d80a1a9083e52b292f9d4a876a8891815 +bfdc99612c4b5c897c478d4c3f9959feba23f908f2a132d32e53ab2130efcbab +59d14e2d50e975a9301406fdf922a3ab18902b311de1a3b36a3ad1d93228039f +d4956c0e29270ce7bfd33fa2f4cfce1ed0a29be6daaa9ef0b25815d84960b83f +e9b0f421e0d030e575a0833ab352d2307190ff2520c52982384486408579f37a +d55aca5b4557ee675e81889d1126530759a0d30820b1bf62a1eaa3fbc30ed56e +a887af491831652de111eca63da0423046089cc5381b347d70ed9bcc21fcee6d +958813406364057b4615566f1ed758a99a21e58d38e97331b680c06c209ee694 +d27fa307db58219a9308804abdbffca489b5247c4d4ed827d8c9a23461b5c96f +1c1b1d00192649e3fbeb0d3788791916572f58ccb3cbe5bf3e71601b02ee4745 +5542a6f2f8f87b5957fbb050fdcb1e68ce827cab703181d6ae56e79554151f8f +30f11d0627b5ba00615f00d7c5f8ae8ede781b4de4a2023b9557fa5c4044558e +909884b7d0b8e514ba9f1bc9925cbc0b4f817d5c5570fed07cbe3409bc8fe82c +ad06f3fc83e69ba0ed5249b1d28be9256854bf600c61ee853ce534b4526c9215 +9145f71698b5ddfe75d65a91abaaf52bc64ddb46f74b84e20bdaa84c95fc3999 +5ea25149284da4afd4e658cea371f0766bcef153f49354e4267fc58bdbdf26aa +b03287337f936b25f2a5b3040abcb0606fdda74a6eaa4063cea1c53a821fb965 +120ebd8c6e16cb08c9752f18ac011b7ca71f571d2fb1624712653a7f633a5cc6 +3874e84d2114a3631b27e2a89eaa2706de9d462cf5ca42b2aea29958649a481e +38d6e8570f385d2bf0cea6d3e3ea963a00b74cd1305111f7351f6f95cc830075 +b93a3dee4b4d3bd2650f1a10dae6434975c846d77f056eeba7be2e28ed51d2ce +ffd919f6375d14bc6cc07af141dc2ca2f4d75a421146c2a3c7746a831efdf1ea +e53be1c11e0f2d225ba0edc6308ba1628599d0649cb66522de745c6ca5ac7537 +053f9278ffafc8e50fd03cfda0982378d9eadf1980ae875541e23978f883b90d +a30e227f2ec8c79561d8f5e1311225dc8476459106dcb32b1769324b9de23afc +b7c2a4671798d7f87266b042043a59d4d3208eb769db7aa63f3a495f75be1d3e +f818642c0ce3294fa440a0d581a4527b8e2e0e70e3f47cd721c0c3e100743d0b +245c007e0de8e24d7cbc8c6ce3ddd0b5ac8cb97bbccb45b3409ce101e329b5e1 +a992d481f7a931e961dd4051399a7982ee93b80313ba52adda62e5cc4e26a324 +24e61a82c563e3517a442af6137fe0f7ae6b88b1d6b09f53af9520bd2159cf1d +dcd0afc8a1b077a4c8fadfac9513aec2f73535440c8d1f673bc66689d7cfd7e7 +3a2779be4f068cebf26f91719ceeae2e1c858928b5519f5eb806379c9364cb2a +c300137ec88c0bcc42558bb00cd377048688942d7f71e0c064a3d9b1751ff61b +abe8327090aa007a1233b80d5e43edaa675d8a753bf69bc3ebd721b6fcfa7462 +ba15e841189cdb3bce58b4a1c5a2248005fbaaf9b57ce3d0851d3edfe26286d5 +99f12bc99f420c60d28263fc8d546f75acd2d7c8cd485e42658dab9f06930fb8 +a61652c21910080e1ee57d6eac3de4176df4b3ecf3ae81153940b98f4a2332e7 +a6a3db0e7c85368f1aad28359855230b6dd6678c751f60e19592ebb487abe17d +f27c6b23682a6912ddb3c0bd0547b920774d0ccd12e87d654f575b233e8555e3 +b08a6d13a5eecc52f35eff8c6cb90749db2ead4c8fe3221a8ffe7e5c01e92206 +623f3167a4a13437b1edfa4f5e6c220913a4635ebf0be1cb5c52e149dc69faa0 +a2223061eb10d65d7a76621f0427ff6c08b238a25b86b5ad93431d2022906615 +20288fd650524b6ce3c45522c412664ceb382bc78b0d72c65e964d7e0a168606 +75eb18c4d8bf7f6076a490672c83bb3d588ddff257e76d44aa58c5d3507aea65 +dd6063ba6e79383f900b7300ecf53e4535173e76b0ed1e46eb80aaed8633156c +91c023eff478100e94768b16a8c6b0c95a3d8abaa51e47117196c26064ad5a71 +9ae6faa95145b7fe72dd5d7c55bcae439d9b0f14f8c9da6eb7fc29aafab8641e +2208d4d9cb8d2c4f0d922dfff75a5610e495fa69346ace3e2460c629272d67e7 +21a35f23ef8bb1f20bd309738445653b35794f195464d641b823416e0d220ee6 +cbcfeb1b4fd45a8145692b81c17f9f8344c292efc023d71f1212d4134c38bacd +dbb5c004fe3f8eb7f720e61cc82caa08a4821cb8049393b893bcc1208223f29c +318bff1092b9f05b5a0921ae93db816dfffa8993f78d057217b5611f61be9be0 +56f0aa36b51c2971c5e5337de1c9ccbed6570f6a57f81c7ce1d4c1c54bc60ae4 +f93a88ca79470a5bfb6d6ee27a33cd2579a9392b23c483ef9bb4a61a8db53401 +3edeabcd5fc164fac3067643bca935d5c81984194a3a48f923a83ed8f63b2c22 +45226dc82b41a1754b1ce1f18e264f447d6fe8d7757d12d601938906965f58d7 +41671c742a150a339237ee364a18347cc82273d67013b82648ff7c0c029ada9f +70f3e24551dc1157ecb91511d9f5ff1e7b6b3e46ce15595d17af9477976f8a24 +7cb537397a8f371c82b503a459cb2efd310d62f1a36d3578e840e0bb59790db8 +415929456b25fcfc8ae9e6f8f8f5a3d91a5003252cbb5a67d5aaa2d2668b52b1 +0c74aa73bd8c3dde03268a2447d1df9dba8ff05c48e14e70f62c785f8c0e0447 +5ecf18f584bdf50ca374bfe49104126da278b87286df2df5d90c73b64133502f +4b8098a0d8f94adf60ebf974800b44964e7232a690cfbde4614c4e05cc7a9d71 +84b42b5184434cece8408c5e0e8708938487adcee6b8622c0ba478dbc361edd2 +8502bc65a8e9b11996a912da4eb42378c57b6a89a0c940422a9e61e62e5932e1 +3ddd5113dd25ac5156fd99967e2cad71fd379b2b2ba32f43191431d50bae1023 +40cbacf6bf1de3d1a9fb786beda3fdb462efae97b54f42cd02c4d998f01e1f22 +4e1d7a2148a78f116442c3c654b875c7835fc8ec221d12685f91e27b44062667 +8b97c2be125d2d4d404c3383fd8072b25d46a482aaaccf55730a56742fb157e5 +119662c086ea71d1b6c89a910a71bd89349018e910395a3c645011374cc96a87 +3e132b65fd592f06d11a4882fa68a7d0d7d85965fb43dbc0a852489bed0c7608 +e6591e00399eed11156c4548a2764b323f67b98639a3f9d11ee7f9089f578005 +941491b05aa5f85b69b1833891f970f1cf0f203fbb5b80541c92e6c2f6473f88 +60adeb042c157c4d80c39f7301813207153cb31154630b05a9602ad8bc2e0741 +1875fb5a0954b458bdaaf2d2051848f0d20ddcfd1f274551d009829a4ca2a599 +f11d3ca9f33bc0c029aa718f1736e0ee854dd41b3cf0e23a03400955989e3f43 +9b7694276ae3f1f73274cb5b3aa730cacf4ca3f050402ba49f35b34ff8726557 +6b24b193560a087522e6e02b53fa1c32cf56b0c14e78c69bb9d529bf5e9c5a4c +25ce6b8d39d02471c43a804569a72015e81f3400170c1b8f5418e6938cd60b74 +8aeb368d2a29616c4fb689fd9f236d9b8587a5b091b18dfe79a40ea947dd1a31 +b51eeb48316bc7143e4deb2b2696b1fb2ca92516a008a7305668cfe491f86b8d +1fb4f0ca43898c25340ca1d504c92cb4187fac8d3154f9adfb437d332337b71e +499fd482997639f924eab76744d1b1c6c3901e6ff944057081b70c7eafda21ad +6f6ef4b81707e397dd1a032c3c3537b3bdaaf83f3d9e1360e3e4d8b273103b13 +00bbd67a57b7ed2e6f46a0bb0b31c221c0867943778db6922ee9cb4bccb1a931 +c94bc2d1b33fc6bdea57b3008f25f3cac75a82bbe644367b13391ec6eef37e93 +ed0da9ea9849d55b25ca6b7074d217aaf442f4bf2a7613b1a1c9b4c3ac07587d +39239656b9e496c7f8199908f229c5c6a2b7822d5c0d3bfd36fb2d6355e2cb1f +91235731f97f5dcf65d7b8cfc7b82b6ad2ee05c4229ea9879ffd693f7356f5d6 +3c096a9f245bcb0ff1ed90278d2752d11a822bf1813f896155519b0b237eb20e +3d2ce117aa4a1e0d271dbe98b8fe1a93d66a22f454611da6b0f903a29d07d86c +6089239513b862adabff013dcb5516fc4d591df9bd74cffec4dc82b5a0244414 +82759072950f9f88c64a7e33bd096899bc70d3d9ffc667b90b15d1698c2b06eb +47e49fb45d2940fd9e7a905cb76e42b51ce1e69d7b990ea3807e56d1c4bc8cd5 +a2ae606640dc06206b09f12e38419b64f640b4e538078971bb0758bf03ec65aa +0eb252c01c7a74e104b664fe14a99b6d6fc4ef5b9eb22a4deb6708030bb993ae +2331814ae1de551b87950fb6540fdc00258971360f341fa848c6cf951144aba9 +83ed46d4a973f050cf9a1fd62b25f58b5bf17d88e2eccfb6223753a44e6a60b2 +0e5aacff67c94420d3fdb5b04d324fadf73d6e4fcc8e3498fa752dc02aca7e01 +368f9821b2a55a4560865f0711624ab207158eed8b72777fd3081a5669704b10 +3d45cfcc8f8ea0126e1e84a4d84d3fb4ba4481c3fdaad535c797e059bf02d9ed +b26b80ba9aef7d5550c2aed207cab3fa463a3fad2fbde60570a17f8e054b10af +39dfedea1f573ed3d848a30afb525c41abddb167c4c2fc5e7723e0c67eb83b13 +d54763e40d7c730668e369beef8d67299eb4df9d4ff5bb3b98d7ca52539bcbc2 +5db4998af317f13c80bc0689c9d8708277767e9267b86b776abb26e5b3e7fd5e +c1da40532a16412890aa0d3d1be8e293a5abc073881ef11ca3d73afb78ba675d +fd569e6f14e5b61abc8b463a65d586cbdecb89d4069691f07e629c579ad97d78 +824fa1787cb630d21b94b07aeff37f04a550280c6b210a2141b1a75fed78a19f +90e470d2c5110f20fb9c2f4dc762de341f13821f2029544926712a421598f4a8 +e93efc232ae23f32afaa98704d4669d3efdc4332463de377c951f542b6232e35 +62baa93a61368d738ff19510d6bdefb483d22db1abea00b8a144a86a538c9043 +1da883051a17ab3a8cb2fa64859fbb717aa481a1c821d33e2717a746f1fbdea3 +27fde813f981a417db07f007393d75b6d877b39cf07cc3ccb08949531b5ac443 +005a926f9b47c844bfa40aa36af1f5c5ed969de1dcaf0817e27b8e8114853bfa +435f717ba0f812cd4032e22feeea49b4ab2a49aab092254923a6152ad303f462 +ce90c1a446bdc7360ca1b51677eb66cb8de05e825152f6b79863fe10295dd785 +e891e36dbd2300630b861c14d633fe3a9855814662ccef52dbee6cab61146346 +bc6dac0eab42c59638a7b0df16e92d5b35535ef6cb925a1cb594d739996b12ba +0ed889d89a873a3b3872b85aeb438331f9f34dbd833a8e0476342df14cc5bf50 +d493c02ede0463a74aaf120dd1cf140c2e7f5b119ec3556ebc5eb6994bead7aa +8a8c77517ae397e1228d9e88d692ce7846f723717379087d811b5da4d4142fdf +fe5b5f5aa3a2651aa69439b3c1c50b168c815c8ccf62eb689ffc325292b52036 +b419e5abfea56ae0b6c0b5ef75633514de82073a41edfc50fe2f9fd1baa4594a +471c98dc80ba06d9e688692ac09fef073be040534f8fe842918b4607b285d9ab +55c895644c2474fad12301a9a9385df92779850efec50d43363109ad501eccb2 +af47c709f231ac70d3729d3d798fc1ab180720568e0d5469280fc6df9456e481 +57a7c1d46ca2bd284d8dbe0f415d0f961a6b02bf134cab8f480346551a028ccd +34b0b230c0de68bc37b137f39fb33fbb6ce7fe301281e72c9fc2cbd06b100680 +71cceb56ecde070538cb20f4a49590690c7f48d9ada1335aa91572ab34708338 +cdaab03910dac6288e916113ff03bcd1c23b0be97f57ebee7ce9b09dff582296 +fddcf1e50a67ac5416609d603e29ba40d250bf67ae9c25ebeb587b82fbbf0373 +a6c482493b714e1380de0933b638d8c5fe911d1408e2443e31ac6b2f76e99c10 +53c67086b5d5aca3d64a8cbcc72a47d6d68b5a8ab7c03091eb272766c01ecee2 +97f76ec97efcc942b172142455f3b20f10742c6d8a6011037657a1d5d5eb4ea4 +0adc4a57d322771a90f93e7b907b6a3510c8065b162bdba54c6d0a9b7e9d1126 +f46c0b0dfc887b3fa919cabccad0d37e06e9820f9b9d420e430165a80da44006 +26dbcffef067af792cb6f3bbdca66fa8ac765f44c02fbe06efcf37909a681842 +cf3d6b3ddc50fc2db5a91659950979c7bde4cbf0640600e6edab37e4548f33b4 +1bead20256df5fbe932c987cc8b2fbc90c74ebc945f8951a1793172b8dea40dc +4c90ba5341aab5f292b4098b373bd85284fc971ef5f542a5c23dfe623f5dc3fe +4d0d75e7381aec27db8239c2771d05e9f689c14b9e13db6ad6273060ea9caedb +96c4c195d0556bd414a88f4cf99fb43f005d8c5fbe4c59f836709b9d709f50b1 +1ece66443e87cc9694a4af0b4fa55f99d0fd444eb84d0bb4e6904d355dc163f3 +6bdd2c7f1ec7f16fd39aa259e3290c27b7528d6f267d82a928b8dcc5d5d2932a +d7b1c521c8dad3de547b8ab06dfb5cc87546f983503421e7e7cbb55d9df264b3 +3cbc09c1a9d11af20dc25ab018a77d33e878029ee76f0bc2d358f23630274d60 +f5c11fab01dfba033bf5f6d7ac7a8305a5d6e156f541a6556d61b02411fc591a +9b8fc94e029b8594ca10de2bbf2488403dc0e45e68186bbc7c64376308743efa +65a4eff69fbf97128816534c40cbd3e4bc4eb1559d9cb314dad354a55661c03f +2c6c58338eddec9db1d914e214446ea718002e30bb5e7334c444b970277ad264 +35ff769d552b6bf8cb43b87c5b9cf928ef853d6c9e4622b5fc46412f509b65a6 +99708d0ba80cc26e8e025e3b1e0e4794e52fa7b94604fcedb9b1c77da16a4958 +956ee08e61f6e266e8de04145cca8a5a1f802cbfb5a93a2f175e501948ff1d1e +0007f518fef4069ed7afe6f093fc73da3447133d5d6abd59c1978a2b597b6aa6 +165498741a253759a44cc7458c9465fc9fee103c90d40d54554b9c9e699d93e9 +30604dde9d5aca8b3afe3085e006be1c315d70726dd5764e7f5a6d633e825a80 +ee967c6d8d73bcc91e49c269d94f96eaa8eea5dceb2a1457c0bd1987682fc5ca +ff1c53cc990d72ca0945b9b9a90b3441b5dffe8945e66b15d726d8cc5eadb90d +c67f93f277fe77cd78ea15fa184fa70f352a82508ad2fa66d972ddf49c08d5c2 +09b341f61a0f40aef8442abe48eece12bb3844178eeee55aee73f003d3e9c37d +157c905fbbbbc2653a371f4b1d8e0672963e45d0baa01cb81ad9b41115eef281 +d9c35922a4c7f0f6e0c7ae397232009375505686d13fabf211f72730977ebcd8 +7da5fd597c97f37ed5e8553ac7f140ca877eb6a4fbcb351c3d070996088bc4da +fd73a72d5d89bdb2048f0f1ad0de85ec4df9b47b422b075caf611886e6ad7d5a +7b145b7994e8aaeb5558086d9991f61bdaa61e5e1a1f8ce9f2801491b5df289c +b0b6560074404e15ae2bba4f10f8730be816c9ba717ea5a6df06bf6d4016a665 +5246ea08c3eca4fb23e14d1d2a040fc55ea7834223eb7d235b2d47eb950f5574 +6ad2432e7390f323434708677009a5a9f160f01d1483c1d84d310256cd623be7 +dc5a626c9414c1d79c0071ee763f621e3a9fac9351bb34ac3cec3241fd5c4cc0 +3371c6a0843d8222b78a48f0a6403636d1ee9cde5b45e224d5141ef3b492f9c0 +15229d101ee197bf179a9e11235901992d486cd99aafa632086733afb1e8008d +99caac952114658b7346bb27358582bccaa3ab9f24543ee981742a0439ee29f0 +cb0e2ffdb432fa68b7c0046235053f3d3a339dcf1c58fbb16cd182aeca0f4734 +b555084484f2a68e036af36c9c10aa4f449a00192e554151abdc3192e188f3d0 +91e8f3330a08b726ae815fa05119e4cd53f329c018306b8568571399314a7ee9 +b730f37f448dea4d9e18ec3e8bb0886a3d0257bd2c8a42a67a5a96815f586c61 +19c40dff6bfee010ca3d3f41461b540109ec1d5d9fbb70584810eb2e4226666c +65502666bae303206d40cb1b8bf2500b6698a8d4a66289b791468f4f49703b7c +698675daa46546737422856fe2eb81fcb4740eb19d1f4cf7837c27603c9f9ce7 +840a97091ab8d4f782c0daf9f3c7f67eaf7eae595a42d5fe0f7778a4cb3797b6 +a9b7552649c39dfb83b9acab7d7cc8529e8b0438b7919e9a9360de28a64c5f02 \ No newline at end of file diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..6fa40af --- /dev/null +++ b/package-lock.json @@ -0,0 +1,182 @@ +{ + "name": "code-challenge-2024-amitx13", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "dependencies": { + "bech32-buffer": "^0.2.1", + "js-sha256": "^0.11.0", + "ripemd160": "^2.0.2", + "secp256k1": "^5.0.0" + } + }, + "node_modules/bech32-buffer": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/bech32-buffer/-/bech32-buffer-0.2.1.tgz", + "integrity": "sha512-fCG1TyZuCN48Sdw97p/IR39fvqpFlWDVpG7qnuU1Uc3+Xtc/0uqAp8U7bMW/bGuVF5CcNVIXwxQsWwUr6un6FQ==", + "engines": { + "node": ">=8" + } + }, + "node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/brorand": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz", + "integrity": "sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==" + }, + "node_modules/elliptic": { + "version": "6.5.5", + "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.5.tgz", + "integrity": "sha512-7EjbcmUm17NQFu4Pmgmq2olYMj8nwMnpcddByChSUjArp8F5DQWcIcpriwO4ZToLNAJig0yiyjswfyGNje/ixw==", + "dependencies": { + "bn.js": "^4.11.9", + "brorand": "^1.1.0", + "hash.js": "^1.0.0", + "hmac-drbg": "^1.0.1", + "inherits": "^2.0.4", + "minimalistic-assert": "^1.0.1", + "minimalistic-crypto-utils": "^1.0.1" + } + }, + "node_modules/hash-base": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/hash-base/-/hash-base-3.1.0.tgz", + "integrity": "sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA==", + "dependencies": { + "inherits": "^2.0.4", + "readable-stream": "^3.6.0", + "safe-buffer": "^5.2.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/hash.js": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz", + "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==", + "dependencies": { + "inherits": "^2.0.3", + "minimalistic-assert": "^1.0.1" + } + }, + "node_modules/hmac-drbg": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz", + "integrity": "sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==", + "dependencies": { + "hash.js": "^1.0.3", + "minimalistic-assert": "^1.0.0", + "minimalistic-crypto-utils": "^1.0.1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "node_modules/js-sha256": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/js-sha256/-/js-sha256-0.11.0.tgz", + "integrity": "sha512-6xNlKayMZvds9h1Y1VWc0fQHQ82BxTXizWPEtEeGvmOUYpBRy4gbWroHLpzowe6xiQhHpelCQiE7HEdznyBL9Q==" + }, + "node_modules/minimalistic-assert": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz", + "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==" + }, + "node_modules/minimalistic-crypto-utils": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz", + "integrity": "sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==" + }, + "node_modules/node-addon-api": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-5.1.0.tgz", + "integrity": "sha512-eh0GgfEkpnoWDq+VY8OyvYhFEzBk6jIYbRKdIlyTiAXIVJ8PyBaKb0rp7oDtoddbdoHWhq8wwr+XZ81F1rpNdA==" + }, + "node_modules/node-gyp-build": { + "version": "4.8.0", + "resolved": "https://registry.npmjs.org/node-gyp-build/-/node-gyp-build-4.8.0.tgz", + "integrity": "sha512-u6fs2AEUljNho3EYTJNBfImO5QTo/J/1Etd+NVdCj7qWKUSN/bSLkZwhDv7I+w/MSC6qJ4cknepkAYykDdK8og==", + "bin": { + "node-gyp-build": "bin.js", + "node-gyp-build-optional": "optional.js", + "node-gyp-build-test": "build-test.js" + } + }, + "node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/ripemd160": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/ripemd160/-/ripemd160-2.0.2.tgz", + "integrity": "sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==", + "dependencies": { + "hash-base": "^3.0.0", + "inherits": "^2.0.1" + } + }, + "node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/secp256k1": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/secp256k1/-/secp256k1-5.0.0.tgz", + "integrity": "sha512-TKWX8xvoGHrxVdqbYeZM9w+izTF4b9z3NhSaDkdn81btvuh+ivbIMGT/zQvDtTFWhRlThpoz6LEYTr7n8A5GcA==", + "hasInstallScript": true, + "dependencies": { + "elliptic": "^6.5.4", + "node-addon-api": "^5.0.0", + "node-gyp-build": "^4.2.0" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/string_decoder": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", + "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", + "dependencies": { + "safe-buffer": "~5.2.0" + } + }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==" + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..78105cc --- /dev/null +++ b/package.json @@ -0,0 +1,8 @@ +{ + "dependencies": { + "bech32-buffer": "^0.2.1", + "js-sha256": "^0.11.0", + "ripemd160": "^2.0.2", + "secp256k1": "^5.0.0" + } +} diff --git a/run.sh b/run.sh index 721aeb2..dd73152 100644 --- a/run.sh +++ b/run.sh @@ -1 +1,8 @@ -# Update this file to run your own code \ No newline at end of file +# Update this file to run your own code +#!/bin/bash + +# Install dependencies +npm install + +# Run the project +node index.js