getblocktemplate: Explicitly handle the distinction between GBT-affecting softforks vs not

This commit is contained in:
Luke Dashjr 2016-06-01 16:47:36 +00:00
parent 72cd6b20ca
commit 98790608a4
3 changed files with 34 additions and 2 deletions

View File

@ -264,7 +264,9 @@ static UniValue BIP22ValidationResult(const CValidationState& state)
std::string gbt_vb_name(const Consensus::DeploymentPos pos) { std::string gbt_vb_name(const Consensus::DeploymentPos pos) {
const struct BIP9DeploymentInfo& vbinfo = VersionBitsDeploymentInfo[pos]; const struct BIP9DeploymentInfo& vbinfo = VersionBitsDeploymentInfo[pos];
std::string s = vbinfo.name; std::string s = vbinfo.name;
if (!vbinfo.gbt_force) {
s.insert(s.begin(), '!'); s.insert(s.begin(), '!');
}
return s; return s;
} }
@ -342,6 +344,7 @@ UniValue getblocktemplate(const UniValue& params, bool fHelp)
std::string strMode = "template"; std::string strMode = "template";
UniValue lpval = NullUniValue; UniValue lpval = NullUniValue;
std::set<std::string> setClientRules;
if (params.size() > 0) if (params.size() > 0)
{ {
const UniValue& oparam = params[0].get_obj(); const UniValue& oparam = params[0].get_obj();
@ -385,6 +388,14 @@ UniValue getblocktemplate(const UniValue& params, bool fHelp)
TestBlockValidity(state, Params(), block, pindexPrev, false, true); TestBlockValidity(state, Params(), block, pindexPrev, false, true);
return BIP22ValidationResult(state); return BIP22ValidationResult(state);
} }
const UniValue& aClientRules = find_value(oparam, "rules");
if (aClientRules.isArray()) {
for (unsigned int i = 0; i < aClientRules.size(); ++i) {
const UniValue& v = aClientRules[i];
setClientRules.insert(v.get_str());
}
}
} }
if (strMode != "template") if (strMode != "template")
@ -544,15 +555,32 @@ UniValue getblocktemplate(const UniValue& params, bool fHelp)
pblock->nVersion |= VersionBitsMask(consensusParams, pos); pblock->nVersion |= VersionBitsMask(consensusParams, pos);
// FALL THROUGH to get vbavailable set... // FALL THROUGH to get vbavailable set...
case THRESHOLD_STARTED: case THRESHOLD_STARTED:
// Add to vbavailable (and it's presumably in version already) {
const struct BIP9DeploymentInfo& vbinfo = VersionBitsDeploymentInfo[pos];
vbavailable.push_back(Pair(gbt_vb_name(pos), consensusParams.vDeployments[pos].bit)); vbavailable.push_back(Pair(gbt_vb_name(pos), consensusParams.vDeployments[pos].bit));
if (setClientRules.find(vbinfo.name) == setClientRules.end()) {
if (!vbinfo.gbt_force) {
// If the client doesn't support this, don't indicate it in the [default] version
pblock->nVersion &= ~VersionBitsMask(consensusParams, pos);
}
}
break; break;
}
case THRESHOLD_ACTIVE: case THRESHOLD_ACTIVE:
{
// Add to rules only // Add to rules only
const struct BIP9DeploymentInfo& vbinfo = VersionBitsDeploymentInfo[pos];
aRules.push_back(gbt_vb_name(pos)); aRules.push_back(gbt_vb_name(pos));
if (setClientRules.find(vbinfo.name) == setClientRules.end()) {
// Not supported by the client; make sure it's safe to proceed
if (!vbinfo.gbt_force) {
throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Support for '%s' rule requires explicit client support", vbinfo.name));
}
}
break; break;
} }
} }
}
result.push_back(Pair("version", pblock->nVersion)); result.push_back(Pair("version", pblock->nVersion));
result.push_back(Pair("rules", aRules)); result.push_back(Pair("rules", aRules));
result.push_back(Pair("vbavailable", vbavailable)); result.push_back(Pair("vbavailable", vbavailable));

View File

@ -9,9 +9,11 @@
const struct BIP9DeploymentInfo VersionBitsDeploymentInfo[Consensus::MAX_VERSION_BITS_DEPLOYMENTS] = { const struct BIP9DeploymentInfo VersionBitsDeploymentInfo[Consensus::MAX_VERSION_BITS_DEPLOYMENTS] = {
{ {
/*.name =*/ "testdummy", /*.name =*/ "testdummy",
/*.gbt_force =*/ true,
}, },
{ {
/*.name =*/ "csv", /*.name =*/ "csv",
/*.gbt_force =*/ true,
} }
}; };

View File

@ -33,6 +33,8 @@ typedef std::map<const CBlockIndex*, ThresholdState> ThresholdConditionCache;
struct BIP9DeploymentInfo { struct BIP9DeploymentInfo {
/** Deployment name */ /** Deployment name */
const char *name; const char *name;
/** Whether GBT clients can safely ignore this rule in simplified usage */
bool gbt_force;
}; };
extern const struct BIP9DeploymentInfo VersionBitsDeploymentInfo[]; extern const struct BIP9DeploymentInfo VersionBitsDeploymentInfo[];