Skip to content

Commit d6c9856

Browse files
theStackjosibake
andcommitted
silentpayments: add routine for tx output scanning (for receiver)
Co-authored-by: josibake <[email protected]>
1 parent 8460be5 commit d6c9856

File tree

2 files changed

+108
-0
lines changed

2 files changed

+108
-0
lines changed

include/secp256k1_silentpayments.h

+49
Original file line numberDiff line numberDiff line change
@@ -240,6 +240,55 @@ SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_silentpayments_sender_c
240240
unsigned int k
241241
) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3) SECP256K1_ARG_NONNULL(4);
242242

243+
typedef struct {
244+
secp256k1_pubkey label;
245+
secp256k1_pubkey label_negated;
246+
} secp256k1_silentpayments_label_data;
247+
248+
/** Scan for Silent Payment transaction output (for receiver).
249+
*
250+
* Given a shared_secret, a recipient's spend public key B_spend,
251+
* an output counter k, and a scanned tx's output x-only public key tx_output,
252+
* calculate the corresponding scanning data:
253+
*
254+
* t_k = hash(shared_secret || ser_32(k))
255+
* P_output = B_spend + t_k * G [not returned]
256+
* if P_output == tx_output
257+
* direct_match = 1
258+
* else
259+
* label1 = tx_output - P_output
260+
* label2 = -tx_output - P_output
261+
* direct_match = 0
262+
*
263+
* The resulting data is needed for the receiver to efficiently scan for labels
264+
* in silent payments eligible outputs.
265+
*
266+
* Returns: 1 if output scanning was successful. 0 if an error occured.
267+
* Args: ctx: pointer to a context object
268+
* Out: direct_match: pointer to the resulting boolean indicating whether
269+
* the calculated output pubkey matches the scanned one
270+
* t_k: pointer to the resulting tweak t_k
271+
* label_data: pointer to the resulting label structure, containing the
272+
* two label candidates, only set if direct_match == 0
273+
* (can be NULL if the data is not needed)
274+
* In: shared_secret33: shared secret, derived from either sender's
275+
* or receiver's perspective with routines from above
276+
* receiver_spend_pubkey: pointer to the receiver's spend pubkey
277+
* k: output counter (usually set to 0, should be increased for
278+
* every additional output to the same recipient)
279+
* tx_output: pointer to the scanned tx's output x-only public key
280+
*/
281+
SECP256K1_API SECP256K1_WARN_UNUSED_RESULT int secp256k1_silentpayments_receiver_scan_output(
282+
const secp256k1_context *ctx,
283+
int *direct_match,
284+
unsigned char *t_k,
285+
secp256k1_silentpayments_label_data *label_data,
286+
const unsigned char *shared_secret33,
287+
const secp256k1_pubkey *receiver_spend_pubkey,
288+
unsigned int k,
289+
const secp256k1_xonly_pubkey *tx_output
290+
) SECP256K1_ARG_NONNULL(1) SECP256K1_ARG_NONNULL(2) SECP256K1_ARG_NONNULL(3) SECP256K1_ARG_NONNULL(5) SECP256K1_ARG_NONNULL(6) SECP256K1_ARG_NONNULL(8);
291+
243292
#ifdef __cplusplus
244293
}
245294
#endif

src/modules/silentpayments/main_impl.h

+59
Original file line numberDiff line numberDiff line change
@@ -325,4 +325,63 @@ int secp256k1_silentpayments_sender_create_output_pubkey(const secp256k1_context
325325
return 1;
326326
}
327327

328+
int secp256k1_silentpayments_receiver_scan_output(const secp256k1_context *ctx, int *direct_match, unsigned char *t_k, secp256k1_silentpayments_label_data *label_data, const unsigned char *shared_secret33, const secp256k1_pubkey *receiver_spend_pubkey, unsigned int k, const secp256k1_xonly_pubkey *tx_output) {
329+
secp256k1_scalar t_k_scalar;
330+
secp256k1_ge P_output_ge;
331+
secp256k1_xonly_pubkey P_output_xonly;
332+
333+
/* Sanity check inputs */
334+
VERIFY_CHECK(ctx != NULL);
335+
ARG_CHECK(direct_match != NULL);
336+
ARG_CHECK(t_k != NULL);
337+
ARG_CHECK(shared_secret33 != NULL);
338+
ARG_CHECK(receiver_spend_pubkey != NULL);
339+
ARG_CHECK(tx_output != NULL);
340+
341+
/* Calculate t_k = hash(shared_secret || ser_32(k)) */
342+
secp256k1_silentpayments_create_t_k(&t_k_scalar, shared_secret33, k);
343+
secp256k1_scalar_get_b32(t_k, &t_k_scalar);
344+
345+
/* Calculate P_output = B_spend + t_k * G */
346+
secp256k1_pubkey_load(ctx, &P_output_ge, receiver_spend_pubkey);
347+
if (!secp256k1_eckey_pubkey_tweak_add(&P_output_ge, &t_k_scalar)) {
348+
return 0;
349+
}
350+
351+
/* If the calculated output matches the one from the tx, we have a direct match and can
352+
* return without labels calculation (one of the two would result in point of infinity) */
353+
secp256k1_xonly_pubkey_save(&P_output_xonly, &P_output_ge);
354+
if (secp256k1_xonly_pubkey_cmp(ctx, &P_output_xonly, tx_output) == 0) {
355+
*direct_match = 1;
356+
return 1;
357+
}
358+
*direct_match = 0;
359+
360+
/* If desired, also calculate label candidates */
361+
if (label_data != NULL) {
362+
secp256k1_ge P_output_negated_ge, tx_output_ge;
363+
secp256k1_ge label_ge;
364+
secp256k1_gej label_gej;
365+
366+
/* Calculate negated P_output (common addend) first */
367+
secp256k1_ge_neg(&P_output_negated_ge, &P_output_ge);
368+
369+
/* Calculate first scan label candidate: label1 = tx_output - P_output */
370+
secp256k1_xonly_pubkey_load(ctx, &tx_output_ge, tx_output);
371+
secp256k1_gej_set_ge(&label_gej, &tx_output_ge);
372+
secp256k1_gej_add_ge_var(&label_gej, &label_gej, &P_output_negated_ge, NULL);
373+
secp256k1_ge_set_gej(&label_ge, &label_gej);
374+
secp256k1_pubkey_save(&label_data->label, &label_ge);
375+
376+
/* Calculate second scan label candidate: label2 = -tx_output - P_output */
377+
secp256k1_gej_set_ge(&label_gej, &tx_output_ge);
378+
secp256k1_gej_neg(&label_gej, &label_gej);
379+
secp256k1_gej_add_ge_var(&label_gej, &label_gej, &P_output_negated_ge, NULL);
380+
secp256k1_ge_set_gej(&label_ge, &label_gej);
381+
secp256k1_pubkey_save(&label_data->label_negated, &label_ge);
382+
}
383+
384+
return 1;
385+
}
386+
328387
#endif

0 commit comments

Comments
 (0)