Skip to content

Commit

Permalink
[WIP] Allow up to 5 data pushes in OP_RETURN outputs
Browse files Browse the repository at this point in the history
  • Loading branch information
bigspider committed May 6, 2024
1 parent e6b45ad commit 92c3337
Show file tree
Hide file tree
Showing 2 changed files with 28 additions and 32 deletions.
52 changes: 22 additions & 30 deletions src/common/script.c
Original file line number Diff line number Diff line change
Expand Up @@ -130,66 +130,58 @@ int format_opscript_script(const uint8_t script[],
// However, signing such outputs is part of BIP-0322, and there's no danger in allowing them.

if (script_len == 1) {
--out_ctr; // remove extra space
} else {
// We parse the rest as a single push opcode.
// This supports a subset of the scripts that bitcoin-core considers standard.
out[out_ctr - 1] = '\0'; // remove extra space
return out_ctr;
}

size_t offset = 1; // start after OP_RETURN
int num_pushes = 0;
const char hex[] = "0123456789abcdef";

while (offset < script_len && num_pushes < 5) {
uint8_t opcode = script[offset++];
size_t hex_length = 0; // Data length to process

uint8_t opcode = script[1]; // the push opcode
if (opcode > OP_16 || opcode == OP_RESERVED || opcode == OP_PUSHDATA2 ||
opcode == OP_PUSHDATA4) {
return -1; // unsupported
}

int hex_offset = 1;
size_t hex_length = 0; // if non-zero, `hex_length` bytes starting from script[hex_offset]
// must be hex-encoded

if (opcode == OP_0) {
if (script_len != 1 + 1) return -1;
out[out_ctr++] = '0';
} else if (opcode >= 1 && opcode <= 75) {
hex_offset += 1;
hex_length = opcode;

if (script_len != 1 + 1 + hex_length) return -1;
if (offset + hex_length > script_len) return -1; // out of bounds
} else if (opcode == OP_PUSHDATA1) {
// OP_RETURN OP_PUSHDATA1 <len:1-byte> <data:len bytes>
hex_offset += 2;
hex_length = script[2];

if (script_len != 1 + 1 + 1 + hex_length || hex_length > 80) return -1;
if (offset >= script_len) return -1; // out of bounds for length byte
hex_length = script[offset++];
if (hex_length > 80 || offset + hex_length > script_len) return -1;
} else if (opcode == OP_1NEGATE) {
if (script_len != 1 + 1) return -1;

out[out_ctr++] = '-';
out[out_ctr++] = '1';
} else if (opcode >= OP_1 && opcode <= OP_16) {
if (script_len != 1 + 1) return -1;

// encode OP_1 to OP_16 as a decimal number
uint8_t num = opcode - 0x50;
if (num >= 10) {
out[out_ctr++] = '0' + (num / 10);
}
out[out_ctr++] = '0' + (num % 10);
} else {
return -1; // can never happen
}

if (hex_length > 0) {
const char hex[] = "0123456789abcdef";

out[out_ctr++] = '0';
out[out_ctr++] = 'x';
for (unsigned int i = 0; i < hex_length; i++) {
uint8_t data = script[hex_offset + i];
uint8_t data = script[offset + i];
out[out_ctr++] = hex[data / 16];
out[out_ctr++] = hex[data % 16];
}
offset += hex_length;
}

num_pushes++;
out[out_ctr++] = ' ';
}

out[out_ctr++] = '\0';
out[out_ctr - 1] = '\0';
return out_ctr;
}
}
8 changes: 6 additions & 2 deletions src/common/script.h
Original file line number Diff line number Diff line change
Expand Up @@ -199,8 +199,12 @@ int get_script_address(const uint8_t script[], size_t script_len, char *out, siz

#endif

// the longest OP_RETURN description "OP_RETURN 0x" followed by 160 hexadecimal characters
#define MAX_OPRETURN_OUTPUT_DESC_SIZE (12 + 80 * 2 + 1)
// the longest OP_RETURN description is upper bounded by:
// - 9 bytes for "OP_RETURN"
// - 5 times 3 for the " 0x"
// - up to 2 * 80 = 160 hexadecimal bytes
// - the termination null character
#define MAX_OPRETURN_OUTPUT_DESC_SIZE (9 + 5 * 3 + 2 * 80 + 1)

/**
* Formats a valid OP_RETURN script for user verification. The resulting string is "OP_RETURN
Expand Down

0 comments on commit 92c3337

Please sign in to comment.