]> Git Repo - VerusCoin.git/blob - src/chain.h
Merge branch 'dev' into jl777
[VerusCoin.git] / src / chain.h
1 // Copyright (c) 2009-2010 Satoshi Nakamoto
2 // Copyright (c) 2009-2014 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_CHAIN_H
7 #define BITCOIN_CHAIN_H
8
9 #include "arith_uint256.h"
10 #include "primitives/block.h"
11 #include "pow.h"
12 #include "tinyformat.h"
13 #include "uint256.h"
14
15 #include <vector>
16
17 #include <boost/foreach.hpp>
18
19 static const int SPROUT_VALUE_VERSION = 1001400;
20
21 struct CDiskBlockPos
22 {
23     int nFile;
24     unsigned int nPos;
25
26     ADD_SERIALIZE_METHODS;
27
28     template <typename Stream, typename Operation>
29     inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion) {
30         READWRITE(VARINT(nFile));
31         READWRITE(VARINT(nPos));
32     }
33
34     CDiskBlockPos() {
35         SetNull();
36     }
37
38     CDiskBlockPos(int nFileIn, unsigned int nPosIn) {
39         nFile = nFileIn;
40         nPos = nPosIn;
41     }
42
43     friend bool operator==(const CDiskBlockPos &a, const CDiskBlockPos &b) {
44         return (a.nFile == b.nFile && a.nPos == b.nPos);
45     }
46
47     friend bool operator!=(const CDiskBlockPos &a, const CDiskBlockPos &b) {
48         return !(a == b);
49     }
50
51     void SetNull() { nFile = -1; nPos = 0; }
52     bool IsNull() const { return (nFile == -1); }
53
54     std::string ToString() const
55     {
56         return strprintf("CBlockDiskPos(nFile=%i, nPos=%i)", nFile, nPos);
57     }
58
59 };
60
61 enum BlockStatus: uint32_t {
62     //! Unused.
63     BLOCK_VALID_UNKNOWN      =    0,
64
65     //! Parsed, version ok, hash satisfies claimed PoW, 1 <= vtx count <= max, timestamp not in future
66     BLOCK_VALID_HEADER       =    1,
67
68     //! All parent headers found, difficulty matches, timestamp >= median previous, checkpoint. Implies all parents
69     //! are also at least TREE.
70     BLOCK_VALID_TREE         =    2,
71
72     /**
73      * Only first tx is coinbase, 2 <= coinbase input script length <= 100, transactions valid, no duplicate txids,
74      * sigops, size, merkle root. Implies all parents are at least TREE but not necessarily TRANSACTIONS. When all
75      * parent blocks also have TRANSACTIONS, CBlockIndex::nChainTx will be set.
76      */
77     BLOCK_VALID_TRANSACTIONS =    3,
78
79     //! Outputs do not overspend inputs, no double spends, coinbase output ok, no immature coinbase spends, BIP30.
80     //! Implies all parents are also at least CHAIN.
81     BLOCK_VALID_CHAIN        =    4,
82
83     //! Scripts & signatures ok. Implies all parents are also at least SCRIPTS.
84     BLOCK_VALID_SCRIPTS      =    5,
85
86     //! All validity bits.
87     BLOCK_VALID_MASK         =   BLOCK_VALID_HEADER | BLOCK_VALID_TREE | BLOCK_VALID_TRANSACTIONS |
88                                  BLOCK_VALID_CHAIN | BLOCK_VALID_SCRIPTS,
89
90     BLOCK_HAVE_DATA          =    8, //! full block available in blk*.dat
91     BLOCK_HAVE_UNDO          =   16, //! undo data available in rev*.dat
92     BLOCK_HAVE_MASK          =   BLOCK_HAVE_DATA | BLOCK_HAVE_UNDO,
93
94     BLOCK_FAILED_VALID       =   32, //! stage after last reached validness failed
95     BLOCK_FAILED_CHILD       =   64, //! descends from failed block
96     BLOCK_FAILED_MASK        =   BLOCK_FAILED_VALID | BLOCK_FAILED_CHILD,
97
98     BLOCK_ACTIVATES_UPGRADE  =   128, //! block activates a network upgrade
99 };
100
101 //! Short-hand for the highest consensus validity we implement.
102 //! Blocks with this validity are assumed to satisfy all consensus rules.
103 static const BlockStatus BLOCK_VALID_CONSENSUS = BLOCK_VALID_SCRIPTS;
104
105 /** The block chain is a tree shaped structure starting with the
106  * genesis block at the root, with each block potentially having multiple
107  * candidates to be the next block. A blockindex may have multiple pprev pointing
108  * to it, but at most one of them can be part of the currently active branch.
109  */
110 class CBlockIndex
111 {
112 public:
113     //! pointer to the hash of the block, if any. Memory is owned by this CBlockIndex
114     const uint256* phashBlock;
115
116     //! pointer to the index of the predecessor of this block
117     CBlockIndex* pprev;
118
119     //! pointer to the index of some further predecessor of this block
120     CBlockIndex* pskip;
121
122     //! height of the entry in the chain. The genesis block has height 0
123     int nHeight;
124
125     //! Which # file this block is stored in (blk?????.dat)
126     int nFile;
127
128     //! Byte offset within blk?????.dat where this block's data is stored
129     unsigned int nDataPos;
130
131     //! Byte offset within rev?????.dat where this block's undo data is stored
132     unsigned int nUndoPos;
133
134     //! (memory only) Total amount of work (expected number of hashes) in the chain up to and including this block
135     arith_uint256 nChainWork;
136
137     //! Number of transactions in this block.
138     //! Note: in a potential headers-first mode, this number cannot be relied upon
139     unsigned int nTx;
140
141     //! (memory only) Number of transactions in the chain up to and including this block.
142     //! This value will be non-zero only if and only if transactions for this block and all its parents are available.
143     //! Change to 64-bit type when necessary; won't happen before 2030
144     unsigned int nChainTx;
145
146     //! Verification status of this block. See enum BlockStatus
147     unsigned int nStatus;
148
149     //! Branch ID corresponding to the consensus rules used to validate this block.
150     //! Only cached if block validity is BLOCK_VALID_CONSENSUS.
151     //! Persisted at each activation height, memory-only for intervening blocks.
152     boost::optional<uint32_t> nCachedBranchId;
153
154     //! The anchor for the tree state up to the start of this block
155     uint256 hashAnchor;
156
157     //! (memory only) The anchor for the tree state up to the end of this block
158     uint256 hashAnchorEnd;
159
160     //! Change in value held by the Sprout circuit over this block.
161     //! Will be boost::none for older blocks on old nodes until a reindex has taken place.
162     boost::optional<CAmount> nSproutValue;
163
164     //! (memory only) Total value held by the Sprout circuit up to and including this block.
165     //! Will be boost::none for on old nodes until a reindex has taken place.
166     //! Will be boost::none if nChainTx is zero.
167     boost::optional<CAmount> nChainSproutValue;
168
169     //! block header
170     int nVersion;
171     uint256 hashMerkleRoot;
172     uint256 hashReserved;
173     unsigned int nTime;
174     unsigned int nBits;
175     uint256 nNonce;
176     std::vector<unsigned char> nSolution;
177
178     //! (memory only) Sequential id assigned to distinguish order in which blocks are received.
179     uint32_t nSequenceId;
180
181     void SetNull()
182     {
183         phashBlock = NULL;
184         pprev = NULL;
185         pskip = NULL;
186         nHeight = 0;
187         nFile = 0;
188         nDataPos = 0;
189         nUndoPos = 0;
190         nChainWork = arith_uint256();
191         nTx = 0;
192         nChainTx = 0;
193         nStatus = 0;
194         nCachedBranchId = boost::none;
195         hashAnchor = uint256();
196         hashAnchorEnd = uint256();
197         nSequenceId = 0;
198         nSproutValue = boost::none;
199         nChainSproutValue = boost::none;
200
201         nVersion       = 0;
202         hashMerkleRoot = uint256();
203         hashReserved   = uint256();
204         nTime          = 0;
205         nBits          = 0;
206         nNonce         = uint256();
207         nSolution.clear();
208     }
209
210     CBlockIndex()
211     {
212         SetNull();
213     }
214
215     CBlockIndex(const CBlockHeader& block)
216     {
217         SetNull();
218
219         nVersion       = block.nVersion;
220         hashMerkleRoot = block.hashMerkleRoot;
221         hashReserved   = block.hashReserved;
222         nTime          = block.nTime;
223         nBits          = block.nBits;
224         nNonce         = block.nNonce;
225         nSolution      = block.nSolution;
226     }
227
228     CDiskBlockPos GetBlockPos() const {
229         CDiskBlockPos ret;
230         if (nStatus & BLOCK_HAVE_DATA) {
231             ret.nFile = nFile;
232             ret.nPos  = nDataPos;
233         }
234         return ret;
235     }
236
237     CDiskBlockPos GetUndoPos() const {
238         CDiskBlockPos ret;
239         if (nStatus & BLOCK_HAVE_UNDO) {
240             ret.nFile = nFile;
241             ret.nPos  = nUndoPos;
242         }
243         return ret;
244     }
245
246     CBlockHeader GetBlockHeader() const
247     {
248         CBlockHeader block;
249         block.nVersion       = nVersion;
250         if (pprev)
251             block.hashPrevBlock = pprev->GetBlockHash();
252         block.hashMerkleRoot = hashMerkleRoot;
253         block.hashReserved   = hashReserved;
254         block.nTime          = nTime;
255         block.nBits          = nBits;
256         block.nNonce         = nNonce;
257         block.nSolution      = nSolution;
258         return block;
259     }
260
261     uint256 GetBlockHash() const
262     {
263         return *phashBlock;
264     }
265
266     int64_t GetBlockTime() const
267     {
268         return (int64_t)nTime;
269     }
270
271     enum { nMedianTimeSpan=11 };
272
273     int64_t GetMedianTimePast() const
274     {
275         int64_t pmedian[nMedianTimeSpan];
276         int64_t* pbegin = &pmedian[nMedianTimeSpan];
277         int64_t* pend = &pmedian[nMedianTimeSpan];
278
279         const CBlockIndex* pindex = this;
280         for (int i = 0; i < nMedianTimeSpan && pindex; i++, pindex = pindex->pprev)
281             *(--pbegin) = pindex->GetBlockTime();
282
283         std::sort(pbegin, pend);
284         return pbegin[(pend - pbegin)/2];
285     }
286
287     std::string ToString() const
288     {
289         return strprintf("CBlockIndex(pprev=%p, nHeight=%d, merkle=%s, hashBlock=%s)",
290             pprev, nHeight,
291             hashMerkleRoot.ToString(),
292             GetBlockHash().ToString());
293     }
294
295     //! Check whether this block index entry is valid up to the passed validity level.
296     bool IsValid(enum BlockStatus nUpTo = BLOCK_VALID_TRANSACTIONS) const
297     {
298         assert(!(nUpTo & ~BLOCK_VALID_MASK)); // Only validity flags allowed.
299         if (nStatus & BLOCK_FAILED_MASK)
300             return false;
301         return ((nStatus & BLOCK_VALID_MASK) >= nUpTo);
302     }
303
304     //! Raise the validity level of this block index entry.
305     //! Returns true if the validity was changed.
306     bool RaiseValidity(enum BlockStatus nUpTo)
307     {
308         assert(!(nUpTo & ~BLOCK_VALID_MASK)); // Only validity flags allowed.
309         if (nStatus & BLOCK_FAILED_MASK)
310             return false;
311         if ((nStatus & BLOCK_VALID_MASK) < nUpTo) {
312             nStatus = (nStatus & ~BLOCK_VALID_MASK) | nUpTo;
313             return true;
314         }
315         return false;
316     }
317
318     //! Build the skiplist pointer for this entry.
319     void BuildSkip();
320
321     //! Efficiently find an ancestor of this block.
322     CBlockIndex* GetAncestor(int height);
323     const CBlockIndex* GetAncestor(int height) const;
324 };
325
326 /** Used to marshal pointers into hashes for db storage. */
327 class CDiskBlockIndex : public CBlockIndex
328 {
329 public:
330     uint256 hashPrev;
331
332     CDiskBlockIndex() {
333         hashPrev = uint256();
334     }
335
336     explicit CDiskBlockIndex(const CBlockIndex* pindex) : CBlockIndex(*pindex) {
337         hashPrev = (pprev ? pprev->GetBlockHash() : uint256());
338     }
339
340     ADD_SERIALIZE_METHODS;
341
342     template <typename Stream, typename Operation>
343     inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion) {
344         if (!(nType & SER_GETHASH))
345             READWRITE(VARINT(nVersion));
346
347         READWRITE(VARINT(nHeight));
348         READWRITE(VARINT(nStatus));
349         READWRITE(VARINT(nTx));
350         if (nStatus & (BLOCK_HAVE_DATA | BLOCK_HAVE_UNDO))
351             READWRITE(VARINT(nFile));
352         if (nStatus & BLOCK_HAVE_DATA)
353             READWRITE(VARINT(nDataPos));
354         if (nStatus & BLOCK_HAVE_UNDO)
355             READWRITE(VARINT(nUndoPos));
356         if (nStatus & BLOCK_ACTIVATES_UPGRADE) {
357             if (ser_action.ForRead()) {
358                 uint32_t branchId;
359                 READWRITE(branchId);
360                 nCachedBranchId = branchId;
361             } else {
362                 // nCachedBranchId must always be set if BLOCK_ACTIVATES_UPGRADE is set.
363                 assert(nCachedBranchId);
364                 uint32_t branchId = *nCachedBranchId;
365                 READWRITE(branchId);
366             }
367         }
368         READWRITE(hashAnchor);
369
370         // block header
371         READWRITE(this->nVersion);
372         READWRITE(hashPrev);
373         READWRITE(hashMerkleRoot);
374         READWRITE(hashReserved);
375         READWRITE(nTime);
376         READWRITE(nBits);
377         READWRITE(nNonce);
378         READWRITE(nSolution);
379
380         // Only read/write nSproutValue if the client version used to create
381         // this index was storing them.
382         if ((nType & SER_DISK) && (nVersion >= SPROUT_VALUE_VERSION)) {
383             READWRITE(nSproutValue);
384         }
385     }
386
387     uint256 GetBlockHash() const
388     {
389         CBlockHeader block;
390         block.nVersion        = nVersion;
391         block.hashPrevBlock   = hashPrev;
392         block.hashMerkleRoot  = hashMerkleRoot;
393         block.hashReserved    = hashReserved;
394         block.nTime           = nTime;
395         block.nBits           = nBits;
396         block.nNonce          = nNonce;
397         block.nSolution       = nSolution;
398         return block.GetHash();
399     }
400
401
402     std::string ToString() const
403     {
404         std::string str = "CDiskBlockIndex(";
405         str += CBlockIndex::ToString();
406         str += strprintf("\n                hashBlock=%s, hashPrev=%s)",
407             GetBlockHash().ToString(),
408             hashPrev.ToString());
409         return str;
410     }
411 };
412
413 /** An in-memory indexed chain of blocks. */
414 class CChain {
415 private:
416     std::vector<CBlockIndex*> vChain;
417
418 public:
419     /** Returns the index entry for the genesis block of this chain, or NULL if none. */
420     CBlockIndex *Genesis() const {
421         return vChain.size() > 0 ? vChain[0] : NULL;
422     }
423
424     /** Returns the index entry for the tip of this chain, or NULL if none. */
425     CBlockIndex *Tip() const {
426         return vChain.size() > 0 ? vChain[vChain.size() - 1] : NULL;
427     }
428
429     /** Returns the index entry at a particular height in this chain, or NULL if no such height exists. */
430     CBlockIndex *operator[](int nHeight) const {
431         if (nHeight < 0 || nHeight >= (int)vChain.size())
432             return NULL;
433         return vChain[nHeight];
434     }
435
436     /** Compare two chains efficiently. */
437     friend bool operator==(const CChain &a, const CChain &b) {
438         return a.vChain.size() == b.vChain.size() &&
439                a.vChain[a.vChain.size() - 1] == b.vChain[b.vChain.size() - 1];
440     }
441
442     /** Efficiently check whether a block is present in this chain. */
443     bool Contains(const CBlockIndex *pindex) const {
444         return (*this)[pindex->nHeight] == pindex;
445     }
446
447     /** Find the successor of a block in this chain, or NULL if the given index is not found or is the tip. */
448     CBlockIndex *Next(const CBlockIndex *pindex) const {
449         if (Contains(pindex))
450             return (*this)[pindex->nHeight + 1];
451         else
452             return NULL;
453     }
454
455     /** Return the maximal height in the chain. Is equal to chain.Tip() ? chain.Tip()->nHeight : -1. */
456     int Height() const {
457         return vChain.size() - 1;
458     }
459
460     /** Set/initialize a chain with a given tip. */
461     void SetTip(CBlockIndex *pindex);
462
463     /** Return a CBlockLocator that refers to a block in this chain (by default the tip). */
464     CBlockLocator GetLocator(const CBlockIndex *pindex = NULL) const;
465
466     /** Find the last common block between this chain and a block index entry. */
467     const CBlockIndex *FindFork(const CBlockIndex *pindex) const;
468 };
469
470 #endif // BITCOIN_CHAIN_H
This page took 0.048883 seconds and 4 git commands to generate.