Merge pull request #308 from AitorATuin/kdf_input

Fixes a problem reading passwords when using the kdf wrapper
This commit is contained in:
Jaromil 2018-09-23 11:04:10 +01:00 committed by GitHub
commit e8919af867
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 125 additions and 32 deletions

View File

@ -20,4 +20,5 @@ before_script:
- sudo make --directory=extras/kdf-keys install - sudo make --directory=extras/kdf-keys install
script: script:
- make --directory=extras/kdf-keys test
- make test - make test

View File

@ -7,6 +7,10 @@ all:
$(CC) -O2 -o tomb-kdb-pbkdf2-gensalt gen_salt.c -lgcrypt $(CC) -O2 -o tomb-kdb-pbkdf2-gensalt gen_salt.c -lgcrypt
$(CC) -O2 -o tomb-kdb-hexencode hexencode.c $(CC) -O2 -o tomb-kdb-hexencode hexencode.c
test:
@echo "Running Tomb-kdb tests"
./test.sh
clean: clean:
rm -f tomb-kdb-pbkdf2 tomb-kdb-pbkdf2-getiter tomb-kdb-pbkdf2-gensalt tomb-kdb-hexencode rm -f tomb-kdb-pbkdf2 tomb-kdb-pbkdf2-getiter tomb-kdb-pbkdf2-gensalt tomb-kdb-hexencode

View File

@ -43,6 +43,9 @@
#include <gcrypt.h> #include <gcrypt.h>
/* block size for password buffer */
#define BLOCK_SIZE 40
/* TODO: move print_hex and hex_to_binary to utils.h, with separate compiling */ /* TODO: move print_hex and hex_to_binary to utils.h, with separate compiling */
void print_hex(unsigned char *buf, int len) void print_hex(unsigned char *buf, int len)
{ {
@ -73,15 +76,37 @@ int hex_to_binary(unsigned char *buf, char *hex)
return count; return count;
} }
void cleanup(char *result, int result_len, char *pass, char *salt, int salt_len) {
int i;
//clear and free everything
if (result) {
for(i=0; i<result_len;i++)
result[i]=0;
free(result);
}
if (pass) {
for(i=0; i<strlen(pass); i++) //blank
pass[i]=0;
free(pass);
}
if (salt) {
for(i=0; i<salt_len; i++) //blank
salt[i]=0;
free(salt);
}
}
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
char *pass = NULL; char *pass = NULL;
unsigned char *salt; unsigned char *salt = NULL;
int salt_len; // salt length in bytes int salt_len; // salt length in bytes
int ic=0; // iterative count int ic=0; // iterative count
int result_len; int result_len;
unsigned char *result; // result (binary - 32+16 chars) unsigned char *result = NULL; // result (binary - 32+16 chars)
int i; int i, pw_len = 0;
int buff_len = BLOCK_SIZE;
if ( argc != 4 ) { if ( argc != 4 ) {
fprintf(stderr, "usage: %s salt count len <passwd >binary_key_iv\n", argv[0]); fprintf(stderr, "usage: %s salt count len <passwd >binary_key_iv\n", argv[0]);
@ -89,7 +114,8 @@ int main(int argc, char *argv[])
} }
//TODO: move to base64decode //TODO: move to base64decode
salt=calloc(strlen(argv[1])/2+3, sizeof(char)); salt_len = strlen(argv[1])/2+3;
salt = calloc(salt_len, sizeof(char));
salt_len=hex_to_binary(salt, argv[1]); salt_len=hex_to_binary(salt, argv[1]);
if( salt_len <= 0 ) { if( salt_len <= 0 ) {
fprintf(stderr, "Error: %s is not a valid salt (it must be a hexadecimal string)\n", argv[1]); fprintf(stderr, "Error: %s is not a valid salt (it must be a hexadecimal string)\n", argv[1]);
@ -105,14 +131,41 @@ int main(int argc, char *argv[])
exit(1); exit(1);
} }
fscanf(stdin, "%ms", &pass); /* Read password char by char.
if ( pass[strlen(pass)-1] == '\n' ) *
pass[strlen(pass)-1] = '\0'; * Doing in this way we make sure that blanks (even null bytes) end up
* in the password.
*
* passwords containing just a bunch of spaces are valid
*/
pass = calloc(buff_len, sizeof(char));
char c = getchar();
while (c != EOF) {
if (pw_len == buff_len) {
buff_len *= 2;
pass = realloc(pass, buff_len);
if (!pass) {
fprintf(stderr, "Error allocating memory");
cleanup(result, result_len, pass, salt, salt_len);
exit(3);
}
}
pass[pw_len] = c;
pw_len++;
c = getchar();
}
if (pw_len <= 1) {
fprintf(stderr, "Error: password is empty\n");
cleanup(result, result_len, pass, salt, salt_len);
exit(1);
}
pass[pw_len-1] = '\0';
// PBKDF 2 // PBKDF 2
result = calloc(result_len, sizeof(unsigned char*)); result = calloc(result_len, sizeof(unsigned char*));
if (!gcry_check_version ("1.5.0")) { if (!gcry_check_version ("1.5.0")) {
fputs ("libgcrypt version mismatch\n", stderr); fputs ("libgcrypt version mismatch\n", stderr);
cleanup(result, result_len, pass, salt, salt_len);
exit (2); exit (2);
} }
/* Allocate a pool of 16k secure memory. This make the secure memory /* Allocate a pool of 16k secure memory. This make the secure memory
@ -124,19 +177,10 @@ int main(int argc, char *argv[])
/* Tell Libgcrypt that initialization has completed. */ /* Tell Libgcrypt that initialization has completed. */
gcry_control (GCRYCTL_INITIALIZATION_FINISHED, 0); gcry_control (GCRYCTL_INITIALIZATION_FINISHED, 0);
gcry_kdf_derive( pass, strlen(pass), GCRY_KDF_PBKDF2, GCRY_MD_SHA1, salt, salt_len, ic, result_len, result); gcry_kdf_derive(pass, pw_len-1, GCRY_KDF_PBKDF2, GCRY_MD_SHA1, salt, salt_len, ic, result_len, result);
print_hex(result, result_len); // Key + IV (as hex string) print_hex(result, result_len); // Key + IV (as hex string)
//clear and free everything cleanup(result, result_len, pass, salt, salt_len);
for(i=0; i<result_len;i++)
result[i]=0;
free(result);
for(i=0; i<strlen(pass); i++) //blank
pass[i]=0;
free(pass);
for(i=0; i<strlen(argv[1])/2+3; i++) //blank
salt[i]=0;
free(salt);
return(0); return(0);
} }

