| /* |
| * QEMU Crypto anti forensic information splitter |
| * |
| * Copyright (c) 2015-2016 Red Hat, Inc. |
| * |
| * 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. |
| * |
| * 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. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, see <http://www.gnu.org/licenses/>. |
| */ |
| |
| #ifndef QCRYPTO_AFSPLIT_H |
| #define QCRYPTO_AFSPLIT_H |
| |
| #include "crypto/hash.h" |
| |
| /** |
| * This module implements the anti-forensic splitter that is specified |
| * as part of the LUKS format: |
| * |
| * http://clemens.endorphin.org/cryptography |
| * http://clemens.endorphin.org/TKS1-draft.pdf |
| * |
| * The core idea is to take a short piece of data (key material) |
| * and process it to expand it to a much larger piece of data. |
| * The expansion process is reversible, to obtain the original |
| * short data. The key property of the expansion is that if any |
| * byte in the larger data set is changed / missing, it should be |
| * impossible to recreate the original short data. |
| * |
| * <example> |
| * <title>Creating a large split key for storage</title> |
| * <programlisting> |
| * size_t nkey = 32; |
| * uint32_t stripes = 32768; // To produce a 1 MB split key |
| * uint8_t *masterkey = ....a 32-byte AES key... |
| * uint8_t *splitkey; |
| * |
| * splitkey = g_new0(uint8_t, nkey * stripes); |
| * |
| * if (qcrypto_afsplit_encode(QCRYPTO_HASH_ALGO_SHA256, |
| * nkey, stripes, |
| * masterkey, splitkey, errp) < 0) { |
| * g_free(splitkey); |
| * g_free(masterkey); |
| * return -1; |
| * } |
| * |
| * ...store splitkey somewhere... |
| * |
| * g_free(splitkey); |
| * g_free(masterkey); |
| * </programlisting> |
| * </example> |
| * |
| * <example> |
| * <title>Retrieving a master key from storage</title> |
| * <programlisting> |
| * size_t nkey = 32; |
| * uint32_t stripes = 32768; // To produce a 1 MB split key |
| * uint8_t *masterkey; |
| * uint8_t *splitkey = .... read in 1 MB of data... |
| * |
| * masterkey = g_new0(uint8_t, nkey); |
| * |
| * if (qcrypto_afsplit_decode(QCRYPTO_HASH_ALGO_SHA256, |
| * nkey, stripes, |
| * splitkey, masterkey, errp) < 0) { |
| * g_free(splitkey); |
| * g_free(masterkey); |
| * return -1; |
| * } |
| * |
| * ..decrypt data with masterkey... |
| * |
| * g_free(splitkey); |
| * g_free(masterkey); |
| * </programlisting> |
| * </example> |
| */ |
| |
| /** |
| * qcrypto_afsplit_encode: |
| * @hash: the hash algorithm to use for data expansion |
| * @blocklen: the size of @in in bytes |
| * @stripes: the number of times to expand @in in size |
| * @in: the master key to be expanded in size |
| * @out: preallocated buffer to hold the split key |
| * @errp: pointer to a NULL-initialized error object |
| * |
| * Split the data in @in, which is @blocklen bytes in |
| * size, to form a larger piece of data @out, which is |
| * @blocklen * @stripes bytes in size. |
| * |
| * Returns: 0 on success, -1 on error; |
| */ |
| int qcrypto_afsplit_encode(QCryptoHashAlgo hash, |
| size_t blocklen, |
| uint32_t stripes, |
| const uint8_t *in, |
| uint8_t *out, |
| Error **errp); |
| |
| /** |
| * qcrypto_afsplit_decode: |
| * @hash: the hash algorithm to use for data compression |
| * @blocklen: the size of @out in bytes |
| * @stripes: the number of times to decrease @in in size |
| * @in: the split key to be recombined |
| * @out: preallocated buffer to hold the master key |
| * @errp: pointer to a NULL-initialized error object |
| * |
| * Join the data in @in, which is @blocklen * @stripes |
| * bytes in size, to form the original small piece of |
| * data @out, which is @blocklen bytes in size. |
| * |
| * Returns: 0 on success, -1 on error; |
| */ |
| int qcrypto_afsplit_decode(QCryptoHashAlgo hash, |
| size_t blocklen, |
| uint32_t stripes, |
| const uint8_t *in, |
| uint8_t *out, |
| Error **errp); |
| |
| #endif /* QCRYPTO_AFSPLIT_H */ |