You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
185 lines
4.1 KiB
185 lines
4.1 KiB
#include "permute_tx.h"
|
|
#include <stdbool.h>
|
|
#include <string.h>
|
|
|
|
static bool input_better(const struct wally_tx_input *a,
|
|
const struct wally_tx_input *b)
|
|
{
|
|
int cmp;
|
|
|
|
cmp = memcmp(a->txhash, b->txhash, sizeof(a->txhash));
|
|
if (cmp != 0)
|
|
return cmp < 0;
|
|
if (a->index != b->index)
|
|
return a->index < b->index;
|
|
|
|
/* These shouldn't happen, but let's get a canonical order anyway. */
|
|
if (a->script_len != b->script_len)
|
|
return a->script_len < b->script_len;
|
|
|
|
cmp = memcmp(a->script, b->script, a->script_len);
|
|
if (cmp != 0)
|
|
return cmp < 0;
|
|
return a->sequence < b->sequence;
|
|
}
|
|
|
|
static size_t find_best_in(struct wally_tx_input *inputs, size_t num)
|
|
{
|
|
size_t i, best = 0;
|
|
|
|
for (i = 1; i < num; i++) {
|
|
if (input_better(&inputs[i], &inputs[best]))
|
|
best = i;
|
|
}
|
|
return best;
|
|
}
|
|
|
|
static void swap_wally_inputs(struct wally_tx_input *inputs,
|
|
const void **map,
|
|
size_t i1, size_t i2)
|
|
{
|
|
struct wally_tx_input tmpinput;
|
|
const void *tmp;
|
|
|
|
if (i1 == i2)
|
|
return;
|
|
|
|
tmpinput = inputs[i1];
|
|
inputs[i1] = inputs[i2];
|
|
inputs[i2] = tmpinput;
|
|
|
|
if (map) {
|
|
tmp = map[i1];
|
|
map[i1] = map[i2];
|
|
map[i2] = tmp;
|
|
}
|
|
}
|
|
|
|
static void swap_input_amounts(struct amount_sat **amounts, size_t i1,
|
|
size_t i2)
|
|
{
|
|
struct amount_sat *tmp = amounts[i1];
|
|
amounts[i1] = amounts[i2];
|
|
amounts[i2] = tmp;
|
|
}
|
|
|
|
void permute_inputs(struct bitcoin_tx *tx, const void **map)
|
|
{
|
|
size_t i, best_pos;
|
|
struct wally_tx_input *inputs = tx->wtx->inputs;
|
|
size_t num_inputs = tx->wtx->num_inputs;
|
|
|
|
/* We can't permute nothing! */
|
|
if (num_inputs == 0)
|
|
return;
|
|
|
|
/* Now do a dumb sort (num_inputs is small). */
|
|
for (i = 0; i < num_inputs-1; i++) {
|
|
best_pos = i + find_best_in(inputs + i, num_inputs - i);
|
|
/* Swap best into first place. */
|
|
swap_wally_inputs(tx->wtx->inputs, map, i, best_pos);
|
|
swap_input_amounts(tx->input_amounts, i, best_pos);
|
|
}
|
|
}
|
|
|
|
static void swap_wally_outputs(struct wally_tx_output *outputs,
|
|
const void **map,
|
|
u32 *cltvs,
|
|
size_t i1, size_t i2)
|
|
{
|
|
struct wally_tx_output tmpoutput;
|
|
|
|
if (i1 == i2)
|
|
return;
|
|
|
|
tmpoutput = outputs[i1];
|
|
outputs[i1] = outputs[i2];
|
|
outputs[i2] = tmpoutput;
|
|
|
|
if (map) {
|
|
const void *tmp = map[i1];
|
|
map[i1] = map[i2];
|
|
map[i2] = tmp;
|
|
}
|
|
|
|
if (cltvs) {
|
|
u32 tmp = cltvs[i1];
|
|
cltvs[i1] = cltvs[i2];
|
|
cltvs[i2] = tmp;
|
|
}
|
|
}
|
|
|
|
static bool output_better(const struct wally_tx_output *a, u32 cltv_a,
|
|
const struct wally_tx_output *b, u32 cltv_b)
|
|
{
|
|
size_t len, lena, lenb;
|
|
int ret;
|
|
|
|
if (a->satoshi != b->satoshi)
|
|
return a->satoshi < b->satoshi;
|
|
|
|
/* Lexicographical sort. */
|
|
lena = a->script_len;
|
|
lenb = b->script_len;
|
|
if (lena < lenb)
|
|
len = lena;
|
|
else
|
|
len = lenb;
|
|
|
|
ret = memcmp(a->script, b->script, len);
|
|
if (ret != 0)
|
|
return ret < 0;
|
|
|
|
if (lena != lenb)
|
|
return lena < lenb;
|
|
|
|
return cltv_a < cltv_b;
|
|
}
|
|
|
|
static u32 cltv_of(const u32 *cltvs, size_t idx)
|
|
{
|
|
if (!cltvs)
|
|
return 0;
|
|
return cltvs[idx];
|
|
}
|
|
|
|
static size_t find_best_out(struct wally_tx_output *outputs, const u32 *cltvs,
|
|
size_t num)
|
|
{
|
|
size_t i, best = 0;
|
|
|
|
for (i = 1; i < num; i++) {
|
|
if (output_better(&outputs[i], cltv_of(cltvs, i),
|
|
&outputs[best], cltv_of(cltvs, best)))
|
|
best = i;
|
|
}
|
|
return best;
|
|
}
|
|
|
|
void permute_outputs(struct bitcoin_tx *tx, u32 *cltvs, const void **map)
|
|
{
|
|
size_t i, best_pos;
|
|
struct wally_tx_output *outputs = tx->wtx->outputs;
|
|
size_t num_outputs = tx->wtx->num_outputs;
|
|
|
|
/* We can't permute nothing! */
|
|
if (num_outputs == 0)
|
|
return;
|
|
|
|
/* Now do a dumb sort (num_outputs is small). */
|
|
for (i = 0; i < num_outputs - 1; i++) {
|
|
best_pos =
|
|
i + find_best_out(outputs + i, cltvs ? cltvs + i : NULL,
|
|
num_outputs - i);
|
|
|
|
/* Swap best into first place. */
|
|
swap_wally_outputs(tx->wtx->outputs, map, cltvs, i, best_pos);
|
|
|
|
/* If output_witscripts are present, swap them to match. */
|
|
if (tx->output_witscripts) {
|
|
struct witscript *tmp = tx->output_witscripts[i];
|
|
tx->output_witscripts[i] = tx->output_witscripts[best_pos];
|
|
tx->output_witscripts[best_pos] = tmp;
|
|
}
|
|
}
|
|
}
|
|
|