mirror of
https://gitlab.torproject.org/tpo/core/tor.git
synced 2024-11-14 07:03:44 +01:00
More complete docs for crypto.c; factor out string partitioning code
svn:r2427
This commit is contained in:
parent
dad7c71686
commit
7b98fb58eb
@ -142,7 +142,7 @@ crypto_log_errors(int severity, const char *doing)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Initialize the crypto library.
|
/** Initialize the crypto library. Return 0 on success, -1 on failure.
|
||||||
*/
|
*/
|
||||||
int crypto_global_init()
|
int crypto_global_init()
|
||||||
{
|
{
|
||||||
@ -153,7 +153,7 @@ int crypto_global_init()
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Uninitialize the crypto library.
|
/** Uninitialize the crypto library. Return 0 on success, -1 on failure.
|
||||||
*/
|
*/
|
||||||
int crypto_global_cleanup()
|
int crypto_global_cleanup()
|
||||||
{
|
{
|
||||||
@ -318,6 +318,7 @@ int crypto_pk_generate_key(crypto_pk_env_t *env)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/** Read a PEM-encoded private key from the string <b>s</b> into <b>env</b>.
|
/** Read a PEM-encoded private key from the string <b>s</b> into <b>env</b>.
|
||||||
|
* Return 0 on success, -1 on failure.
|
||||||
*/
|
*/
|
||||||
static int crypto_pk_read_private_key_from_string(crypto_pk_env_t *env,
|
static int crypto_pk_read_private_key_from_string(crypto_pk_env_t *env,
|
||||||
const char *s)
|
const char *s)
|
||||||
@ -502,10 +503,9 @@ int crypto_pk_DER64_encode_public_key(crypto_pk_env_t *env, char **out)
|
|||||||
*/
|
*/
|
||||||
crypto_pk_env_t *crypto_pk_DER64_decode_public_key(const char *in)
|
crypto_pk_env_t *crypto_pk_DER64_decode_public_key(const char *in)
|
||||||
{
|
{
|
||||||
char buf1[PK_BYTES*2 + PK_BYTES/64 + 2];
|
char partitioned[PK_BYTES*2 + 16];
|
||||||
char buf[PK_BYTES*2];
|
char buf[PK_BYTES*2];
|
||||||
int len;
|
int len;
|
||||||
int i;
|
|
||||||
tor_assert(in);
|
tor_assert(in);
|
||||||
len = strlen(in);
|
len = strlen(in);
|
||||||
|
|
||||||
@ -514,11 +514,11 @@ crypto_pk_env_t *crypto_pk_DER64_decode_public_key(const char *in)
|
|||||||
}
|
}
|
||||||
/* base64_decode doesn't work unless we insert linebreaks every 64
|
/* base64_decode doesn't work unless we insert linebreaks every 64
|
||||||
* characters. how dumb. */
|
* characters. how dumb. */
|
||||||
for(i=0;i*64<len;i+=1) {
|
if (tor_strpartition(partitioned, sizeof(partitioned), in, "\n", 64))
|
||||||
strncpy(buf1+(64+1)*i, in+64*i, 64);
|
return NULL;
|
||||||
strcpy(buf1+(64+1)*i + 64, "\n");
|
if (strlcat(partitioned, "\n",sizeof(partitioned))>=sizeof(partitioned))
|
||||||
}
|
return NULL;
|
||||||
len = base64_decode(buf, sizeof(buf), buf1, strlen(buf1));
|
len = base64_decode(buf, sizeof(buf), partitioned, strlen(partitioned));
|
||||||
if (len<0) {
|
if (len<0) {
|
||||||
log_fn(LOG_WARN,"Error base-64 decoding key");
|
log_fn(LOG_WARN,"Error base-64 decoding key");
|
||||||
return NULL;
|
return NULL;
|
||||||
@ -567,7 +567,7 @@ int crypto_pk_keysize(crypto_pk_env_t *env)
|
|||||||
return RSA_size(env->key);
|
return RSA_size(env->key);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Increase the reference count of <b>env</b>.
|
/** Increase the reference count of <b>env</b>, and return it.
|
||||||
*/
|
*/
|
||||||
crypto_pk_env_t *crypto_pk_dup_key(crypto_pk_env_t *env) {
|
crypto_pk_env_t *crypto_pk_dup_key(crypto_pk_env_t *env) {
|
||||||
tor_assert(env && env->key);
|
tor_assert(env && env->key);
|
||||||
@ -874,6 +874,7 @@ crypto_pk_env_t *crypto_pk_asn1_decode(const char *str, int len)
|
|||||||
|
|
||||||
/** Given a private or public key <b>pk</b>, put a SHA1 hash of the
|
/** Given a private or public key <b>pk</b>, put a SHA1 hash of the
|
||||||
* public key into <b>digest_out</b> (must have DIGEST_LEN bytes of space).
|
* public key into <b>digest_out</b> (must have DIGEST_LEN bytes of space).
|
||||||
|
* Return 0 on success, -1 on failure.
|
||||||
*/
|
*/
|
||||||
int crypto_pk_get_digest(crypto_pk_env_t *pk, char *digest_out)
|
int crypto_pk_get_digest(crypto_pk_env_t *pk, char *digest_out)
|
||||||
{
|
{
|
||||||
@ -900,7 +901,7 @@ int crypto_pk_get_digest(crypto_pk_env_t *pk, char *digest_out)
|
|||||||
|
|
||||||
/** Given a private or public key <b>pk</b>, put a fingerprint of the
|
/** Given a private or public key <b>pk</b>, put a fingerprint of the
|
||||||
* public key into <b>fp_out</b> (must have at least FINGERPRINT_LEN+1 bytes of
|
* public key into <b>fp_out</b> (must have at least FINGERPRINT_LEN+1 bytes of
|
||||||
* space).
|
* space). Return 0 on success, -1 on failure.
|
||||||
*
|
*
|
||||||
* Fingerprints are computed as the SHA1 digest of the ASN.1 encoding
|
* Fingerprints are computed as the SHA1 digest of the ASN.1 encoding
|
||||||
* of the public key, converted to hexadecimal, in upper case, with a
|
* of the public key, converted to hexadecimal, in upper case, with a
|
||||||
@ -911,29 +912,17 @@ int crypto_pk_get_digest(crypto_pk_env_t *pk, char *digest_out)
|
|||||||
int
|
int
|
||||||
crypto_pk_get_fingerprint(crypto_pk_env_t *pk, char *fp_out, int add_space)
|
crypto_pk_get_fingerprint(crypto_pk_env_t *pk, char *fp_out, int add_space)
|
||||||
{
|
{
|
||||||
unsigned char *bufp;
|
|
||||||
unsigned char digest[DIGEST_LEN];
|
unsigned char digest[DIGEST_LEN];
|
||||||
unsigned char buf[FINGERPRINT_LEN+1];
|
unsigned char hexdigest[HEX_DIGEST_LEN+1];
|
||||||
int i;
|
|
||||||
if (crypto_pk_get_digest(pk, digest)) {
|
if (crypto_pk_get_digest(pk, digest)) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
bufp = buf;
|
base16_encode(hexdigest,sizeof(hexdigest),digest,DIGEST_LEN);
|
||||||
for (i = 0; i < DIGEST_LEN; ++i) {
|
|
||||||
sprintf(bufp,"%02X",digest[i]);
|
|
||||||
bufp += 2;
|
|
||||||
if (add_space) {
|
|
||||||
if (i%2 && i != 19) {
|
|
||||||
*bufp++ = ' ';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
*bufp = '\0';
|
|
||||||
if (add_space) {
|
if (add_space) {
|
||||||
tor_assert(strlen(buf) == FINGERPRINT_LEN);
|
tor_strpartition(fp_out, FINGERPRINT_LEN+1, hexdigest, " ", 4);
|
||||||
tor_assert(crypto_pk_check_fingerprint_syntax(buf));
|
} else {
|
||||||
|
strcpy(fp_out, hexdigest);
|
||||||
}
|
}
|
||||||
strcpy(fp_out, buf);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -968,6 +957,7 @@ int crypto_cipher_generate_key(crypto_cipher_env_t *env)
|
|||||||
|
|
||||||
/** Set the symmetric key for the cipher in <b>env</b> to the first
|
/** Set the symmetric key for the cipher in <b>env</b> to the first
|
||||||
* CIPHER_KEY_LEN bytes of <b>key</b>. Does not initialize the cipher.
|
* CIPHER_KEY_LEN bytes of <b>key</b>. Does not initialize the cipher.
|
||||||
|
* Return 0 on success, -1 on failure.
|
||||||
*/
|
*/
|
||||||
int crypto_cipher_set_key(crypto_cipher_env_t *env, const unsigned char *key)
|
int crypto_cipher_set_key(crypto_cipher_env_t *env, const unsigned char *key)
|
||||||
{
|
{
|
||||||
@ -988,7 +978,8 @@ const unsigned char *crypto_cipher_get_key(crypto_cipher_env_t *env)
|
|||||||
return env->key;
|
return env->key;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Initialize the cipher in <b>env</b> for encryption.
|
/** Initialize the cipher in <b>env</b> for encryption. Return 0 on
|
||||||
|
* success, -1 on failure.
|
||||||
*/
|
*/
|
||||||
int crypto_cipher_encrypt_init_cipher(crypto_cipher_env_t *env)
|
int crypto_cipher_encrypt_init_cipher(crypto_cipher_env_t *env)
|
||||||
{
|
{
|
||||||
@ -998,7 +989,8 @@ int crypto_cipher_encrypt_init_cipher(crypto_cipher_env_t *env)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Initialize the cipher in <b>env</b> for decryption.
|
/** Initialize the cipher in <b>env</b> for decryption. Return 0 on
|
||||||
|
* success, -1 on failure.
|
||||||
*/
|
*/
|
||||||
int crypto_cipher_decrypt_init_cipher(crypto_cipher_env_t *env)
|
int crypto_cipher_decrypt_init_cipher(crypto_cipher_env_t *env)
|
||||||
{
|
{
|
||||||
@ -1033,6 +1025,7 @@ int crypto_cipher_decrypt(crypto_cipher_env_t *env, const unsigned char *from, u
|
|||||||
}
|
}
|
||||||
|
|
||||||
/** Move the position of the cipher stream backwards by <b>delta</b> bytes.
|
/** Move the position of the cipher stream backwards by <b>delta</b> bytes.
|
||||||
|
* Return 0 on suuccess, -1 on failure.
|
||||||
*/
|
*/
|
||||||
int
|
int
|
||||||
crypto_cipher_rewind(crypto_cipher_env_t *env, long delta)
|
crypto_cipher_rewind(crypto_cipher_env_t *env, long delta)
|
||||||
@ -1041,6 +1034,7 @@ crypto_cipher_rewind(crypto_cipher_env_t *env, long delta)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/** Move the position of the cipher stream forwards by <b>delta</b> bytes.
|
/** Move the position of the cipher stream forwards by <b>delta</b> bytes.
|
||||||
|
* Return 0 on suuccess, -1 on failure.
|
||||||
*/
|
*/
|
||||||
int
|
int
|
||||||
crypto_cipher_advance(crypto_cipher_env_t *env, long delta)
|
crypto_cipher_advance(crypto_cipher_env_t *env, long delta)
|
||||||
@ -1053,6 +1047,7 @@ crypto_cipher_advance(crypto_cipher_env_t *env, long delta)
|
|||||||
|
|
||||||
/** Compute the SHA1 digest of <b>len</b> bytes in data stored in
|
/** Compute the SHA1 digest of <b>len</b> bytes in data stored in
|
||||||
* <b>m</b>. Write the DIGEST_LEN byte result into <b>digest</b>.
|
* <b>m</b>. Write the DIGEST_LEN byte result into <b>digest</b>.
|
||||||
|
* Return 0 on suuccess, -1 on failure.
|
||||||
*/
|
*/
|
||||||
int crypto_digest(const unsigned char *m, int len, unsigned char *digest)
|
int crypto_digest(const unsigned char *m, int len, unsigned char *digest)
|
||||||
{
|
{
|
||||||
@ -1269,9 +1264,10 @@ int crypto_dh_get_public(crypto_dh_env_t *dh, char *pubkey, int pubkey_len)
|
|||||||
#undef MIN
|
#undef MIN
|
||||||
#define MIN(a,b) ((a)<(b)?(a):(b))
|
#define MIN(a,b) ((a)<(b)?(a):(b))
|
||||||
/** Given a DH key exchange object, and our peer's value of g^y (as a
|
/** Given a DH key exchange object, and our peer's value of g^y (as a
|
||||||
* <b>pubkey_len</b> byte value in <b>pubkey</b>) generate
|
* <b>pubkey_len</b>-byte value in <b>pubkey</b>) generate
|
||||||
* <b>secret_bytes_out</b> bytes of shared key material and write them
|
* <b>secret_bytes_out</b> bytes of shared key material and write them
|
||||||
* to <b>secret_out</b>.
|
* to <b>secret_out</b>. Return the number of bytes generated on suuccess,
|
||||||
|
* or -1 on failure.
|
||||||
*
|
*
|
||||||
* (We generate key material by computing
|
* (We generate key material by computing
|
||||||
* SHA1( g^xy || "\x00" ) || SHA1( g^xy || "\x01" ) || ...
|
* SHA1( g^xy || "\x00" ) || SHA1( g^xy || "\x01" ) || ...
|
||||||
@ -1324,7 +1320,7 @@ void crypto_dh_free(crypto_dh_env_t *dh)
|
|||||||
/* random numbers */
|
/* random numbers */
|
||||||
|
|
||||||
/** Seed OpenSSL's random number generator with DIGEST_LEN bytes from the
|
/** Seed OpenSSL's random number generator with DIGEST_LEN bytes from the
|
||||||
* operating system.
|
* operating system. Return 0 on suuccess, -1 on failure.
|
||||||
*/
|
*/
|
||||||
int crypto_seed_rng()
|
int crypto_seed_rng()
|
||||||
{
|
{
|
||||||
|
@ -216,6 +216,31 @@ int tor_strstrip(char *s, const char *strip)
|
|||||||
return read-s;
|
return read-s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Set the <b>dest_len</b>-byte buffer <b>buf</b> to contain the
|
||||||
|
* string <b>s</b>, with the string <b>insert</b> inserted after every
|
||||||
|
* <b>n</b> characters. Return 0 on success, -1 on failure.
|
||||||
|
*/
|
||||||
|
int tor_strpartition(char *dest, size_t dest_len,
|
||||||
|
const char *s, const char *insert, size_t n)
|
||||||
|
{
|
||||||
|
tor_assert(s && insert && n > 0);
|
||||||
|
int len_in, len_out, len_ins;
|
||||||
|
len_in = strlen(s);
|
||||||
|
len_ins = strlen(insert);
|
||||||
|
len_out = len_in + (len_in/n)*len_ins;
|
||||||
|
if (dest_len < len_out+1)
|
||||||
|
return -1;
|
||||||
|
while(len_in) {
|
||||||
|
strncpy(dest, s, n);
|
||||||
|
len_in -= n;
|
||||||
|
if (len_in < 0) break;
|
||||||
|
strcpy(dest+n, insert);
|
||||||
|
s += n;
|
||||||
|
dest += n+len_ins;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
#ifndef UNALIGNED_INT_ACCESS_OK
|
#ifndef UNALIGNED_INT_ACCESS_OK
|
||||||
/**
|
/**
|
||||||
* Read a 16-bit value beginning at <b>cp</b>. Equaivalent to
|
* Read a 16-bit value beginning at <b>cp</b>. Equaivalent to
|
||||||
|
@ -88,6 +88,9 @@ char *tor_strndup(const char *s, size_t n);
|
|||||||
void tor_strlower(char *s);
|
void tor_strlower(char *s);
|
||||||
int strcmpstart(const char *s1, const char *s2);
|
int strcmpstart(const char *s1, const char *s2);
|
||||||
int tor_strstrip(char *s, const char *strip);
|
int tor_strstrip(char *s, const char *strip);
|
||||||
|
int tor_strpartition(char *dest, size_t dest_len,
|
||||||
|
const char *s, const char *insert, size_t n);
|
||||||
|
|
||||||
|
|
||||||
/* Some platforms segfault when you try to access a multi-byte type
|
/* Some platforms segfault when you try to access a multi-byte type
|
||||||
* that isn't aligned to a word boundary. The macros and/or functions
|
* that isn't aligned to a word boundary. The macros and/or functions
|
||||||
|
Loading…
Reference in New Issue
Block a user