Coverage Report

Created: 2026-04-29 19:21

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/tmp/bitcoin/src/compressor.h
Line
Count
Source
1
// Copyright (c) 2009-2010 Satoshi Nakamoto
2
// Copyright (c) 2009-present The Bitcoin Core developers
3
// Distributed under the MIT software license, see the accompanying
4
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
5
6
#ifndef BITCOIN_COMPRESSOR_H
7
#define BITCOIN_COMPRESSOR_H
8
9
#include <prevector.h>
10
#include <primitives/transaction.h>
11
#include <script/script.h>
12
#include <serialize.h>
13
#include <span.h>
14
15
/**
16
 * This saves us from making many heap allocations when serializing
17
 * and deserializing compressed scripts.
18
 *
19
 * This prevector size is determined by the largest .resize() in the
20
 * CompressScript function. The largest compressed script format is a
21
 * compressed public key, which is 33 bytes.
22
 */
23
using CompressedScript = prevector<33, unsigned char>;
24
25
26
bool CompressScript(const CScript& script, CompressedScript& out);
27
unsigned int GetSpecialScriptSize(unsigned int nSize);
28
bool DecompressScript(CScript& script, unsigned int nSize, const CompressedScript& in);
29
30
/**
31
 * Compress amount.
32
 *
33
 * nAmount is of type uint64_t and thus cannot be negative. If you're passing in
34
 * a CAmount (int64_t), make sure to properly handle the case where the amount
35
 * is negative before calling CompressAmount(...).
36
 *
37
 * @pre Function defined only for 0 <= nAmount <= MAX_MONEY.
38
 */
39
uint64_t CompressAmount(uint64_t nAmount);
40
41
uint64_t DecompressAmount(uint64_t nAmount);
42
43
/** Compact serializer for scripts.
44
 *
45
 *  It detects common cases and encodes them much more efficiently.
46
 *  3 special cases are defined:
47
 *  * Pay to pubkey hash (encoded as 21 bytes)
48
 *  * Pay to script hash (encoded as 21 bytes)
49
 *  * Pay to pubkey starting with 0x02, 0x03 or 0x04 (encoded as 33 bytes)
50
 *
51
 *  Other scripts up to 121 bytes require 1 byte + script length. Above
52
 *  that, scripts up to 16505 bytes require 2 bytes + script length.
53
 */
