2022-03-01 19:30:57 +01:00
|
|
|
/*
|
|
|
|
This file is part of TALER
|
|
|
|
Copyright (C) 2022 Taler Systems SA
|
|
|
|
|
|
|
|
TALER is free software; you can redistribute it and/or modify it under the
|
|
|
|
terms of the GNU General Public License as published by the Free Software
|
|
|
|
Foundation; either version 3, or (at your option) any later version.
|
|
|
|
|
|
|
|
TALER is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
|
|
|
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License along with
|
|
|
|
TALER; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* @file util/age_restriction.c
|
|
|
|
* @brief Functions that are used for age restriction
|
|
|
|
* @author Özgür Kesim
|
|
|
|
*/
|
|
|
|
#include "platform.h"
|
|
|
|
#include "taler_util.h"
|
|
|
|
#include "taler_signatures.h"
|
|
|
|
#include <gcrypt.h>
|
|
|
|
|
|
|
|
void
|
|
|
|
TALER_age_commitment_hash (
|
|
|
|
const struct TALER_AgeCommitment *commitment,
|
|
|
|
struct TALER_AgeCommitmentHash *ahash)
|
|
|
|
{
|
|
|
|
struct GNUNET_HashContext *hash_context;
|
|
|
|
struct GNUNET_HashCode hash;
|
|
|
|
|
|
|
|
GNUNET_assert (NULL != ahash);
|
|
|
|
if (NULL == commitment)
|
|
|
|
{
|
|
|
|
memset (ahash, 0, sizeof(struct TALER_AgeCommitmentHash));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-02 10:59:42 +01:00
|
|
|
GNUNET_assert (__builtin_popcount (commitment->mask.bits) - 1 ==
|
2022-03-01 19:30:57 +01:00
|
|
|
commitment->num);
|
|
|
|
|
|
|
|
hash_context = GNUNET_CRYPTO_hash_context_start ();
|
|
|
|
|
|
|
|
for (size_t i = 0; i < commitment->num; i++)
|
|
|
|
{
|
|
|
|
GNUNET_CRYPTO_hash_context_read (hash_context,
|
2022-03-03 19:35:24 +01:00
|
|
|
&commitment->keys[i],
|
2022-03-01 19:30:57 +01:00
|
|
|
sizeof(struct
|
2022-03-03 19:35:24 +01:00
|
|
|
GNUNET_CRYPTO_EcdsaPublicKey));
|
2022-03-01 19:30:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
GNUNET_CRYPTO_hash_context_finish (hash_context,
|
|
|
|
&hash);
|
|
|
|
GNUNET_memcpy (&ahash->shash.bits,
|
|
|
|
&hash.bits,
|
|
|
|
sizeof(ahash->shash.bits));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* To a given age value between 0 and 31, returns the index of the age group
|
|
|
|
* defined by the given mask.
|
|
|
|
*/
|
2022-03-03 19:35:24 +01:00
|
|
|
uint8_t
|
2022-03-01 19:30:57 +01:00
|
|
|
get_age_group (
|
|
|
|
const struct TALER_AgeMask *mask,
|
|
|
|
uint8_t age)
|
|
|
|
{
|
2022-03-02 10:59:42 +01:00
|
|
|
uint32_t m = mask->bits;
|
2022-03-01 19:30:57 +01:00
|
|
|
uint8_t i = 0;
|
|
|
|
|
|
|
|
while (m > 0)
|
|
|
|
{
|
|
|
|
if (0 >= age)
|
|
|
|
break;
|
|
|
|
m = m >> 1;
|
|
|
|
i += m & 1;
|
|
|
|
age--;
|
|
|
|
}
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
enum GNUNET_GenericReturnValue
|
|
|
|
TALER_age_restriction_commit (
|
|
|
|
const struct TALER_AgeMask *mask,
|
|
|
|
const uint8_t age,
|
|
|
|
const uint64_t salt,
|
|
|
|
struct TALER_AgeCommitmentProof *new)
|
|
|
|
{
|
2022-03-02 10:59:42 +01:00
|
|
|
uint8_t num_pub = __builtin_popcount (mask->bits) - 1;
|
|
|
|
uint8_t num_priv = get_age_group (mask, age);
|
2022-03-01 19:30:57 +01:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
GNUNET_assert (NULL != new);
|
2022-03-02 10:59:42 +01:00
|
|
|
GNUNET_assert (mask->bits & 1); /* fist bit must have been set */
|
2022-03-01 19:30:57 +01:00
|
|
|
GNUNET_assert (0 <= num_priv);
|
|
|
|
GNUNET_assert (31 > num_priv);
|
|
|
|
GNUNET_assert (num_priv <= num_pub);
|
|
|
|
|
2022-03-02 10:59:42 +01:00
|
|
|
new->commitment.mask.bits = mask->bits;
|
2022-03-01 19:30:57 +01:00
|
|
|
new->commitment.num = num_pub;
|
|
|
|
new->proof.num = num_priv;
|
2022-03-03 19:35:24 +01:00
|
|
|
new->proof.keys = NULL;
|
2022-03-01 19:30:57 +01:00
|
|
|
|
2022-03-03 19:35:24 +01:00
|
|
|
new->commitment.keys = GNUNET_new_array (
|
2022-03-01 19:30:57 +01:00
|
|
|
num_pub,
|
|
|
|
struct TALER_AgeCommitmentPublicKeyP);
|
2022-03-02 10:59:42 +01:00
|
|
|
|
|
|
|
if (0 < num_priv)
|
2022-03-03 19:35:24 +01:00
|
|
|
new->proof.keys = GNUNET_new_array (
|
2022-03-02 10:59:42 +01:00
|
|
|
num_priv,
|
|
|
|
struct TALER_AgeCommitmentPrivateKeyP);
|
2022-03-01 19:30:57 +01:00
|
|
|
|
|
|
|
/* Create as many private keys as we need and fill the rest of the
|
|
|
|
* public keys with valid curve points.
|
|
|
|
* We need to make sure that the public keys are proper points on the
|
|
|
|
* elliptic curve, so we can't simply fill the struct with random values. */
|
|
|
|
for (i = 0; i < num_pub; i++)
|
|
|
|
{
|
2022-03-03 19:35:24 +01:00
|
|
|
uint64_t salti = salt + i;
|
2022-03-01 19:30:57 +01:00
|
|
|
struct TALER_AgeCommitmentPrivateKeyP key = {0};
|
2022-03-03 19:35:24 +01:00
|
|
|
struct TALER_AgeCommitmentPrivateKeyP *pkey = &key;
|
|
|
|
|
2022-03-01 19:30:57 +01:00
|
|
|
|
|
|
|
/* Only save the private keys for age groups less than num_priv */
|
|
|
|
if (i < num_priv)
|
2022-03-03 19:35:24 +01:00
|
|
|
pkey = &new->proof.keys[i];
|
2022-03-01 19:30:57 +01:00
|
|
|
|
|
|
|
if (GNUNET_OK !=
|
2022-03-03 19:35:24 +01:00
|
|
|
GNUNET_CRYPTO_kdf (pkey,
|
|
|
|
sizeof (*pkey),
|
|
|
|
&salti,
|
|
|
|
sizeof (salti),
|
|
|
|
"age commitment",
|
|
|
|
strlen ("age derivation"),
|
2022-03-01 19:30:57 +01:00
|
|
|
NULL, 0))
|
|
|
|
goto FAIL;
|
|
|
|
|
2022-03-03 19:35:24 +01:00
|
|
|
/* See GNUNET_CRYPTO_ecdsa_key_create */
|
|
|
|
pkey->priv.d[0] &= 248;
|
|
|
|
pkey->priv.d[31] &= 127;
|
|
|
|
pkey->priv.d[31] |= 64;
|
|
|
|
|
|
|
|
GNUNET_CRYPTO_ecdsa_key_get_public (&pkey->priv,
|
|
|
|
&new->commitment.keys[i].pub);
|
2022-03-01 19:30:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return GNUNET_OK;
|
|
|
|
|
|
|
|
FAIL:
|
2022-03-03 19:35:24 +01:00
|
|
|
GNUNET_free (new->commitment.keys);
|
|
|
|
if (NULL != new->proof.keys)
|
|
|
|
GNUNET_free (new->proof.keys);
|
2022-03-01 19:30:57 +01:00
|
|
|
return GNUNET_SYSERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
enum GNUNET_GenericReturnValue
|
|
|
|
TALER_age_commitment_derive (
|
|
|
|
const struct TALER_AgeCommitmentProof *orig,
|
|
|
|
const uint64_t salt,
|
|
|
|
struct TALER_AgeCommitmentProof *new)
|
|
|
|
{
|
2022-03-03 19:35:24 +01:00
|
|
|
char label[sizeof(uint64_t) + 1] = {0};
|
2022-03-01 19:30:57 +01:00
|
|
|
|
|
|
|
GNUNET_assert (NULL != new);
|
2022-03-03 19:35:24 +01:00
|
|
|
GNUNET_assert (orig->proof.num <=
|
|
|
|
orig->commitment.num);
|
|
|
|
GNUNET_assert (orig->commitment.num ==
|
|
|
|
__builtin_popcount (orig->commitment.mask.bits) - 1);
|
2022-03-01 19:30:57 +01:00
|
|
|
|
|
|
|
new->commitment.mask = orig->commitment.mask;
|
|
|
|
new->commitment.num = orig->commitment.num;
|
2022-03-03 19:35:24 +01:00
|
|
|
new->commitment.keys = GNUNET_new_array (
|
2022-03-01 19:30:57 +01:00
|
|
|
new->commitment.num,
|
|
|
|
struct TALER_AgeCommitmentPublicKeyP);
|
|
|
|
|
2022-03-03 19:35:24 +01:00
|
|
|
new->proof.num = orig->proof.num;
|
|
|
|
new->proof.keys = NULL;
|
|
|
|
if (0 != new->proof.num)
|
|
|
|
new->proof.keys = GNUNET_new_array (
|
|
|
|
new->proof.num,
|
|
|
|
struct TALER_AgeCommitmentPrivateKeyP);
|
|
|
|
|
2022-03-08 11:47:51 +01:00
|
|
|
{
|
|
|
|
/* Because GNUNET_CRYPTO_ecdsa_public_key_derive expects char * (and calls
|
|
|
|
* strlen on it), we must avoid 0's in the label. */
|
|
|
|
uint64_t nz_salt = salt | 0x8040201008040201;
|
|
|
|
memcpy (label, &nz_salt, sizeof(nz_salt));
|
|
|
|
}
|
2022-03-03 19:35:24 +01:00
|
|
|
|
|
|
|
/* 1. Derive the public keys */
|
2022-03-01 19:30:57 +01:00
|
|
|
for (size_t i = 0; i < orig->commitment.num; i++)
|
|
|
|
{
|
2022-03-03 19:35:24 +01:00
|
|
|
GNUNET_CRYPTO_ecdsa_public_key_derive (
|
|
|
|
&orig->commitment.keys[i].pub,
|
|
|
|
label,
|
|
|
|
"age commitment derive",
|
|
|
|
&new->commitment.keys[i].pub);
|
2022-03-01 19:30:57 +01:00
|
|
|
}
|
|
|
|
|
2022-03-03 19:35:24 +01:00
|
|
|
/* 2. Derive the private keys */
|
|
|
|
for (size_t i = 0; i < orig->proof.num; i++)
|
2022-03-01 19:30:57 +01:00
|
|
|
{
|
2022-03-03 19:35:24 +01:00
|
|
|
struct GNUNET_CRYPTO_EcdsaPrivateKey *priv;
|
|
|
|
priv = GNUNET_CRYPTO_ecdsa_private_key_derive (
|
|
|
|
&orig->proof.keys[i].priv,
|
|
|
|
label,
|
|
|
|
"age commitment derive");
|
|
|
|
new->proof.keys[i].priv = *priv;
|
|
|
|
GNUNET_free (priv);
|
2022-03-01 19:30:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return GNUNET_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
enum GNUNET_GenericReturnValue
|
|
|
|
TALER_age_commitment_attest (
|
|
|
|
const struct TALER_AgeCommitmentProof *cp,
|
|
|
|
uint8_t age,
|
|
|
|
struct TALER_AgeAttestation *attest)
|
|
|
|
{
|
|
|
|
uint8_t group;
|
|
|
|
|
|
|
|
GNUNET_assert (NULL != attest);
|
|
|
|
GNUNET_assert (NULL != cp);
|
|
|
|
|
|
|
|
group = get_age_group (&cp->commitment.mask,
|
|
|
|
age);
|
|
|
|
|
|
|
|
GNUNET_assert (group < 32);
|
|
|
|
|
|
|
|
if (0 == group)
|
|
|
|
{
|
|
|
|
/* Age group 0 means: no attestation necessary.
|
|
|
|
* We set the signature to zero and communicate success. */
|
|
|
|
memset (attest,
|
|
|
|
0,
|
|
|
|
sizeof(struct TALER_AgeAttestation));
|
|
|
|
return GNUNET_OK;
|
|
|
|
}
|
|
|
|
|
2022-03-03 19:35:24 +01:00
|
|
|
if (group > cp->proof.num)
|
2022-03-01 19:30:57 +01:00
|
|
|
return GNUNET_NO;
|
|
|
|
|
|
|
|
{
|
|
|
|
struct TALER_AgeAttestationPS at = {
|
|
|
|
.purpose.size = htonl (sizeof(at)),
|
|
|
|
.purpose.purpose = htonl (TALER_SIGNATURE_WALLET_AGE_ATTESTATION),
|
|
|
|
.mask = cp->commitment.mask,
|
|
|
|
.age = age
|
|
|
|
};
|
|
|
|
|
2022-03-03 19:35:24 +01:00
|
|
|
GNUNET_CRYPTO_ecdsa_sign (&cp->proof.keys[group - 1].priv,
|
2022-03-01 19:30:57 +01:00
|
|
|
&at,
|
2022-03-03 19:35:24 +01:00
|
|
|
&attest->signature);
|
2022-03-01 19:30:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return GNUNET_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
enum GNUNET_GenericReturnValue
|
|
|
|
TALER_age_commitment_verify (
|
|
|
|
const struct TALER_AgeCommitment *comm,
|
|
|
|
uint8_t age,
|
|
|
|
const struct TALER_AgeAttestation *attest)
|
|
|
|
{
|
|
|
|
uint8_t group;
|
|
|
|
|
|
|
|
GNUNET_assert (NULL != attest);
|
|
|
|
GNUNET_assert (NULL != comm);
|
|
|
|
|
|
|
|
group = get_age_group (&comm->mask,
|
|
|
|
age);
|
|
|
|
|
|
|
|
GNUNET_assert (group < 32);
|
|
|
|
|
|
|
|
/* Age group 0 means: no attestation necessary. */
|
|
|
|
if (0 == group)
|
|
|
|
return GNUNET_OK;
|
|
|
|
|
2022-03-03 19:35:24 +01:00
|
|
|
if (group > comm->num)
|
2022-03-01 19:30:57 +01:00
|
|
|
return GNUNET_NO;
|
|
|
|
|
|
|
|
{
|
|
|
|
struct TALER_AgeAttestationPS at = {
|
|
|
|
.purpose.size = htonl (sizeof(at)),
|
|
|
|
.purpose.purpose = htonl (TALER_SIGNATURE_WALLET_AGE_ATTESTATION),
|
|
|
|
.mask = comm->mask,
|
|
|
|
.age = age,
|
|
|
|
};
|
|
|
|
|
|
|
|
return
|
2022-03-03 19:35:24 +01:00
|
|
|
GNUNET_CRYPTO_ecdsa_verify (TALER_SIGNATURE_WALLET_AGE_ATTESTATION,
|
2022-03-01 19:30:57 +01:00
|
|
|
&at,
|
2022-03-03 19:35:24 +01:00
|
|
|
&attest->signature,
|
|
|
|
&comm->keys[group - 1].pub);
|
2022-03-01 19:30:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
TALER_age_commitment_free (
|
|
|
|
struct TALER_AgeCommitment *commitment)
|
|
|
|
{
|
|
|
|
if (NULL == commitment)
|
|
|
|
return;
|
|
|
|
|
2022-03-03 19:35:24 +01:00
|
|
|
if (NULL != commitment->keys)
|
2022-03-01 19:30:57 +01:00
|
|
|
{
|
2022-03-03 19:35:24 +01:00
|
|
|
GNUNET_free (commitment->keys);
|
|
|
|
commitment->keys = NULL;
|
2022-03-01 19:30:57 +01:00
|
|
|
}
|
|
|
|
GNUNET_free (commitment);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
TALER_age_proof_free (
|
|
|
|
struct TALER_AgeProof *proof)
|
|
|
|
{
|
2022-03-03 19:35:24 +01:00
|
|
|
if (NULL != proof->keys)
|
2022-03-01 19:30:57 +01:00
|
|
|
{
|
|
|
|
GNUNET_CRYPTO_zero_keys (
|
2022-03-03 19:35:24 +01:00
|
|
|
proof->keys,
|
|
|
|
sizeof(*proof->keys) * proof->num);
|
2022-03-01 19:30:57 +01:00
|
|
|
|
2022-03-03 19:35:24 +01:00
|
|
|
GNUNET_free (proof->keys);
|
|
|
|
proof->keys = NULL;
|
2022-03-01 19:30:57 +01:00
|
|
|
}
|
|
|
|
GNUNET_free (proof);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
TALER_age_commitment_proof_free (
|
|
|
|
struct TALER_AgeCommitmentProof *cp)
|
|
|
|
{
|
2022-03-03 19:35:24 +01:00
|
|
|
if (NULL != cp->proof.keys)
|
2022-03-01 19:30:57 +01:00
|
|
|
{
|
|
|
|
GNUNET_CRYPTO_zero_keys (
|
2022-03-03 19:35:24 +01:00
|
|
|
cp->proof.keys,
|
|
|
|
sizeof(*cp->proof.keys) * cp->proof.num);
|
2022-03-01 19:30:57 +01:00
|
|
|
|
2022-03-03 19:35:24 +01:00
|
|
|
GNUNET_free (cp->proof.keys);
|
|
|
|
cp->proof.keys = NULL;
|
2022-03-01 19:30:57 +01:00
|
|
|
}
|
|
|
|
|
2022-03-03 19:35:24 +01:00
|
|
|
if (NULL != cp->commitment.keys)
|
2022-03-01 19:30:57 +01:00
|
|
|
{
|
2022-03-03 19:35:24 +01:00
|
|
|
GNUNET_free (cp->commitment.keys);
|
|
|
|
cp->commitment.keys = NULL;
|
2022-03-01 19:30:57 +01:00
|
|
|
}
|
|
|
|
}
|