@@ -325,4 +325,63 @@ int secp256k1_silentpayments_sender_create_output_pubkey(const secp256k1_context
325
325
return 1 ;
326
326
}
327
327
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
+
328
387
#endif
0 commit comments