Browse Source
This is stolen from William's clightning-dumpkeys (https://github.com/jb55/clightning-dumpkeys),
itself adapted from 42b66a6b81/src/script/descriptor.cpp (L25)
Co-authored-by: William Casarin <jb55@jb55.com>
Signed-off-by: Antoine Poinsot <darosior@protonmail.com>
ppa-prep
Antoine Poinsot
4 years ago
committed by
neil saitug
3 changed files with 99 additions and 0 deletions
@ -0,0 +1,82 @@ |
|||
#include <ccan/short_types/short_types.h> |
|||
#include <common/descriptor_checksum.h> |
|||
#include <stdio.h> |
|||
#include <stdlib.h> |
|||
|
|||
|
|||
static const char CHECKSUM_CHARSET[] = "qpzry9x8gf2tvdw0s3jn54khce6mua7l"; |
|||
|
|||
static const char INPUT_CHARSET[] = |
|||
"0123456789()[],'/*abcdefgh@:$%{}" |
|||
"IJKLMNOPQRSTUVWXYZ&+-.;<=>?!^_|~" |
|||
"ijklmnopqrstuvwxyzABCDEFGH`#\"\\ "; |
|||
|
|||
static inline int charset_find(char ch) { |
|||
for (size_t i = 0; i < sizeof(INPUT_CHARSET); i++) { |
|||
if (INPUT_CHARSET[i] == ch) |
|||
return i; |
|||
} |
|||
return -1; |
|||
} |
|||
|
|||
static u64 polymod(u64 c, int val) |
|||
{ |
|||
u8 c0 = c >> 35; |
|||
c = ((c & 0x7ffffffff) << 5) ^ val; |
|||
if (c0 & 1) c ^= 0xf5dee51989; |
|||
if (c0 & 2) c ^= 0xa9fdca3312; |
|||
if (c0 & 4) c ^= 0x1bab10e32d; |
|||
if (c0 & 8) c ^= 0x3706b1677a; |
|||
if (c0 & 16) c ^= 0x644d626ffd; |
|||
return c; |
|||
} |
|||
|
|||
|
|||
bool descriptor_checksum(const char *descriptor, int desc_size, |
|||
struct descriptor_checksum *checksum) |
|||
{ |
|||
checksum->csum[0] = 0; |
|||
|
|||
int j; |
|||
u64 c = 1; |
|||
int cls = 0; |
|||
int clscount = 0; |
|||
|
|||
for (int i = 0; i < desc_size; i++) { |
|||
char ch = descriptor[i]; |
|||
int pos = charset_find(ch); |
|||
if (pos == -1) { |
|||
checksum->csum[0] = 0; |
|||
return false; |
|||
} |
|||
/* Emit a symbol for the position inside the group, for every
|
|||
* character. */ |
|||
c = polymod(c, pos & 31); |
|||
|
|||
/* Accumulate the group numbers */ |
|||
cls = cls * 3 + (pos >> 5); |
|||
|
|||
if (++clscount == 3) { |
|||
c = polymod(c, cls); |
|||
cls = 0; |
|||
clscount = 0; |
|||
} |
|||
} |
|||
|
|||
if (clscount > 0) |
|||
c = polymod(c, cls); |
|||
|
|||
/* Shift further to determine the checksum. */ |
|||
for (j = 0; j < DESCRIPTOR_CHECKSUM_LENGTH; ++j) |
|||
c = polymod(c, 0); |
|||
|
|||
/* Prevent appending zeroes from not affecting the checksum. */ |
|||
c ^= 1; |
|||
|
|||
for (j = 0; j < DESCRIPTOR_CHECKSUM_LENGTH; ++j) |
|||
checksum->csum[j] = CHECKSUM_CHARSET[(c >> (5 * (7 - j))) & 31]; |
|||
|
|||
checksum->csum[DESCRIPTOR_CHECKSUM_LENGTH] = 0; |
|||
|
|||
return true; |
|||
} |
@ -0,0 +1,16 @@ |
|||
#ifndef LIGHTNING_COMMON_DESCRIPTOR_CHECKSUM_H |
|||
#define LIGHTNING_COMMON_DESCRIPTOR_CHECKSUM_H |
|||
#include "config.h" |
|||
#include <stdbool.h> |
|||
|
|||
/* https://github.com/bitcoin/bitcoin/blob/master/doc/descriptors.md#reference */ |
|||
#define DESCRIPTOR_CHECKSUM_LENGTH 8 |
|||
|
|||
struct descriptor_checksum { |
|||
char csum[DESCRIPTOR_CHECKSUM_LENGTH + 1]; |
|||
}; |
|||
|
|||
bool descriptor_checksum(const char *descriptor, int desc_size, |
|||
struct descriptor_checksum *checksum); |
|||
|
|||
#endif /* LIGHTNING_COMMON_DESCRIPTOR_CHECKSUM_H */ |
Loading…
Reference in new issue