Skip to content

Commit a4bf840

Browse files
committed
Separate WitnessV1Taproot variant in CTxDestination
1 parent 41839bd commit a4bf840

File tree

6 files changed

+45
-10
lines changed

6 files changed

+45
-10
lines changed

src/key_io.cpp

Lines changed: 15 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,14 @@ class DestinationEncoder
5454
return bech32::Encode(bech32::Encoding::BECH32, m_params.Bech32HRP(), data);
5555
}
5656

57+
std::string operator()(const WitnessV1Taproot& tap) const
58+
{
59+
std::vector<unsigned char> data = {1};
60+
data.reserve(53);
61+
ConvertBits<8, 5, true>([&](unsigned char c) { data.push_back(c); }, tap.begin(), tap.end());
62+
return bech32::Encode(bech32::Encoding::BECH32M, m_params.Bech32HRP(), data);
63+
}
64+
5765
std::string operator()(const WitnessUnknown& id) const
5866
{
5967
if (id.version < 1 || id.version > 16 || id.length < 2 || id.length > 40) {
@@ -135,6 +143,13 @@ CTxDestination DecodeDestination(const std::string& str, const CChainParams& par
135143
return CNoDestination();
136144
}
137145

146+
if (version == 1 && data.size() == WITNESS_V1_TAPROOT_SIZE) {
147+
static_assert(WITNESS_V1_TAPROOT_SIZE == WitnessV1Taproot::size());
148+
WitnessV1Taproot tap;
149+
std::copy(data.begin(), data.end(), tap.begin());
150+
return tap;
151+
}
152+
138153
if (version > 16) {
139154
error_str = "Invalid Bech32 address witness version";
140155
return CNoDestination();

src/rpc/util.cpp

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -301,6 +301,16 @@ class DescribeAddressVisitor
301301
return obj;
302302
}
303303

304+
UniValue operator()(const WitnessV1Taproot& tap) const
305+
{
306+
UniValue obj(UniValue::VOBJ);
307+
obj.pushKV("isscript", true);
308+
obj.pushKV("iswitness", true);
309+
obj.pushKV("witness_version", 1);
310+
obj.pushKV("witness_program", HexStr(tap));
311+
return obj;
312+
}
313+
304314
UniValue operator()(const WitnessUnknown& id) const
305315
{
306316
UniValue obj(UniValue::VOBJ);

src/script/descriptor.cpp

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -645,6 +645,7 @@ static std::optional<OutputType> OutputTypeFromDestination(const CTxDestination&
645645
}
646646
if (std::holds_alternative<WitnessV0KeyHash>(dest) ||
647647
std::holds_alternative<WitnessV0ScriptHash>(dest) ||
648+
std::holds_alternative<WitnessV1Taproot>(dest) ||
648649
std::holds_alternative<WitnessUnknown>(dest)) {
649650
return OutputType::BECH32;
650651
}

src/script/standard.cpp

Lines changed: 8 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -242,13 +242,9 @@ bool ExtractDestination(const CScript& scriptPubKey, CTxDestination& addressRet)
242242
return true;
243243
}
244244
case TxoutType::WITNESS_V1_TAPROOT: {
245-
/* For now, no WitnessV1Taproot variant in CTxDestination exists, so map
246-
* this to WitnessUnknown. */
247-
WitnessUnknown unk;
248-
unk.version = 1;
249-
std::copy(vSolutions[0].begin(), vSolutions[0].end(), unk.program);
250-
unk.length = vSolutions[0].size();
251-
addressRet = unk;
245+
WitnessV1Taproot tap;
246+
std::copy(vSolutions[0].begin(), vSolutions[0].end(), tap.begin());
247+
addressRet = tap;
252248
return true;
253249
}
254250
case TxoutType::WITNESS_UNKNOWN: {
@@ -337,6 +333,11 @@ class CScriptVisitor
337333
return CScript() << OP_0 << ToByteVector(id);
338334
}
339335

336+
CScript operator()(const WitnessV1Taproot& tap) const
337+
{
338+
return CScript() << OP_1 << ToByteVector(tap);
339+
}
340+
340341
CScript operator()(const WitnessUnknown& id) const
341342
{
342343
return CScript() << CScript::EncodeOP_N(id.version) << std::vector<unsigned char>(id.program, id.program + id.length);

src/script/standard.h

Lines changed: 10 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66
#ifndef BITCOIN_SCRIPT_STANDARD_H
77
#define BITCOIN_SCRIPT_STANDARD_H
88

9+
#include <pubkey.h>
910
#include <script/interpreter.h>
1011
#include <uint256.h>
1112
#include <util/hash_type.h>
@@ -113,6 +114,12 @@ struct WitnessV0KeyHash : public BaseHash<uint160>
113114
};
114115
CKeyID ToKeyID(const WitnessV0KeyHash& key_hash);
115116

117+
struct WitnessV1Taproot : public XOnlyPubKey
118+
{
119+
WitnessV1Taproot() : XOnlyPubKey() {}
120+
explicit WitnessV1Taproot(const XOnlyPubKey& xpk) : XOnlyPubKey(xpk) {}
121+
};
122+
116123
//! CTxDestination subtype to encode any future Witness version
117124
struct WitnessUnknown
118125
{
@@ -142,11 +149,11 @@ struct WitnessUnknown
142149
* * ScriptHash: TxoutType::SCRIPTHASH destination (P2SH)
143150
* * WitnessV0ScriptHash: TxoutType::WITNESS_V0_SCRIPTHASH destination (P2WSH)
144151
* * WitnessV0KeyHash: TxoutType::WITNESS_V0_KEYHASH destination (P2WPKH)
145-
* * WitnessUnknown: TxoutType::WITNESS_UNKNOWN/WITNESS_V1_TAPROOT destination (P2W???)
146-
* (taproot outputs do not require their own type as long as no wallet support exists)
152+
* * WitnessV1Taproot: TxoutType::WITNESS_V1_TAPROOT destination (P2TR)
153+
* * WitnessUnknown: TxoutType::WITNESS_UNKNOWN destination (P2W???)
147154
* A CTxDestination is the internal data type encoded in a bitcoin address
148155
*/
149-
using CTxDestination = std::variant<CNoDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessUnknown>;
156+
using CTxDestination = std::variant<CNoDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, WitnessUnknown>;
150157

151158
/** Check whether a CTxDestination is a CNoDestination. */
152159
bool IsValidDestination(const CTxDestination& dest);

src/wallet/rpcwallet.cpp

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -3737,6 +3737,7 @@ class DescribeWalletAddressVisitor
37373737
return obj;
37383738
}
37393739

3740+
UniValue operator()(const WitnessV1Taproot& id) const { return UniValue(UniValue::VOBJ); }
37403741
UniValue operator()(const WitnessUnknown& id) const { return UniValue(UniValue::VOBJ); }
37413742
};
37423743

0 commit comments

Comments
 (0)