52
extras/kdf-keys/test.sh Normal file → Executable file
View File

@ -1,7 +1,9 @@
#!/usr/bin/env zsh #!/usr/bin/env zsh
error=0 error=0
while read line; do
check_kdf() {
while read line; do
pass=`cut -f1 <<<$line` pass=`cut -f1 <<<$line`
salt=`cut -f2 <<<$line` salt=`cut -f2 <<<$line`
iter=`cut -f3 <<<$line` iter=`cut -f3 <<<$line`
@ -9,13 +11,55 @@ while read line; do
expected=`cut -f5 <<<$line` expected=`cut -f5 <<<$line`
hexsalt=`cut -f6 <<<$line` hexsalt=`cut -f6 <<<$line`
#TODO: check! #TODO: check!
derived=`./pbkdf2 $hexsalt $iter $keylen <<<$pass` derived=`./tomb-kdb-pbkdf2 $hexsalt $iter $keylen <<<$pass`
if [[ $derived != $expected ]]; then if [[ $derived != $expected ]]; then
echo ./pbkdf2 $hexsalt $iter $keylen "<<<$pass"
echo "Expected $expected, got $derived" >&2 echo "Expected $expected, got $derived" >&2
error=$((error + 1)) error=$((error + 1))
fi fi
done < test.txt done < test.txt
}
check_white_spaces() {
hexsalt="73616c74"
iter=4096
keylen=20
typeset -a results
passwords=('one two three' 'one two' 'one')
for pwd in $passwords; do
results+=`./tomb-kdb-pbkdf2 $hexsalt $iter $keylen <<<$pwd`
done
for ((i=1;i<=3;i++)); do
d1=$results[$i]
i1=$passwords[$i]
for ((j=(($i+1));j<=3;j++)); do
d2=$results[$j]
i2=$passwords[$j]
if [[ $d1 == $d2 ]]; then
echo "Inputs \"$i1\" and \"$i2\" produce the same output $d1" >&2
error=$((error + 1))
fi
done
done
}
check_password_len() {
hexsalt="73616c74"
iter=4096
keylen=20
./tomb-kdb-pbkdf2 $hexsalt $iter $keylen 2>/dev/null <<<"" && {
echo "Empty passwords are accepted"
error=$((error + 1))
}
bigpassword=`perl -e 'print "a"x3000'`
./tomb-kdb-pbkdf2 $hexsalt $iter $keylen &>/dev/null <<<"$bigpassword" || {
echo "Error when using long password"
error=$((error + 1))
}
}
check_kdf
check_white_spaces
check_password_len
if [[ $error == 1 ]]; then if [[ $error == 1 ]]; then
exit $error exit $error