#include "bitcoin/privkey.h" #include "bitcoin/shadouble.h" #include "bitcoin/signature.h" #include "lightningd.h" #include "log.h" #include "peer.h" #include "secrets.h" #include #include #include #include #include #include #include #include #include #include #include #include #include struct secret { /* Secret ID of our node; public is dstate->id. */ struct privkey privkey; }; void privkey_sign(struct peer *peer, const void *src, size_t len, struct signature *sig) { struct sha256_double h; sha256_double(&h, memcheck(src, len), len); sign_hash(peer->dstate->secpctx, &peer->dstate->secret->privkey, &h, sig); } struct peer_secrets { /* Two private keys, one for commit txs, one for final output. */ struct privkey commit, final; /* Seed from which we generate revocation hashes. */ struct sha256 revocation_seed; }; void peer_sign_theircommit(const struct peer *peer, struct bitcoin_tx *commit, struct signature *sig) { /* Commit tx only has one input: that of the anchor. */ sign_tx_input(peer->dstate->secpctx, commit, 0, NULL, 0, peer->anchor.witnessscript, &peer->secrets->commit, &peer->us.commitkey, sig); } void peer_sign_ourcommit(const struct peer *peer, struct bitcoin_tx *commit, struct signature *sig) { /* Commit tx only has one input: that of the anchor. */ sign_tx_input(peer->dstate->secpctx, commit, 0, NULL, 0, peer->anchor.witnessscript, &peer->secrets->commit, &peer->us.commitkey, sig); } void peer_sign_spend(const struct peer *peer, struct bitcoin_tx *spend, const u8 *commit_witnessscript, struct signature *sig) { /* Spend tx only has one input: that of the commit tx. */ sign_tx_input(peer->dstate->secpctx, spend, 0, NULL, 0, commit_witnessscript, &peer->secrets->final, &peer->us.finalkey, sig); } void peer_sign_htlc_refund(const struct peer *peer, struct bitcoin_tx *spend, const u8 *htlc_witnessscript, struct signature *sig) { /* Spend tx only has one input: that of the commit tx. */ sign_tx_input(peer->dstate->secpctx, spend, 0, NULL, 0, htlc_witnessscript, &peer->secrets->final, &peer->us.finalkey, sig); } void peer_sign_mutual_close(const struct peer *peer, struct bitcoin_tx *close, struct signature *sig) { sign_tx_input(peer->dstate->secpctx, close, 0, NULL, 0, peer->anchor.witnessscript, &peer->secrets->commit, &peer->us.commitkey, sig); } void peer_sign_steal_input(const struct peer *peer, struct bitcoin_tx *spend, size_t i, const u8 *witnessscript, struct signature *sig) { /* Spend tx only has one input: that of the commit tx. */ sign_tx_input(peer->dstate->secpctx, spend, i, NULL, 0, witnessscript, &peer->secrets->final, &peer->us.finalkey, sig); } static void new_keypair(struct lightningd_state *dstate, struct privkey *privkey, struct pubkey *pubkey) { do { if (RAND_bytes(privkey->secret, sizeof(privkey->secret)) != 1) fatal("Could not get random bytes for privkey"); } while (!pubkey_from_privkey(dstate->secpctx, privkey, pubkey, SECP256K1_EC_COMPRESSED)); } void peer_secrets_init(struct peer *peer) { peer->secrets = tal(peer, struct peer_secrets); new_keypair(peer->dstate, &peer->secrets->commit, &peer->us.commitkey); new_keypair(peer->dstate, &peer->secrets->final, &peer->us.finalkey); if (RAND_bytes(peer->secrets->revocation_seed.u.u8, sizeof(peer->secrets->revocation_seed.u.u8)) != 1) fatal("Could not get random bytes for revocation seed"); } void peer_get_revocation_preimage(const struct peer *peer, u64 index, struct sha256 *preimage) { shachain_from_seed(&peer->secrets->revocation_seed, index, preimage); } void peer_get_revocation_hash(const struct peer *peer, u64 index, struct sha256 *rhash) { struct sha256 preimage; peer_get_revocation_preimage(peer, index, &preimage); sha256(rhash, preimage.u.u8, sizeof(preimage.u.u8)); } void secrets_init(struct lightningd_state *dstate) { int fd; dstate->secret = tal(dstate, struct secret); fd = open("privkey", O_RDONLY); if (fd < 0) { if (errno != ENOENT) fatal("Failed to open privkey: %s", strerror(errno)); log_unusual(dstate->base_log, "Creating privkey file"); new_keypair(dstate, &dstate->secret->privkey, &dstate->id); fd = open("privkey", O_CREAT|O_EXCL|O_WRONLY, 0400); if (fd < 0) fatal("Failed to create privkey file: %s", strerror(errno)); if (!write_all(fd, dstate->secret->privkey.secret, sizeof(dstate->secret->privkey.secret))) { unlink_noerr("privkey"); fatal("Failed to write to privkey file: %s", strerror(errno)); } if (fsync(fd) != 0) fatal("Failed to sync to privkey file: %s", strerror(errno)); close(fd); fd = open("privkey", O_RDONLY); if (fd < 0) fatal("Failed to reopen privkey: %s", strerror(errno)); } if (!read_all(fd, dstate->secret->privkey.secret, sizeof(dstate->secret->privkey.secret))) fatal("Failed to read privkey: %s", strerror(errno)); close(fd); if (!pubkey_from_privkey(dstate->secpctx, &dstate->secret->privkey, &dstate->id, SECP256K1_EC_COMPRESSED)) fatal("Invalid privkey"); log_info(dstate->base_log, "ID: "); log_add_hex(dstate->base_log, dstate->id.der, sizeof(dstate->id.der)); }