2013-11-16 00:26:54 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <openssl/sha.h>
|
|
|
|
#include <openssl/pem.h>
|
|
|
|
#include <limits.h>
|
|
|
|
|
2013-11-17 01:08:10 +01:00
|
|
|
#include "dispass.h"
|
|
|
|
|
2013-11-16 00:26:54 +01:00
|
|
|
#define MIN(A, B) ((A) < (B) ? (A) : (B))
|
|
|
|
#define MAXLEN (SHA512_DIGEST_LENGTH * 2)
|
|
|
|
|
2013-11-17 01:08:10 +01:00
|
|
|
static char *
|
2013-11-16 00:26:54 +01:00
|
|
|
base64encode(const void *data, int len)
|
|
|
|
{ /* Copied from http://stackoverflow.com/a/16511093/459915 */
|
|
|
|
BIO *b64_bio, *mem_bio;
|
|
|
|
BUF_MEM *mem_bio_mem_ptr;
|
2013-11-16 00:52:49 +01:00
|
|
|
char *ret;
|
2013-11-16 00:26:54 +01:00
|
|
|
|
|
|
|
b64_bio = BIO_new(BIO_f_base64());
|
|
|
|
mem_bio = BIO_new(BIO_s_mem());
|
|
|
|
BIO_push(b64_bio, mem_bio);
|
|
|
|
BIO_set_flags(b64_bio, BIO_FLAGS_BASE64_NO_NL);
|
|
|
|
|
|
|
|
BIO_write(b64_bio, data, len);
|
|
|
|
BIO_flush(b64_bio);
|
|
|
|
|
|
|
|
BIO_get_mem_ptr(mem_bio, &mem_bio_mem_ptr);
|
|
|
|
BIO_set_close(mem_bio, BIO_NOCLOSE);
|
|
|
|
BIO_free_all(b64_bio);
|
|
|
|
|
|
|
|
(*mem_bio_mem_ptr).data[(*mem_bio_mem_ptr).length] = '\0';
|
2013-11-16 00:52:49 +01:00
|
|
|
ret = calloc((*mem_bio_mem_ptr).length, sizeof(char));
|
|
|
|
strncpy(ret, (*mem_bio_mem_ptr).data, (*mem_bio_mem_ptr).length);
|
|
|
|
BUF_MEM_free(mem_bio_mem_ptr);
|
2013-11-16 00:26:54 +01:00
|
|
|
|
2013-11-16 00:52:49 +01:00
|
|
|
return ret;
|
2013-11-16 00:26:54 +01:00
|
|
|
}
|
|
|
|
|
2013-11-17 01:08:10 +01:00
|
|
|
static void
|
2013-11-16 00:26:54 +01:00
|
|
|
sha512_to_string(unsigned char *data, char *buff)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < SHA512_DIGEST_LENGTH; i++) {
|
|
|
|
char sbuff[3] = { '\0' };
|
|
|
|
sprintf(sbuff, "%02x", data[i]);
|
|
|
|
strcat(buff, sbuff);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-17 01:08:10 +01:00
|
|
|
static void
|
2013-11-17 00:34:14 +01:00
|
|
|
rmchar(char rm, char **s)
|
|
|
|
{
|
|
|
|
int i, j = 0;
|
|
|
|
char *new;
|
|
|
|
size_t len;
|
|
|
|
|
|
|
|
if (!strchr(*s, rm))
|
|
|
|
return;
|
|
|
|
|
|
|
|
len = strlen(*s);
|
|
|
|
new = calloc(len + 1, sizeof(char));
|
|
|
|
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
char c;
|
|
|
|
|
|
|
|
if ((c = (*s)[i]) != rm)
|
|
|
|
new[j++] = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
strncpy(*s, new, len);
|
|
|
|
free(new);
|
|
|
|
}
|
|
|
|
|
2013-11-16 00:26:54 +01:00
|
|
|
char *
|
2013-11-17 00:38:25 +01:00
|
|
|
dispass1(char *label, char *password, int len, long long unsigned seqno)
|
2013-11-16 00:26:54 +01:00
|
|
|
{
|
|
|
|
unsigned char *d;
|
2013-11-17 00:38:25 +01:00
|
|
|
size_t tbufflen = strlen(label) + strlen(password) + 1;
|
2013-11-16 00:26:54 +01:00
|
|
|
char *tbuff = calloc(tbufflen, sizeof(char));
|
|
|
|
char buff[MAXLEN + 1] = { '\0' };
|
|
|
|
char *b64;
|
|
|
|
|
|
|
|
strcat(tbuff, label);
|
|
|
|
strcat(tbuff, password);
|
2013-11-17 00:38:25 +01:00
|
|
|
d = SHA512((unsigned char *)tbuff, strlen(tbuff), 0);
|
2013-11-16 00:26:54 +01:00
|
|
|
free(tbuff);
|
|
|
|
sha512_to_string(d, buff);
|
|
|
|
b64 = base64encode(buff, strlen(buff));
|
|
|
|
b64[MIN(len, MAXLEN)] = '\0';
|
2013-11-17 00:34:14 +01:00
|
|
|
rmchar('=', &b64);
|
2013-11-16 00:26:54 +01:00
|
|
|
|
|
|
|
return b64;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *
|
2013-11-17 00:38:25 +01:00
|
|
|
dispass2(char *label, char *password, int len, long long unsigned seqno)
|
2013-11-16 00:26:54 +01:00
|
|
|
{
|
|
|
|
unsigned char *d;
|
|
|
|
char ibuff[300];
|
|
|
|
char *tbuff, *b64;
|
|
|
|
char buff[MAXLEN + 1] = { '\0' };
|
|
|
|
|
|
|
|
sprintf(ibuff, "%llu", seqno);
|
|
|
|
tbuff = calloc(strlen(label) + strlen(ibuff) + strlen(password) + 1,
|
|
|
|
sizeof(char));
|
|
|
|
strcat(tbuff, label);
|
|
|
|
strcat(tbuff, ibuff);
|
|
|
|
strcat(tbuff, password);
|
2013-11-17 00:38:25 +01:00
|
|
|
d = SHA512((unsigned char *)tbuff, strlen(tbuff), 0);
|
2013-11-16 00:26:54 +01:00
|
|
|
free(tbuff);
|
|
|
|
sha512_to_string(d, buff);
|
|
|
|
b64 = base64encode(buff, strlen(buff));
|
|
|
|
b64[MIN(len, MAXLEN)] = '\0';
|
2013-11-17 00:34:14 +01:00
|
|
|
rmchar('=', &b64);
|
2013-11-16 00:26:54 +01:00
|
|
|
|
|
|
|
return b64;
|
|
|
|
}
|