2008-08-06 18:36:30 +00:00
|
|
|
// This file is part of BOINC.
|
2005-01-20 23:22:22 +00:00
|
|
|
// http://boinc.berkeley.edu
|
2008-08-06 18:36:30 +00:00
|
|
|
// Copyright (C) 2008 University of California
|
2003-09-03 05:42:33 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// BOINC is free software; you can redistribute it and/or modify it
|
|
|
|
// under the terms of the GNU Lesser General Public License
|
|
|
|
// as published by the Free Software Foundation,
|
|
|
|
// either version 3 of the License, or (at your option) any later version.
|
2003-09-03 05:42:33 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// BOINC is distributed in the hope that it will be useful,
|
2005-01-20 23:22:22 +00:00
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
// See the GNU Lesser General Public License for more details.
|
2002-09-26 18:11:06 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with BOINC. If not, see <http://www.gnu.org/licenses/>.
|
2002-09-26 18:11:06 +00:00
|
|
|
|
2004-03-04 11:41:43 +00:00
|
|
|
#ifdef _WIN32
|
2004-06-16 23:16:08 +00:00
|
|
|
#include "boinc_win.h"
|
2004-03-04 11:41:43 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef _WIN32
|
2005-11-21 18:34:44 +00:00
|
|
|
#include "config.h"
|
2003-10-10 05:28:38 +00:00
|
|
|
#include <cctype>
|
2004-07-13 13:54:09 +00:00
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
2007-12-21 21:09:40 +00:00
|
|
|
#include <cstring>
|
2004-03-04 11:41:43 +00:00
|
|
|
#endif
|
2002-06-14 05:49:34 +00:00
|
|
|
|
|
|
|
#include "md5_file.h"
|
2002-07-11 01:09:53 +00:00
|
|
|
#include "error_numbers.h"
|
2002-06-14 05:49:34 +00:00
|
|
|
|
2002-08-30 20:56:02 +00:00
|
|
|
#include "crypt.h"
|
|
|
|
|
2004-07-03 21:38:15 +00:00
|
|
|
#ifdef _USING_FCGI_
|
|
|
|
#include "fcgi_stdio.h"
|
|
|
|
#endif
|
|
|
|
|
2002-07-09 00:10:58 +00:00
|
|
|
// NOTE: the fast CGI I/O library doesn't have fscanf(),
|
|
|
|
// so some of the following have been modified to use
|
|
|
|
// fgets() and sscanf() instead
|
|
|
|
|
2002-07-05 05:33:40 +00:00
|
|
|
// write some data in hex notation.
|
2002-06-14 05:49:34 +00:00
|
|
|
// NOTE: since length may not be known to the reader,
|
|
|
|
// we follow the data with a non-hex character '.'
|
|
|
|
//
|
|
|
|
int print_hex_data(FILE* f, DATA_BLOCK& x) {
|
2002-07-11 01:09:53 +00:00
|
|
|
unsigned int i;
|
2002-08-30 20:56:02 +00:00
|
|
|
|
2002-06-14 05:49:34 +00:00
|
|
|
for (i=0; i<x.len; i++) {
|
|
|
|
fprintf(f, "%02x", x.data[i]);
|
|
|
|
if (i%32==31) fprintf(f, "\n");
|
|
|
|
}
|
|
|
|
if (x.len%32 != 0) fprintf(f, "\n");
|
|
|
|
fprintf(f, ".\n");
|
2002-07-11 01:09:53 +00:00
|
|
|
return 0;
|
2002-06-14 05:49:34 +00:00
|
|
|
}
|
|
|
|
|
2002-07-05 05:33:40 +00:00
|
|
|
// same, but write to buffer
|
|
|
|
//
|
2008-05-09 21:27:22 +00:00
|
|
|
int sprint_hex_data(char* out_buf, DATA_BLOCK& x) {
|
2002-07-11 01:09:53 +00:00
|
|
|
unsigned int i;
|
2008-05-09 21:27:22 +00:00
|
|
|
const char hex[] = "0123456789abcdef";
|
|
|
|
char* p = out_buf;
|
2002-08-30 20:56:02 +00:00
|
|
|
|
2002-07-05 05:33:40 +00:00
|
|
|
for (i=0; i<x.len; i++) {
|
2008-05-09 21:27:22 +00:00
|
|
|
*p++ = hex[x.data[i]/16];
|
|
|
|
*p++ = hex[x.data[i]%16];
|
|
|
|
if (i%32==31) *p++ = '\n';
|
2002-07-05 05:33:40 +00:00
|
|
|
}
|
2008-05-09 21:27:22 +00:00
|
|
|
if (x.len%32 != 0) *p++ = '\n';
|
|
|
|
strcpy(p, ".\n");
|
2002-07-16 19:31:37 +00:00
|
|
|
|
2002-07-11 01:09:53 +00:00
|
|
|
return 0;
|
2002-07-05 05:33:40 +00:00
|
|
|
}
|
|
|
|
|
2002-06-14 05:49:34 +00:00
|
|
|
// scan data in hex notation.
|
|
|
|
// stop when you reach a non-parsed character.
|
2002-07-09 00:10:58 +00:00
|
|
|
// NOTE: buffer must be big enough; no checking is done.
|
2002-06-14 05:49:34 +00:00
|
|
|
//
|
|
|
|
int scan_hex_data(FILE* f, DATA_BLOCK& x) {
|
|
|
|
int n;
|
2002-08-30 20:56:02 +00:00
|
|
|
|
2002-06-14 05:49:34 +00:00
|
|
|
x.len = 0;
|
2002-07-09 00:10:58 +00:00
|
|
|
#if _USING_FCGI_
|
|
|
|
char *p, buf[256];
|
2002-07-11 01:09:53 +00:00
|
|
|
int i, j;
|
2002-07-09 00:10:58 +00:00
|
|
|
while (1) {
|
|
|
|
p = fgets(buf, 256, f);
|
2003-10-21 04:06:55 +00:00
|
|
|
if (!p) return ERR_GETS;
|
2002-07-09 00:10:58 +00:00
|
|
|
n = strlen(p)/2;
|
|
|
|
if (n == 0) break;
|
|
|
|
for (i=0; i<n; i++) {
|
|
|
|
sscanf(buf+i*2, "%2x", &j);
|
|
|
|
x.data[x.len] = j;
|
|
|
|
x.len++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
2002-06-14 05:49:34 +00:00
|
|
|
while (1) {
|
2003-06-17 20:18:56 +00:00
|
|
|
int j;
|
|
|
|
n = fscanf(f, "%2x", &j);
|
2002-07-09 00:10:58 +00:00
|
|
|
if (n <= 0) break;
|
2003-06-17 20:18:56 +00:00
|
|
|
x.data[x.len] = j;
|
2002-06-14 05:49:34 +00:00
|
|
|
x.len++;
|
|
|
|
}
|
2002-07-09 00:10:58 +00:00
|
|
|
#endif
|
2002-06-14 05:49:34 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-07-05 05:33:40 +00:00
|
|
|
// same, but read from buffer
|
|
|
|
//
|
2005-02-16 23:17:43 +00:00
|
|
|
static int sscan_hex_data(const char* p, DATA_BLOCK& x) {
|
2002-07-07 20:39:24 +00:00
|
|
|
int m, n, nleft=x.len;
|
2002-08-30 20:56:02 +00:00
|
|
|
|
2002-07-05 05:33:40 +00:00
|
|
|
x.len = 0;
|
|
|
|
while (1) {
|
2003-10-10 05:28:38 +00:00
|
|
|
if (isspace(*p)) {
|
|
|
|
++p;
|
|
|
|
continue;
|
|
|
|
}
|
2002-07-07 20:39:24 +00:00
|
|
|
n = sscanf(p, "%2x", &m);
|
2002-07-05 05:33:40 +00:00
|
|
|
if (n <= 0) break;
|
2002-07-07 20:39:24 +00:00
|
|
|
x.data[x.len++] = m;
|
|
|
|
nleft--;
|
|
|
|
if (nleft<0) {
|
|
|
|
fprintf(stderr, "sscan_hex_data: buffer overflow\n");
|
2003-09-03 23:54:11 +00:00
|
|
|
return ERR_BAD_HEX_FORMAT;
|
2002-07-07 20:39:24 +00:00
|
|
|
}
|
2002-07-05 05:33:40 +00:00
|
|
|
p += 2;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-06-14 05:49:34 +00:00
|
|
|
// print a key in ASCII form
|
|
|
|
//
|
|
|
|
int print_key_hex(FILE* f, KEY* key, int size) {
|
2002-07-11 01:09:53 +00:00
|
|
|
int len;
|
2002-06-14 05:49:34 +00:00
|
|
|
DATA_BLOCK x;
|
2002-08-30 20:56:02 +00:00
|
|
|
|
2002-06-14 05:49:34 +00:00
|
|
|
fprintf(f, "%d\n", key->bits);
|
|
|
|
len = size - sizeof(key->bits);
|
|
|
|
x.data = key->data;
|
|
|
|
x.len = len;
|
|
|
|
return print_hex_data(f, x);
|
|
|
|
}
|
|
|
|
|
|
|
|
int scan_key_hex(FILE* f, KEY* key, int size) {
|
2002-07-05 05:33:40 +00:00
|
|
|
int len, i, n;
|
2002-07-09 00:10:58 +00:00
|
|
|
int num_bits;
|
2002-08-30 20:56:02 +00:00
|
|
|
|
2002-07-09 00:10:58 +00:00
|
|
|
#if _USING_FCGI_
|
2002-07-11 01:09:53 +00:00
|
|
|
char *p, buf[256];
|
2002-07-09 00:10:58 +00:00
|
|
|
int j = 0, b;
|
|
|
|
fgets(buf, 256, f);
|
2002-07-05 22:02:01 +00:00
|
|
|
sscanf(buf, "%d", &num_bits);
|
|
|
|
key->bits = num_bits;
|
2002-06-14 05:49:34 +00:00
|
|
|
len = size - sizeof(key->bits);
|
2002-07-09 00:10:58 +00:00
|
|
|
while (1) {
|
|
|
|
p = fgets(buf, 256, f);
|
2008-02-06 19:32:51 +00:00
|
|
|
if (!p) break;
|
|
|
|
n = (strlen(p)-1)/2;
|
2002-07-09 00:10:58 +00:00
|
|
|
if (n == 0) break;
|
|
|
|
for (i=0; i<n; i++) {
|
|
|
|
sscanf(buf+i*2, "%2x", &b);
|
2008-02-06 19:32:51 +00:00
|
|
|
if (j == len) break;
|
2002-07-09 00:10:58 +00:00
|
|
|
key->data[j++] = b;
|
2002-07-05 22:02:01 +00:00
|
|
|
}
|
2002-06-14 05:49:34 +00:00
|
|
|
}
|
2008-02-06 19:32:51 +00:00
|
|
|
if (j != len) return ERR_NULL;
|
2002-07-09 00:10:58 +00:00
|
|
|
#else
|
|
|
|
fscanf(f, "%d", &num_bits);
|
|
|
|
key->bits = num_bits;
|
|
|
|
len = size - sizeof(key->bits);
|
|
|
|
for (i=0; i<len; i++) {
|
|
|
|
fscanf(f, "%2x", &n);
|
|
|
|
key->data[i] = n;
|
|
|
|
}
|
|
|
|
fscanf(f, ".");
|
|
|
|
#endif
|
2002-06-14 05:49:34 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-07-07 20:39:24 +00:00
|
|
|
// parse a text-encoded key from a memory buffer
|
|
|
|
//
|
2005-02-16 23:17:43 +00:00
|
|
|
int sscan_key_hex(const char* buf, KEY* key, int size) {
|
2002-07-11 01:09:53 +00:00
|
|
|
int n, retval,num_bits;
|
2002-07-07 20:39:24 +00:00
|
|
|
DATA_BLOCK db;
|
2002-07-16 19:31:37 +00:00
|
|
|
|
2002-07-11 01:09:53 +00:00
|
|
|
//fprintf(stderr, "buf = %s\n", buf);
|
|
|
|
n = sscanf(buf, "%d", &num_bits);
|
|
|
|
key->bits = num_bits; //key->bits is a short
|
|
|
|
//fprintf(stderr, "key->bits = %d\n", key->bits);
|
2002-07-16 19:31:37 +00:00
|
|
|
|
2006-12-19 20:49:12 +00:00
|
|
|
if (n != 1) return ERR_XML_PARSE;
|
2002-07-07 20:39:24 +00:00
|
|
|
buf = strchr(buf, '\n');
|
2006-12-19 20:49:12 +00:00
|
|
|
if (!buf) return ERR_XML_PARSE;
|
2002-07-07 20:39:24 +00:00
|
|
|
buf += 1;
|
|
|
|
db.data = key->data;
|
2002-07-11 01:09:53 +00:00
|
|
|
db.len = size - sizeof(key->bits); //huh???
|
2002-07-07 20:39:24 +00:00
|
|
|
retval = sscan_hex_data(buf, db);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2002-06-14 05:49:34 +00:00
|
|
|
// encrypt some data.
|
|
|
|
// The amount encrypted may be less than what's supplied.
|
|
|
|
// The output buffer must be at least MIN_OUT_BUFFER_SIZE.
|
|
|
|
// The output block must be decrypted in its entirety.
|
|
|
|
//
|
2005-08-11 17:41:08 +00:00
|
|
|
int encrypt_private(R_RSA_PRIVATE_KEY& key, DATA_BLOCK& in, DATA_BLOCK& out) {
|
2005-08-02 16:56:09 +00:00
|
|
|
int n, modulus_len;
|
2002-08-30 20:56:02 +00:00
|
|
|
|
2002-07-05 05:33:40 +00:00
|
|
|
modulus_len = (key.bits+7)/8;
|
2002-06-14 05:49:34 +00:00
|
|
|
n = in.len;
|
2002-07-05 05:33:40 +00:00
|
|
|
if (n >= modulus_len-11) {
|
|
|
|
n = modulus_len-11;
|
2002-06-14 05:49:34 +00:00
|
|
|
}
|
2005-08-01 22:44:40 +00:00
|
|
|
RSA* rp = RSA_new();
|
|
|
|
private_to_openssl(key, rp);
|
|
|
|
RSA_private_encrypt(n, in.data, out.data, rp, RSA_PKCS1_PADDING);
|
|
|
|
out.len = RSA_size(rp);
|
|
|
|
RSA_free(rp);
|
2002-06-14 05:49:34 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int decrypt_public(R_RSA_PUBLIC_KEY& key, DATA_BLOCK& in, DATA_BLOCK& out) {
|
2005-08-01 22:44:40 +00:00
|
|
|
RSA* rp = RSA_new();
|
|
|
|
public_to_openssl(key, rp);
|
|
|
|
RSA_public_decrypt(in.len, in.data, out.data, rp, RSA_PKCS1_PADDING);
|
|
|
|
out.len = RSA_size(rp);
|
|
|
|
return 0;
|
2002-06-14 05:49:34 +00:00
|
|
|
}
|
|
|
|
|
2005-02-16 23:17:43 +00:00
|
|
|
int sign_file(const char* path, R_RSA_PRIVATE_KEY& key, DATA_BLOCK& signature) {
|
2002-07-05 05:33:40 +00:00
|
|
|
char md5_buf[MD5_LEN];
|
2002-06-14 05:49:34 +00:00
|
|
|
double file_length;
|
|
|
|
DATA_BLOCK in_block;
|
2005-08-11 17:41:08 +00:00
|
|
|
int retval;
|
2002-08-30 20:56:02 +00:00
|
|
|
|
2002-06-14 05:49:34 +00:00
|
|
|
retval = md5_file(path, md5_buf, file_length);
|
|
|
|
if (retval) return retval;
|
|
|
|
in_block.data = (unsigned char*)md5_buf;
|
2005-11-13 20:59:27 +00:00
|
|
|
in_block.len = (unsigned int)strlen(md5_buf);
|
2005-08-11 17:41:08 +00:00
|
|
|
retval = encrypt_private(key, in_block, signature);
|
2002-06-14 05:49:34 +00:00
|
|
|
if (retval) return retval;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-07-05 05:33:40 +00:00
|
|
|
int sign_block(DATA_BLOCK& data_block, R_RSA_PRIVATE_KEY& key, DATA_BLOCK& signature) {
|
|
|
|
char md5_buf[MD5_LEN];
|
2005-08-11 17:41:08 +00:00
|
|
|
int retval;
|
2002-07-05 05:33:40 +00:00
|
|
|
DATA_BLOCK in_block;
|
|
|
|
|
|
|
|
md5_block(data_block.data, data_block.len, md5_buf);
|
|
|
|
in_block.data = (unsigned char*)md5_buf;
|
2005-11-13 20:59:27 +00:00
|
|
|
in_block.len = (unsigned int)strlen(md5_buf);
|
2005-08-11 17:41:08 +00:00
|
|
|
retval = encrypt_private(key, in_block, signature);
|
2002-07-05 05:33:40 +00:00
|
|
|
if (retval) {
|
|
|
|
printf("sign_block: encrypt_private returned %d\n", retval);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-12-14 23:43:50 +00:00
|
|
|
// compute an XML signature element for some text
|
|
|
|
//
|
|
|
|
int generate_signature(
|
|
|
|
char* text_to_sign, char* signature_hex, R_RSA_PRIVATE_KEY& key
|
|
|
|
) {
|
|
|
|
DATA_BLOCK block, signature_data;
|
|
|
|
unsigned char signature_buf[SIGNATURE_SIZE_BINARY];
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
block.data = (unsigned char*)text_to_sign;
|
2008-08-13 17:26:26 +00:00
|
|
|
block.len = (unsigned int)strlen(text_to_sign);
|
2005-12-14 23:43:50 +00:00
|
|
|
signature_data.data = signature_buf;
|
|
|
|
signature_data.len = SIGNATURE_SIZE_BINARY;
|
|
|
|
retval = sign_block(block, key, signature_data);
|
|
|
|
if (retval) return retval;
|
|
|
|
sprint_hex_data(signature_hex, signature_data);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-06-14 05:49:34 +00:00
|
|
|
int verify_file(
|
2005-02-16 23:17:43 +00:00
|
|
|
const char* path, R_RSA_PUBLIC_KEY& key, DATA_BLOCK& signature, bool& answer
|
2002-06-14 05:49:34 +00:00
|
|
|
) {
|
2002-07-07 20:39:24 +00:00
|
|
|
char md5_buf[MD5_LEN], clear_buf[MD5_LEN];
|
2002-06-14 05:49:34 +00:00
|
|
|
double file_length;
|
|
|
|
int n, retval;
|
|
|
|
DATA_BLOCK clear_signature;
|
2002-08-30 20:56:02 +00:00
|
|
|
|
2002-06-14 05:49:34 +00:00
|
|
|
retval = md5_file(path, md5_buf, file_length);
|
2002-07-11 01:09:53 +00:00
|
|
|
if (retval) {
|
|
|
|
fprintf(stderr, "error: verify_file: md5_file error %d\n", retval);
|
|
|
|
return retval;
|
|
|
|
}
|
2005-11-13 20:59:27 +00:00
|
|
|
n = (int)strlen(md5_buf);
|
2002-06-14 05:49:34 +00:00
|
|
|
clear_signature.data = (unsigned char*)clear_buf;
|
2002-07-07 20:39:24 +00:00
|
|
|
clear_signature.len = MD5_LEN;
|
2002-06-14 05:49:34 +00:00
|
|
|
retval = decrypt_public(key, signature, clear_signature);
|
2002-07-11 01:09:53 +00:00
|
|
|
if (retval) {
|
|
|
|
fprintf(stderr, "error: verify_file: decrypt_public error %d\n", retval);
|
|
|
|
return retval;
|
|
|
|
}
|
2002-06-14 05:49:34 +00:00
|
|
|
answer = !strncmp(md5_buf, clear_buf, n);
|
|
|
|
return 0;
|
|
|
|
}
|
2002-07-05 05:33:40 +00:00
|
|
|
|
2002-07-07 20:39:24 +00:00
|
|
|
int verify_file2(
|
2005-02-16 23:17:43 +00:00
|
|
|
const char* path, const char* signature_text, const char* key_text, bool& answer
|
2002-07-07 20:39:24 +00:00
|
|
|
) {
|
|
|
|
R_RSA_PUBLIC_KEY key;
|
|
|
|
unsigned char signature_buf[SIGNATURE_SIZE_BINARY];
|
|
|
|
int retval;
|
|
|
|
DATA_BLOCK signature;
|
2002-08-30 20:56:02 +00:00
|
|
|
|
2002-07-07 20:39:24 +00:00
|
|
|
retval = sscan_key_hex(key_text, (KEY*)&key, sizeof(key));
|
2002-07-11 01:09:53 +00:00
|
|
|
if (retval) {
|
|
|
|
fprintf(stderr, "error: verify_file2: sscan_key_hex did not work\n");
|
|
|
|
return retval;
|
|
|
|
}
|
2002-07-07 20:39:24 +00:00
|
|
|
signature.data = signature_buf;
|
|
|
|
signature.len = sizeof(signature_buf);
|
2003-09-03 23:54:11 +00:00
|
|
|
retval = sscan_hex_data(signature_text, signature);
|
|
|
|
if (retval) return retval;
|
2002-07-07 20:39:24 +00:00
|
|
|
return verify_file(path, key, signature, answer);
|
|
|
|
}
|
|
|
|
|
2002-07-05 05:33:40 +00:00
|
|
|
// verify, where both text and signature are char strings
|
|
|
|
//
|
|
|
|
int verify_string(
|
2005-02-16 23:17:43 +00:00
|
|
|
const char* text, const char* signature_text, R_RSA_PUBLIC_KEY& key, bool& answer
|
2002-07-05 05:33:40 +00:00
|
|
|
) {
|
|
|
|
char md5_buf[MD5_LEN];
|
2002-07-07 20:39:24 +00:00
|
|
|
unsigned char signature_buf[SIGNATURE_SIZE_BINARY];
|
2002-07-05 05:33:40 +00:00
|
|
|
char clear_buf[MD5_LEN];
|
|
|
|
int retval, n;
|
|
|
|
DATA_BLOCK signature, clear_signature;
|
2002-08-30 20:56:02 +00:00
|
|
|
|
2005-11-13 20:59:27 +00:00
|
|
|
retval = md5_block((const unsigned char*)text, (int)strlen(text), md5_buf);
|
2002-07-05 05:33:40 +00:00
|
|
|
if (retval) return retval;
|
2005-11-13 20:59:27 +00:00
|
|
|
n = (int)strlen(md5_buf);
|
2002-07-05 05:33:40 +00:00
|
|
|
signature.data = signature_buf;
|
2002-07-07 20:39:24 +00:00
|
|
|
signature.len = sizeof(signature_buf);
|
2003-09-03 23:54:11 +00:00
|
|
|
retval = sscan_hex_data(signature_text, signature);
|
|
|
|
if (retval) return retval;
|
2002-07-05 05:33:40 +00:00
|
|
|
clear_signature.data = (unsigned char*)clear_buf;
|
|
|
|
clear_signature.len = 256;
|
|
|
|
retval = decrypt_public(key, signature, clear_signature);
|
|
|
|
if (retval) return retval;
|
|
|
|
answer = !strncmp(md5_buf, clear_buf, n);
|
|
|
|
return 0;
|
|
|
|
}
|
2002-07-07 20:39:24 +00:00
|
|
|
|
|
|
|
// Same, where public key is also encoded as text
|
|
|
|
//
|
|
|
|
int verify_string2(
|
2005-02-16 23:17:43 +00:00
|
|
|
const char* text, const char* signature_text, const char* key_text, bool& answer
|
2002-07-07 20:39:24 +00:00
|
|
|
) {
|
|
|
|
R_RSA_PUBLIC_KEY key;
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
retval = sscan_key_hex(key_text, (KEY*)&key, sizeof(key));
|
|
|
|
if (retval) return retval;
|
|
|
|
return verify_string(text, signature_text, key, answer);
|
|
|
|
}
|
2003-09-03 05:42:33 +00:00
|
|
|
|
2005-02-16 23:17:43 +00:00
|
|
|
int read_key_file(const char* keyfile, R_RSA_PRIVATE_KEY& key) {
|
2003-09-03 05:42:33 +00:00
|
|
|
int retval;
|
|
|
|
FILE* fkey = fopen(keyfile, "r");
|
|
|
|
if (!fkey) {
|
|
|
|
fprintf(stderr, "can't open key file (%s)\n", keyfile);
|
2003-10-21 04:06:55 +00:00
|
|
|
return ERR_FOPEN;
|
2003-09-03 05:42:33 +00:00
|
|
|
}
|
|
|
|
retval = scan_key_hex(fkey, (KEY*)&key, sizeof(key));
|
|
|
|
fclose(fkey);
|
|
|
|
if (retval) {
|
|
|
|
fprintf(stderr, "can't parse key\n");
|
2003-10-21 04:06:55 +00:00
|
|
|
return retval;
|
2003-09-03 05:42:33 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2004-12-08 00:40:19 +00:00
|
|
|
|
2005-08-01 22:44:40 +00:00
|
|
|
#ifdef USE_OPENSSL
|
|
|
|
|
|
|
|
static void bn_to_bin(BIGNUM* bn, unsigned char* bin, int n) {
|
|
|
|
memset(bin, 0, n);
|
|
|
|
int m = BN_num_bytes(bn);
|
|
|
|
BN_bn2bin(bn, bin+n-m);
|
|
|
|
}
|
|
|
|
|
|
|
|
void openssl_to_keys(
|
|
|
|
RSA* rp, int nbits, R_RSA_PRIVATE_KEY& priv, R_RSA_PUBLIC_KEY& pub
|
|
|
|
) {
|
|
|
|
pub.bits = nbits;
|
|
|
|
bn_to_bin(rp->n, pub.modulus, sizeof(pub.modulus));
|
|
|
|
bn_to_bin(rp->e, pub.exponent, sizeof(pub.exponent));
|
|
|
|
|
|
|
|
memset(&priv, 0, sizeof(priv));
|
|
|
|
priv.bits = nbits;
|
|
|
|
bn_to_bin(rp->n, priv.modulus, sizeof(priv.modulus));
|
|
|
|
bn_to_bin(rp->e, priv.publicExponent, sizeof(priv.publicExponent));
|
|
|
|
bn_to_bin(rp->d, priv.exponent, sizeof(priv.exponent));
|
|
|
|
bn_to_bin(rp->p, priv.prime[0], sizeof(priv.prime[0]));
|
|
|
|
bn_to_bin(rp->q, priv.prime[1], sizeof(priv.prime[1]));
|
|
|
|
bn_to_bin(rp->dmp1, priv.primeExponent[0], sizeof(priv.primeExponent[0]));
|
|
|
|
bn_to_bin(rp->dmq1, priv.primeExponent[1], sizeof(priv.primeExponent[1]));
|
|
|
|
bn_to_bin(rp->iqmp, priv.coefficient, sizeof(priv.coefficient));
|
|
|
|
}
|
|
|
|
|
|
|
|
void private_to_openssl(R_RSA_PRIVATE_KEY& priv, RSA* rp) {
|
|
|
|
rp->n = BN_bin2bn(priv.modulus, sizeof(priv.modulus), 0);
|
|
|
|
rp->e = BN_bin2bn(priv.publicExponent, sizeof(priv.publicExponent), 0);
|
|
|
|
rp->d = BN_bin2bn(priv.exponent, sizeof(priv.exponent), 0);
|
|
|
|
rp->p = BN_bin2bn(priv.prime[0], sizeof(priv.prime[0]), 0);
|
|
|
|
rp->q = BN_bin2bn(priv.prime[1], sizeof(priv.prime[1]), 0);
|
|
|
|
rp->dmp1 = BN_bin2bn(priv.primeExponent[0], sizeof(priv.primeExponent[0]), 0);
|
|
|
|
rp->dmq1 = BN_bin2bn(priv.primeExponent[1], sizeof(priv.primeExponent[1]), 0);
|
|
|
|
rp->iqmp = BN_bin2bn(priv.coefficient, sizeof(priv.coefficient), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void public_to_openssl(R_RSA_PUBLIC_KEY& pub, RSA* rp) {
|
|
|
|
rp->n = BN_bin2bn(pub.modulus, sizeof(pub.modulus), 0);
|
|
|
|
rp->e = BN_bin2bn(pub.exponent, sizeof(pub.exponent), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
2005-01-02 18:29:53 +00:00
|
|
|
const char *BOINC_RCSID_4f0c2e42ea = "$Id$";
|