#include #include "utilsx2.h" #include "address.h" #include "params.h" #include "thashx2.h" #include "utils.h" /* * Generate the entire Merkle tree, computing the authentication path for leaf_idx, * and the resulting root node using Merkle's TreeHash algorithm. * Expects the layer and tree parts of the tree_addr to be set, as well as the * tree type (i.e. SPX_ADDR_TYPE_HASHTREE or SPX_ADDR_TYPE_FORSTREE) * * This expects tree_addrx2 to be initialized to 2 parallel addr structures for * the Merkle tree nodes * * Applies the offset idx_offset to indices before building addresses, so that * it is possible to continue counting indices across trees. * * This works by using the standard Merkle tree building algorithm, except * that each 'node' tracked is actually 2 consecutive nodes in the real tree. * When we combine two logical nodes AB and WX, we perform the H * operation on adjacent real nodes, forming the parent logical node * (AB)(WX) * * When we get to the top level of the real tree (where there is only * one logical node), we continue this operation one more time; the right * most real node will by the actual root (and the other node will be * garbage). We follow the same thashx2 logic so that the 'extract * authentication path components' part of the loop is still executed (and * to simplify the code somewhat) */ void treehashx2(unsigned char *root, unsigned char *auth_path, const spx_ctx *ctx, uint32_t leaf_idx, uint32_t idx_offset, uint32_t tree_height, void (*gen_leafx2)( unsigned char * /* Where to write the leaves */, const spx_ctx *, uint32_t idx, void *info), uint32_t tree_addrx2[2 * 8], void *info) { /* This is where we keep the intermediate nodes */ unsigned char stackx2[tree_height * 2 * SPX_N]; uint32_t left_adj = 0, prev_left_adj = 0; /* When we're doing the top */ /* level, the left-most part of the tree isn't at the beginning */ /* of current[]. These give the offset of the actual start */ uint32_t idx; uint32_t max_idx = (1 << (tree_height - 1)) - 1; for (idx = 0;; idx++) { unsigned char current[2 * SPX_N]; /* Current logical node */ gen_leafx2( current, ctx, 2 * idx + idx_offset, info ); /* Now combine the freshly generated right node with previously */ /* generated left ones */ uint32_t internal_idx_offset = idx_offset; uint32_t internal_idx = idx; uint32_t internal_leaf = leaf_idx; uint32_t h; /* The height we are in the Merkle tree */ for (h = 0;; h++, internal_idx >>= 1, internal_leaf >>= 1) { /* Special processing if we're at the top of the tree */ if (h >= tree_height - 1) { if (h == tree_height) { /* We hit the root; return it */ memcpy( root, ¤t[1 * SPX_N], SPX_N ); return; } /* The tree indexing logic is a bit off in this case */ /* Adjust it so that the left-most node of the part of */ /* the tree that we're processing has index 0 */ prev_left_adj = left_adj; left_adj = 2 - (1 << (tree_height - h - 1)); } /* Check if we hit the top of the tree */ if (h == tree_height) { /* We hit the root; return it */ memcpy( root, ¤t[1 * SPX_N], SPX_N ); return; } /* * Check if one of the nodes we have is a part of the * authentication path; if it is, write it out */ if ((((internal_idx << 1) ^ internal_leaf) & ~0x1) == 0) { memcpy( &auth_path[ h * SPX_N ], ¤t[(((internal_leaf & 1) ^ 1) + prev_left_adj) * SPX_N], SPX_N ); } /* * Check if we're at a left child; if so, stop going up the stack * Exception: if we've reached the end of the tree, keep on going * (so we combine the last 2 nodes into the one root node in two * more iterations) */ if ((internal_idx & 1) == 0 && idx < max_idx) { break; } /* Ok, we're at a right node (or doing the top 3 levels) */ /* Now combine the left and right logical nodes together */ /* Set the address of the node we're creating. */ int j; internal_idx_offset >>= 1; for (j = 0; j < 2; j++) { set_tree_height(tree_addrx2 + j * 8, h + 1); set_tree_index(tree_addrx2 + j * 8, (2 / 2) * (internal_idx & ~1U) + j - left_adj + internal_idx_offset ); } unsigned char *left = &stackx2[h * 2 * SPX_N]; thashx2( ¤t[0 * SPX_N], ¤t[1 * SPX_N], &left [0 * SPX_N], ¤t[0 * SPX_N], 2, ctx, tree_addrx2); } /* We've hit a left child; save the current for when we get the */ /* corresponding right right */ memcpy( &stackx2[h * 2 * SPX_N], current, 2 * SPX_N); } }