2010-12-15 23:45:21 +01:00
|
|
|
/*
|
2011-02-20 13:02:15 +01:00
|
|
|
Copyright (C) 2006-2011 Con Kolivas
|
2011-02-21 06:11:59 +01:00
|
|
|
Copyright (C) 2008, 2011 Peter Hyman
|
2010-12-15 23:45:21 +01:00
|
|
|
Copyright (C) 1998 Andrew Tridgell
|
2010-11-04 11:14:55 +01:00
|
|
|
|
2010-03-29 01:07:08 +02:00
|
|
|
This program 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 2 of the License, or
|
|
|
|
|
(at your option) any later version.
|
2010-12-15 23:45:21 +01:00
|
|
|
|
2010-03-29 01:07:08 +02:00
|
|
|
This program 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.
|
2010-12-15 23:45:21 +01:00
|
|
|
|
2010-03-29 01:07:08 +02:00
|
|
|
You should have received a copy of the GNU General Public License
|
2010-12-15 23:45:21 +01:00
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2010-03-29 01:07:08 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Utilities used in rzip
|
|
|
|
|
|
|
|
|
|
tridge, June 1996
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Realloc removed
|
|
|
|
|
* Functions added
|
|
|
|
|
* read_config()
|
|
|
|
|
* Peter Hyman, December 2008
|
|
|
|
|
*/
|
|
|
|
|
|
2011-03-08 22:34:44 +01:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
|
# include "config.h"
|
|
|
|
|
#endif
|
|
|
|
|
|
2011-03-08 22:37:26 +01:00
|
|
|
#include <stdarg.h>
|
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
|
# include <unistd.h>
|
|
|
|
|
#endif
|
2011-03-15 06:32:32 +01:00
|
|
|
#include <termios.h>
|
2011-03-08 22:37:26 +01:00
|
|
|
|
|
|
|
|
#ifdef _SC_PAGE_SIZE
|
|
|
|
|
# define PAGE_SIZE (sysconf(_SC_PAGE_SIZE))
|
|
|
|
|
#else
|
|
|
|
|
# define PAGE_SIZE (4096)
|
|
|
|
|
#endif
|
|
|
|
|
|
2011-03-15 02:56:23 +01:00
|
|
|
#include <sys/types.h>
|
|
|
|
|
#include <sys/stat.h>
|
2011-03-18 00:22:58 +01:00
|
|
|
#include <sys/mman.h>
|
2011-03-15 02:56:23 +01:00
|
|
|
#include <fcntl.h>
|
2011-03-08 22:37:26 +01:00
|
|
|
#include "lrzip_private.h"
|
2011-03-15 02:56:23 +01:00
|
|
|
#include "liblrzip.h"
|
2011-03-21 10:54:53 +01:00
|
|
|
#include "util.h"
|
2011-03-18 00:22:58 +01:00
|
|
|
#include "sha4.h"
|
|
|
|
|
#include "aes.h"
|
|
|
|
|
|
2011-03-08 22:32:14 +01:00
|
|
|
static const char *infile = NULL;
|
|
|
|
|
static char delete_infile = 0;
|
|
|
|
|
static const char *outfile = NULL;
|
|
|
|
|
static char delete_outfile = 0;
|
|
|
|
|
static FILE *outputfile = NULL;
|
|
|
|
|
|
|
|
|
|
void register_infile(const char *name, char delete)
|
|
|
|
|
{
|
|
|
|
|
infile = name;
|
|
|
|
|
delete_infile = delete;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void register_outfile(const char *name, char delete)
|
|
|
|
|
{
|
|
|
|
|
outfile = name;
|
|
|
|
|
delete_outfile = delete;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void register_outputfile(FILE *f)
|
|
|
|
|
{
|
|
|
|
|
outputfile = f;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-08 22:34:44 +01:00
|
|
|
void unlink_files(void)
|
2011-02-21 04:17:05 +01:00
|
|
|
{
|
|
|
|
|
/* Delete temporary files generated for testing or faking stdio */
|
2011-03-08 22:32:14 +01:00
|
|
|
if (outfile && delete_outfile)
|
|
|
|
|
unlink(outfile);
|
2011-02-21 04:17:05 +01:00
|
|
|
|
2011-03-08 22:32:14 +01:00
|
|
|
if (infile && delete_infile)
|
|
|
|
|
unlink(infile);
|
2011-02-21 04:17:05 +01:00
|
|
|
}
|
|
|
|
|
|
2011-02-21 06:11:59 +01:00
|
|
|
static void fatal_exit(void)
|
|
|
|
|
{
|
2011-03-15 06:32:32 +01:00
|
|
|
struct termios termios_p;
|
|
|
|
|
|
|
|
|
|
/* Make sure we haven't died after disabling stdin echo */
|
|
|
|
|
tcgetattr(fileno(stdin), &termios_p);
|
|
|
|
|
termios_p.c_lflag |= ECHO;
|
|
|
|
|
tcsetattr(fileno(stdin), 0, &termios_p);
|
|
|
|
|
|
2011-02-21 06:11:59 +01:00
|
|
|
unlink_files();
|
2011-03-08 22:32:14 +01:00
|
|
|
fprintf(outputfile, "Fatal error - exiting\n");
|
|
|
|
|
fflush(outputfile);
|
2011-02-21 06:11:59 +01:00
|
|
|
exit(1);
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-21 04:51:20 +01:00
|
|
|
/* Failure when there is likely to be a meaningful error in perror */
|
2010-03-29 01:07:08 +02:00
|
|
|
void fatal(const char *format, ...)
|
|
|
|
|
{
|
|
|
|
|
va_list ap;
|
|
|
|
|
|
|
|
|
|
if (format) {
|
|
|
|
|
va_start(ap, format);
|
|
|
|
|
vfprintf(stderr, format, ap);
|
|
|
|
|
va_end(ap);
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-04 11:14:55 +01:00
|
|
|
perror(NULL);
|
2011-02-21 06:11:59 +01:00
|
|
|
fatal_exit();
|
2010-03-29 01:07:08 +02:00
|
|
|
}
|
|
|
|
|
|
2011-02-21 04:51:20 +01:00
|
|
|
void failure(const char *format, ...)
|
|
|
|
|
{
|
|
|
|
|
va_list ap;
|
|
|
|
|
|
|
|
|
|
if (format) {
|
|
|
|
|
va_start(ap, format);
|
|
|
|
|
vfprintf(stderr, format, ap);
|
|
|
|
|
va_end(ap);
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-21 06:11:59 +01:00
|
|
|
fatal_exit();
|
2011-02-21 04:51:20 +01:00
|
|
|
}
|
|
|
|
|
|
2011-02-21 14:49:50 +01:00
|
|
|
void round_to_page(i64 *size)
|
|
|
|
|
{
|
2011-03-08 22:32:14 +01:00
|
|
|
*size -= *size % PAGE_SIZE;
|
2011-02-21 14:49:50 +01:00
|
|
|
if (unlikely(!*size))
|
2011-03-08 22:32:14 +01:00
|
|
|
*size = PAGE_SIZE;
|
2011-02-21 14:49:50 +01:00
|
|
|
}
|
2011-03-15 02:56:23 +01:00
|
|
|
|
|
|
|
|
void get_rand(uchar *buf, int len)
|
|
|
|
|
{
|
|
|
|
|
int fd, i;
|
|
|
|
|
|
|
|
|
|
fd = open("/dev/urandom", O_RDONLY);
|
|
|
|
|
if (fd == -1) {
|
|
|
|
|
for (i = 0; i < len; i++)
|
|
|
|
|
buf[i] = (uchar)random();
|
|
|
|
|
} else {
|
|
|
|
|
if (unlikely(read(fd, buf, len) != len))
|
|
|
|
|
fatal("Failed to read fd in get_rand\n");
|
|
|
|
|
if (unlikely(close(fd)))
|
|
|
|
|
fatal("Failed to close fd in get_rand\n");
|
|
|
|
|
}
|
|
|
|
|
}
|
2011-03-17 04:10:58 +01:00
|
|
|
|
|
|
|
|
static void xor128 (void *pa, const void *pb)
|
|
|
|
|
{
|
|
|
|
|
i64 *a = pa;
|
|
|
|
|
const i64 *b = pb;
|
|
|
|
|
|
|
|
|
|
a [0] ^= b [0];
|
|
|
|
|
a [1] ^= b [1];
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-21 10:54:53 +01:00
|
|
|
static void lrz_keygen(const rzip_control *control, const uchar *salt, uchar *key, uchar *iv)
|
|
|
|
|
{
|
|
|
|
|
uchar buf [HASH_LEN + SALT_LEN + PASS_LEN];
|
|
|
|
|
mlock(buf, HASH_LEN + SALT_LEN + PASS_LEN);
|
|
|
|
|
|
|
|
|
|
memcpy(buf, control->hash, HASH_LEN);
|
|
|
|
|
memcpy(buf + HASH_LEN, salt, SALT_LEN);
|
|
|
|
|
memcpy(buf + HASH_LEN + SALT_LEN, control->salt_pass, control->salt_pass_len);
|
|
|
|
|
sha4(buf, HASH_LEN + SALT_LEN + control->salt_pass_len, key, 0);
|
|
|
|
|
|
|
|
|
|
memcpy(buf, key, HASH_LEN);
|
|
|
|
|
memcpy(buf + HASH_LEN, salt, SALT_LEN);
|
|
|
|
|
memcpy(buf + HASH_LEN + SALT_LEN, control->salt_pass, control->salt_pass_len);
|
|
|
|
|
sha4(buf, HASH_LEN + SALT_LEN + control->salt_pass_len, iv, 0);
|
|
|
|
|
|
|
|
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
|
munlock(buf, sizeof(buf));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void lrz_crypt(const rzip_control *control, uchar *buf, i64 len, const uchar *salt, int encrypt)
|
2011-03-17 04:10:58 +01:00
|
|
|
{
|
|
|
|
|
/* Encryption requires CBC_LEN blocks so we can use ciphertext
|
|
|
|
|
* stealing to not have to pad the block */
|
2011-03-21 10:54:53 +01:00
|
|
|
uchar key[HASH_LEN], iv[HASH_LEN];
|
2011-03-18 00:22:58 +01:00
|
|
|
uchar tmp0[CBC_LEN], tmp1[CBC_LEN];
|
|
|
|
|
aes_context aes_ctx;
|
2011-03-17 04:10:58 +01:00
|
|
|
i64 N, M;
|
2011-03-17 14:06:11 +01:00
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
/* Generate unique key and IV for each block of data based on salt */
|
2011-03-18 00:22:58 +01:00
|
|
|
mlock(&aes_ctx, sizeof(aes_ctx));
|
2011-03-21 10:54:53 +01:00
|
|
|
mlock(key, HASH_LEN);
|
|
|
|
|
mlock(iv, HASH_LEN);
|
|
|
|
|
|
|
|
|
|
lrz_keygen(control, salt, key, iv);
|
2011-03-17 04:10:58 +01:00
|
|
|
|
|
|
|
|
M = len % CBC_LEN;
|
|
|
|
|
N = len - M;
|
|
|
|
|
|
2011-03-19 04:04:22 +01:00
|
|
|
if (encrypt == LRZ_ENCRYPT) {
|
2011-03-17 04:10:58 +01:00
|
|
|
print_maxverbose("Encrypting data \n");
|
2011-03-18 00:22:58 +01:00
|
|
|
if (unlikely(aes_setkey_enc(&aes_ctx, key, 128)))
|
2011-03-17 14:06:11 +01:00
|
|
|
failure("Failed to aes_setkey_enc in lrz_crypt\n");
|
2011-03-18 00:22:58 +01:00
|
|
|
aes_crypt_cbc(&aes_ctx, AES_ENCRYPT, N, iv, buf, buf);
|
2011-03-17 04:10:58 +01:00
|
|
|
|
|
|
|
|
if (M) {
|
2011-03-18 00:22:58 +01:00
|
|
|
memset(tmp0, 0, CBC_LEN);
|
2011-03-17 04:10:58 +01:00
|
|
|
memcpy(tmp0, buf + N, M);
|
2011-03-18 00:22:58 +01:00
|
|
|
aes_crypt_cbc(&aes_ctx, AES_ENCRYPT, CBC_LEN,
|
|
|
|
|
iv, tmp0, tmp1);
|
2011-03-17 04:10:58 +01:00
|
|
|
memcpy(buf + N, buf + N - CBC_LEN, M);
|
|
|
|
|
memcpy(buf + N - CBC_LEN, tmp1, CBC_LEN);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2011-03-18 00:22:58 +01:00
|
|
|
if (unlikely(aes_setkey_dec(&aes_ctx, key, 128)))
|
2011-03-17 14:06:11 +01:00
|
|
|
failure("Failed to aes_setkey_dec in lrz_crypt\n");
|
2011-03-17 04:10:58 +01:00
|
|
|
print_maxverbose("Decrypting data \n");
|
|
|
|
|
if (M) {
|
2011-03-18 00:22:58 +01:00
|
|
|
aes_crypt_cbc(&aes_ctx, AES_DECRYPT, N - CBC_LEN,
|
|
|
|
|
iv, buf, buf);
|
|
|
|
|
aes_crypt_ecb(&aes_ctx, AES_DECRYPT,
|
2011-03-17 04:10:58 +01:00
|
|
|
buf + N - CBC_LEN, tmp0);
|
|
|
|
|
memset(tmp1, 0, CBC_LEN);
|
|
|
|
|
memcpy(tmp1, buf + N, M);
|
|
|
|
|
xor128(tmp0, tmp1);
|
|
|
|
|
memcpy(buf + N, tmp0, M);
|
|
|
|
|
memcpy(tmp1 + M, tmp0 + M, CBC_LEN - M);
|
2011-03-18 00:22:58 +01:00
|
|
|
aes_crypt_ecb(&aes_ctx, AES_DECRYPT, tmp1,
|
2011-03-17 04:10:58 +01:00
|
|
|
buf + N - CBC_LEN);
|
2011-03-18 00:22:58 +01:00
|
|
|
xor128(buf + N - CBC_LEN, iv);
|
2011-03-17 04:10:58 +01:00
|
|
|
} else
|
2011-03-18 00:22:58 +01:00
|
|
|
aes_crypt_cbc(&aes_ctx, AES_DECRYPT, len,
|
|
|
|
|
iv, buf, buf);
|
2011-03-17 04:10:58 +01:00
|
|
|
}
|
2011-03-17 14:06:11 +01:00
|
|
|
|
2011-03-18 00:22:58 +01:00
|
|
|
memset(&aes_ctx, 0, sizeof(aes_ctx));
|
2011-03-21 10:54:53 +01:00
|
|
|
memset(iv, 0, HASH_LEN);
|
|
|
|
|
memset(key, 0, HASH_LEN);
|
2011-03-18 00:22:58 +01:00
|
|
|
munlock(&aes_ctx, sizeof(aes_ctx));
|
2011-03-21 10:54:53 +01:00
|
|
|
munlock(iv, HASH_LEN);
|
|
|
|
|
munlock(key, HASH_LEN);
|
2011-03-18 00:22:58 +01:00
|
|
|
}
|
|
|
|
|
|
2011-03-21 10:54:53 +01:00
|
|
|
void lrz_stretch(rzip_control *control)
|
2011-03-19 04:04:22 +01:00
|
|
|
{
|
2011-03-21 10:54:53 +01:00
|
|
|
sha4_context ctx;
|
|
|
|
|
i64 j, n, counter;
|
|
|
|
|
|
|
|
|
|
mlock(&ctx, sizeof(ctx));
|
|
|
|
|
sha4_starts(&ctx, 0);
|
|
|
|
|
|
|
|
|
|
n = control->encloops * HASH_LEN / (control->salt_pass_len + sizeof(i64));
|
|
|
|
|
print_maxverbose("Hashing passphrase %lld (%lld) times \n", control->encloops, n);
|
|
|
|
|
for (j = 0; j < n; j ++) {
|
|
|
|
|
counter = htole64(j);
|
|
|
|
|
sha4_update(&ctx, (uchar *)&counter, sizeof(counter));
|
|
|
|
|
sha4_update(&ctx, control->salt_pass, control->salt_pass_len);
|
2011-03-18 00:22:58 +01:00
|
|
|
}
|
2011-03-21 10:54:53 +01:00
|
|
|
sha4_finish(&ctx, control->hash);
|
|
|
|
|
memset(&ctx, 0, sizeof(ctx));
|
|
|
|
|
munlock(&ctx, sizeof(ctx));
|
2011-03-18 00:22:58 +01:00
|
|
|
}
|