#include "bitcoin/block.h" #include "bitcoin/pullpush.h" #include "bitcoin/tx.h" #include #include /* Encoding is ... */ struct bitcoin_block * bitcoin_block_from_hex(const tal_t *ctx, const struct chainparams *chainparams, const char *hex, size_t hexlen) { struct bitcoin_block *b; u8 *linear_tx; const u8 *p; size_t len, i, num; struct sha256_ctx shactx; if (hexlen && hex[hexlen-1] == '\n') hexlen--; /* Set up the block for success. */ b = tal(ctx, struct bitcoin_block); /* De-hex the array. */ len = hex_data_size(hexlen); p = linear_tx = tal_arr(ctx, u8, len); if (!hex_decode(hex, hexlen, linear_tx, len)) return tal_free(b); sha256_init(&shactx); b->hdr.version = pull_le32(&p, &len); sha256_le32(&shactx, b->hdr.version); pull(&p, &len, &b->hdr.prev_hash, sizeof(b->hdr.prev_hash)); sha256_update(&shactx, &b->hdr.prev_hash, sizeof(b->hdr.prev_hash)); pull(&p, &len, &b->hdr.merkle_hash, sizeof(b->hdr.merkle_hash)); sha256_update(&shactx, &b->hdr.merkle_hash, sizeof(b->hdr.merkle_hash)); b->hdr.timestamp = pull_le32(&p, &len); sha256_le32(&shactx, b->hdr.timestamp); if (is_elements(chainparams)) { b->elements_hdr = tal(b, struct elements_block_hdr); b->elements_hdr->block_height = pull_le32(&p, &len); size_t challenge_len = pull_varint(&p, &len); b->elements_hdr->proof.challenge = tal_arr(b->elements_hdr, u8, challenge_len); pull(&p, &len, b->elements_hdr->proof.challenge, challenge_len); size_t solution_len = pull_varint(&p, &len); b->elements_hdr->proof.solution = tal_arr(b->elements_hdr, u8, solution_len); pull(&p, &len, b->elements_hdr->proof.solution, solution_len); } else { b->hdr.target = pull_le32(&p, &len); sha256_le32(&shactx, b->hdr.target); b->hdr.nonce = pull_le32(&p, &len); sha256_le32(&shactx, b->hdr.nonce); } sha256_double_done(&shactx, &b->hdr.hash); num = pull_varint(&p, &len); b->tx = tal_arr(b, struct bitcoin_tx *, num); for (i = 0; i < num; i++) { b->tx[i] = pull_bitcoin_tx(b->tx, &p, &len); b->tx[i]->chainparams = chainparams; } /* We should end up not overrunning, nor have extra */ if (!p || len) return tal_free(b); tal_free(linear_tx); return b; } void bitcoin_block_blkid(const struct bitcoin_block *b, struct bitcoin_blkid *out) { struct sha256_ctx shactx; u8 vt[VARINT_MAX_LEN]; size_t vtlen; sha256_init(&shactx); sha256_le32(&shactx, b->hdr.version); sha256_update(&shactx, &b->hdr.prev_hash, sizeof(b->hdr.prev_hash)); sha256_update(&shactx, &b->hdr.merkle_hash, sizeof(b->hdr.merkle_hash)); sha256_le32(&shactx, b->hdr.timestamp); if (is_elements(chainparams)) { size_t clen = tal_bytelen(b->elements_hdr->proof.challenge); sha256_le32(&shactx, b->elements_hdr->block_height); vtlen = varint_put(vt, clen); sha256_update(&shactx, vt, vtlen); sha256_update(&shactx, b->elements_hdr->proof.challenge, clen); /* The solution is skipped, since that'd create a circular * dependency apparently */ } else { sha256_le32(&shactx, b->hdr.target); sha256_le32(&shactx, b->hdr.nonce); } sha256_double_done(&shactx, &out->shad); } /* We do the same hex-reversing crud as txids. */ bool bitcoin_blkid_from_hex(const char *hexstr, size_t hexstr_len, struct bitcoin_blkid *blockid) { struct bitcoin_txid fake_txid; if (!bitcoin_txid_from_hex(hexstr, hexstr_len, &fake_txid)) return false; blockid->shad = fake_txid.shad; return true; } bool bitcoin_blkid_to_hex(const struct bitcoin_blkid *blockid, char *hexstr, size_t hexstr_len) { struct bitcoin_txid fake_txid; fake_txid.shad = blockid->shad; return bitcoin_txid_to_hex(&fake_txid, hexstr, hexstr_len); } static char *fmt_bitcoin_blkid(const tal_t *ctx, const struct bitcoin_blkid *blkid) { char *hexstr = tal_arr(ctx, char, hex_str_size(sizeof(*blkid))); bitcoin_blkid_to_hex(blkid, hexstr, hex_str_size(sizeof(*blkid))); return hexstr; } REGISTER_TYPE_TO_STRING(bitcoin_blkid, fmt_bitcoin_blkid);