54
struct ScriptCompression
55
{
56
    /**
57
     * make this static for now (there are only 6 special scripts defined)
58
     * this can potentially be extended together with a new version for
59
     * transactions, in which case this value becomes dependent on version
60
     * and nHeight of the enclosing transaction.
61
     */
62
    static const unsigned int nSpecialScripts = 6;
63
64
    template<typename Stream>
65
482k
    void Ser(Stream &s, const CScript& script) {
66
482k
        CompressedScript compr;
67
482k
        if (CompressScript(script, compr)) {
68
127k
            s << std::span{compr};
69
127k
            return;
70
127k
        }
71
355k
        unsigned int nSize = script.size() + nSpecialScripts;
72
355k
        s << VARINT(nSize);
73
355k
        s << std::span{script};
74
355k
    }
void ScriptCompression::Ser<SizeComputer>(SizeComputer&, CScript const&)
Line
Count
Source
65
67.6k
    void Ser(Stream &s, const CScript& script) {
66
67.6k
        CompressedScript compr;
67
67.6k
        if (CompressScript(script, compr)) {
68
8.90k
            s << std::span{compr};
69
8.90k
            return;
70
8.90k
        }
71
58.7k
        unsigned int nSize = script.size() + nSpecialScripts;
72
58.7k
        s << VARINT(nSize);
73
58.7k
        s << std::span{script};
74
58.7k
    }
void ScriptCompression::Ser<HashWriter>(HashWriter&, CScript const&)
Line
Count
Source
65
67.6k
    void Ser(Stream &s, const CScript& script) {
66
67.6k
        CompressedScript compr;
67
67.6k
        if (CompressScript(script, compr)) {
68
8.90k
            s << std::span{compr};
69
8.90k
            return;
70
8.90k
        }
71
58.7k
        unsigned int nSize = script.size() + nSpecialScripts;
72
58.7k
        s << VARINT(nSize);
73
58.7k
        s << std::span{script};
74
58.7k
    }
void ScriptCompression::Ser<BufferedWriter<AutoFile>>(BufferedWriter<AutoFile>&, CScript const&)
Line
Count
Source
65
67.6k
    void Ser(Stream &s, const CScript& script) {
66
67.6k
        CompressedScript compr;
67
67.6k
        if (CompressScript(script, compr)) {
68
8.90k
            s << std::span{compr};
69
8.90k
            return;
70
8.90k
        }
71
58.7k
        unsigned int nSize = script.size() + nSpecialScripts;
72
58.7k
        s << VARINT(nSize);
73
58.7k
        s << std::span{script};
74
58.7k
    }
void ScriptCompression::Ser<AutoFile>(AutoFile&, CScript const&)
Line
Count
Source
65
6.58k
    void Ser(Stream &s, const CScript& script) {
66
6.58k
        CompressedScript compr;
67
6.58k
        if (CompressScript(script, compr)) {
68
6.11k
            s << std::span{compr};
69
6.11k
            return;
70
6.11k
        }
71
472
        unsigned int nSize = script.size() + nSpecialScripts;
72
472
        s << VARINT(nSize);
73
472
        s << std::span{script};
74
472
    }
void ScriptCompression::Ser<DataStream>(DataStream&, CScript const&)
Line
Count
Source
65
272k
    void Ser(Stream &s, const CScript& script) {
66
272k
        CompressedScript compr;
67
272k
        if (CompressScript(script, compr)) {
68
94.1k
            s << std::span{compr};
69
94.1k
            return;
70
94.1k
        }
71
178k
        unsigned int nSize = script.size() + nSpecialScripts;
72
178k
        s << VARINT(nSize);
73
178k
        s << std::span{script};
74
178k
    }
75
76
    template<typename Stream>
77
339k
    void Unser(Stream &s, CScript& script) {
78
339k
        unsigned int nSize = 0;
79
339k
        s >> VARINT(nSize);
80
339k
        if (nSize < nSpecialScripts) {
81
85.7k
            CompressedScript vch(GetSpecialScriptSize(nSize), 0x00);
82
85.7k
            s >> std::span{vch};
83
85.7k
            DecompressScript(script, nSize, vch);
84
85.7k
            return;
85
85.7k
        }
86
253k
        nSize -= nSpecialScripts;
87
253k
        if (nSize > MAX_SCRIPT_SIZE) {
88
            // Overly long script, replace with a short invalid one
89
1
            script << OP_RETURN;
90
1
            s.ignore(nSize);
91
253k
        } else {
92
253k
            script.resize(nSize);
93
253k
            s >> std::span{script};
94
253k
        }
95
253k
    }
void ScriptCompression::Unser<SpanReader>(SpanReader&, CScript&)
Line
Count
Source
77
83.5k
    void Unser(Stream &s, CScript& script) {
78
83.5k
        unsigned int nSize = 0;
79
83.5k
        s >> VARINT(nSize);
80
83.5k
        if (nSize < nSpecialScripts) {
81
4.78k
            CompressedScript vch(GetSpecialScriptSize(nSize), 0x00);
82
4.78k
            s >> std::span{vch};
83
4.78k
            DecompressScript(script, nSize, vch);
84
4.78k
            return;
85
4.78k
        }
86
78.7k
        nSize -= nSpecialScripts;
87
78.7k
        if (nSize > MAX_SCRIPT_SIZE) {
88
            // Overly long script, replace with a short invalid one
89
1
            script << OP_RETURN;
90
1
            s.ignore(nSize);
91
78.7k
        } else {
92
78.7k
            script.resize(nSize);
93
78.7k
            s >> std::span{script};
94
78.7k
        }
95
78.7k
    }
void ScriptCompression::Unser<AutoFile>(AutoFile&, CScript&)
Line
Count
Source
77
6.35k
    void Unser(Stream &s, CScript& script) {
78
6.35k
        unsigned int nSize = 0;
79
6.35k
        s >> VARINT(nSize);
80
6.35k
        if (nSize < nSpecialScripts) {
81
5.61k
            CompressedScript vch(GetSpecialScriptSize(nSize), 0x00);
82
5.61k
            s >> std::span{vch};
83
5.61k
            DecompressScript(script, nSize, vch);
84
5.61k
            return;
85
5.61k
        }
86
736
        nSize -= nSpecialScripts;
87
736
        if (nSize > MAX_SCRIPT_SIZE) {
88
            // Overly long script, replace with a short invalid one
89
0
            script << OP_RETURN;
90
0
            s.ignore(nSize);
91
736
        } else {
92
736
            script.resize(nSize);
93
736
            s >> std::span{script};
94
736
        }
95
736
    }
void ScriptCompression::Unser<HashVerifier<BufferedReader<AutoFile>>>(HashVerifier<BufferedReader<AutoFile>>&, CScript&)
Line
Count
Source
77
27.8k
    void Unser(Stream &s, CScript& script) {
78
27.8k
        unsigned int nSize = 0;
79
27.8k
        s >> VARINT(nSize);
80
27.8k
        if (nSize < nSpecialScripts) {
81
5.02k
            CompressedScript vch(GetSpecialScriptSize(nSize), 0x00);
82
5.02k
            s >> std::span{vch};
83
5.02k
            DecompressScript(script, nSize, vch);
84
5.02k
            return;
85
5.02k
        }
86
22.8k
        nSize -= nSpecialScripts;
87
22.8k
        if (nSize > MAX_SCRIPT_SIZE) {
88
            // Overly long script, replace with a short invalid one
89
0
            script << OP_RETURN;
90
0
            s.ignore(nSize);
91
22.8k
        } else {
92
22.8k
            script.resize(nSize);
93
22.8k
            s >> std::span{script};
94
22.8k
        }
95
22.8k
    }
void ScriptCompression::Unser<DataStream>(DataStream&, CScript&)
Line
Count
Source
77
221k
    void Unser(Stream &s, CScript& script) {
78
221k
        unsigned int nSize = 0;
79
221k
        s >> VARINT(nSize);
80
221k
        if (nSize < nSpecialScripts) {
81
70.2k
            CompressedScript vch(GetSpecialScriptSize(nSize), 0x00);
82
70.2k
            s >> std::span{vch};
83
70.2k
            DecompressScript(script, nSize, vch);
84
70.2k
            return;
85
70.2k
        }
86
151k
        nSize -= nSpecialScripts;
87
151k
        if (nSize > MAX_SCRIPT_SIZE) {
88
            // Overly long script, replace with a short invalid one
89
0
            script << OP_RETURN;
90
0
            s.ignore(nSize);
91
151k
        } else {
92
151k
            script.resize(nSize);
93
151k
            s >> std::span{script};
94
151k
        }
95
151k
    }
96
};
97
98
struct AmountCompression
99
{
100
    template<typename Stream, typename I> void Ser(Stream& s, I val)
101
482k
    {
102
482k
        s << VARINT(CompressAmount(val));
103
482k
    }
void AmountCompression::Ser<SizeComputer, long>(SizeComputer&, long)
Line
Count
Source
101
67.6k
    {
102
67.6k
        s << VARINT(CompressAmount(val));
103
67.6k
    }
void AmountCompression::Ser<HashWriter, long>(HashWriter&, long)
Line
Count
Source
101
67.6k
    {
102
67.6k
        s << VARINT(CompressAmount(val));
103
67.6k
    }
void AmountCompression::Ser<BufferedWriter<AutoFile>, long>(BufferedWriter<AutoFile>&, long)
Line
Count
Source
101
67.6k
    {
102
67.6k
        s << VARINT(CompressAmount(val));
103
67.6k
    }
void AmountCompression::Ser<AutoFile, long>(AutoFile&, long)
Line
Count
Source
101
6.58k
    {
102
6.58k
        s << VARINT(CompressAmount(val));
103
6.58k
    }
void AmountCompression::Ser<DataStream, long>(DataStream&, long)
Line
Count
Source
101
272k
    {
102
272k
        s << VARINT(CompressAmount(val));
103
272k
    }
104
    template<typename Stream, typename I> void Unser(Stream& s, I& val)
105
339k
    {
106
339k
        uint64_t v;
107
339k
        s >> VARINT(v);
108
339k
        val = DecompressAmount(v);
109
339k
    }
void AmountCompression::Unser<SpanReader, long>(SpanReader&, long&)
Line
Count
Source
105
83.5k
    {
106
83.5k
        uint64_t v;
107
83.5k
        s >> VARINT(v);
108
83.5k
        val = DecompressAmount(v);
109
83.5k
    }
void AmountCompression::Unser<AutoFile, long>(AutoFile&, long&)
Line
Count
Source
105
6.35k
    {
106
6.35k
        uint64_t v;
107
6.35k
        s >> VARINT(v);
108
6.35k
        val = DecompressAmount(v);
109
6.35k
    }
void AmountCompression::Unser<HashVerifier<BufferedReader<AutoFile>>, long>(HashVerifier<BufferedReader<AutoFile>>&, long&)
Line
Count
Source
105
27.8k
    {
106
27.8k
        uint64_t v;
107
27.8k
        s >> VARINT(v);
108
27.8k
        val = DecompressAmount(v);
109
27.8k
    }
void AmountCompression::Unser<DataStream, long>(DataStream&, long&)
Line
Count
Source
105
221k
    {
106
221k
        uint64_t v;
107
221k
        s >> VARINT(v);
108
221k
        val = DecompressAmount(v);
109
221k
    }
110
};
111
112
/** wrapper for CTxOut that provides a more compact serialization */
113
struct TxOutCompression
114
{
115
821k
    FORMATTER_METHODS(CTxOut, obj) { READWRITE(Using<AmountCompression>(obj.nValue), Using<ScriptCompression>(obj.scriptPubKey)); }
void TxOutCompression::SerializationOps<SpanReader, CTxOut, ActionUnserialize>(CTxOut&, SpanReader&, ActionUnserialize)
Line
Count
Source
115
83.5k
    FORMATTER_METHODS(CTxOut, obj) { READWRITE(Using<AmountCompression>(obj.nValue), Using<ScriptCompression>(obj.scriptPubKey)); }
void TxOutCompression::SerializationOps<AutoFile, CTxOut, ActionUnserialize>(CTxOut&, AutoFile&, ActionUnserialize)
Line
Count
Source
115
6.35k
    FORMATTER_METHODS(CTxOut, obj) { READWRITE(Using<AmountCompression>(obj.nValue), Using<ScriptCompression>(obj.scriptPubKey)); }
void TxOutCompression::SerializationOps<HashVerifier<BufferedReader<AutoFile>>, CTxOut, ActionUnserialize>(CTxOut&, HashVerifier<BufferedReader<AutoFile>>&, ActionUnserialize)
Line
Count
Source
115
27.8k
    FORMATTER_METHODS(CTxOut, obj) { READWRITE(Using<AmountCompression>(obj.nValue), Using<ScriptCompression>(obj.scriptPubKey)); }
void TxOutCompression::SerializationOps<SizeComputer, CTxOut const, ActionSerialize>(CTxOut const&, SizeComputer&, ActionSerialize)
Line
Count
Source
115
67.6k
    FORMATTER_METHODS(CTxOut, obj) { READWRITE(Using<AmountCompression>(obj.nValue), Using<ScriptCompression>(obj.scriptPubKey)); }
void TxOutCompression::SerializationOps<HashWriter, CTxOut const, ActionSerialize>(CTxOut const&, HashWriter&, ActionSerialize)
Line
Count
Source
115
67.6k
    FORMATTER_METHODS(CTxOut, obj) { READWRITE(Using<AmountCompression>(obj.nValue), Using<ScriptCompression>(obj.scriptPubKey)); }
void TxOutCompression::SerializationOps<BufferedWriter<AutoFile>, CTxOut const, ActionSerialize>(CTxOut const&, BufferedWriter<AutoFile>&, ActionSerialize)
Line
Count
Source
115
67.6k
    FORMATTER_METHODS(CTxOut, obj) { READWRITE(Using<AmountCompression>(obj.nValue), Using<ScriptCompression>(obj.scriptPubKey)); }
void TxOutCompression::SerializationOps<AutoFile, CTxOut const, ActionSerialize>(CTxOut const&, AutoFile&, ActionSerialize)
Line
Count
Source
115
6.58k
    FORMATTER_METHODS(CTxOut, obj) { READWRITE(Using<AmountCompression>(obj.nValue), Using<ScriptCompression>(obj.scriptPubKey)); }
void TxOutCompression::SerializationOps<DataStream, CTxOut const, ActionSerialize>(CTxOut const&, DataStream&, ActionSerialize)
Line
Count
Source
115
272k
    FORMATTER_METHODS(CTxOut, obj) { READWRITE(Using<AmountCompression>(obj.nValue), Using<ScriptCompression>(obj.scriptPubKey)); }
void TxOutCompression::SerializationOps<DataStream, CTxOut, ActionUnserialize>(CTxOut&, DataStream&, ActionUnserialize)
Line
Count
Source
115
221k
    FORMATTER_METHODS(CTxOut, obj) { READWRITE(Using<AmountCompression>(obj.nValue), Using<ScriptCompression>(obj.scriptPubKey)); }
116
};
117
118
#endif // BITCOIN_COMPRESSOR_H