cryptsetup.c revision 8cf3ca80680b43015971cbbf4625517ae859d50c
/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
/***
This file is part of systemd.
Copyright 2010 Lennart Poettering
under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation; either version 2.1 of the License, or
(at your option) any later version.
systemd 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
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with systemd; If not, see <http://www.gnu.org/licenses/>.
***/
#include <string.h>
#include <errno.h>
#include <mntent.h>
#include <libcryptsetup.h>
#include <libudev.h>
#include "fileio.h"
#include "log.h"
#include "util.h"
#include "path-util.h"
#include "strv.h"
#include "ask-password-api.h"
#include "def.h"
static char *opt_cipher = NULL;
static unsigned opt_key_size = 0;
static unsigned opt_keyfile_size = 0;
static unsigned opt_keyfile_offset = 0;
static unsigned opt_tries = 0;
static bool opt_readonly = false;
static bool opt_verify = false;
static bool opt_discards = false;
static bool opt_tcrypt_hidden = false;
static bool opt_tcrypt_system = false;
static char **opt_tcrypt_keyfiles = NULL;
static usec_t opt_timeout = 0;
/* Options Debian's crypttab knows we don't:
offset=
skip=
precheck=
check=
checkargs=
noearly=
loud=
keyscript=
*/
static int parse_one_option(const char *option) {
/* Handled outside of this tool */
return 0;
char *t;
if (!t)
return -ENOMEM;
opt_cipher = t;
log_error("size= parse failure, ignoring.");
return 0;
}
else
log_error("keyfile-size= parse failure, ignoring.");
return 0;
}
log_error("keyfile-offset= parse failure, ignoring.");
return 0;
}
char *t;
if (!t)
return -ENOMEM;
opt_hash = t;
log_error("tries= parse failure, ignoring.");
return 0;
}
opt_readonly = true;
opt_verify = true;
opt_discards = true;
opt_tcrypt_hidden = true;
opt_tcrypt_system = true;
log_error("timeout= parse failure, ignoring.");
return 0;
}
return 0;
}
static int parse_options(const char *options) {
char *state, *w;
size_t l;
int r;
_cleanup_free_ char *o;
o = strndup(w, l);
if (!o)
return -ENOMEM;
r = parse_one_option(o);
if (r < 0)
return r;
}
return 0;
}
}
static char *disk_description(const char *path) {
static const char name_fields[] = {
"ID_PART_ENTRY_NAME\0"
"DM_NAME\0"
"ID_MODEL_FROM_DATABASE\0"
"ID_MODEL\0"
};
char *description = NULL;
const char *i;
return NULL;
return NULL;
if (!udev)
return NULL;
if (!device)
goto finish;
NULSTR_FOREACH(i, name_fields) {
const char *name;
break;
}
}
if (device)
if (udev)
return description;
}
static char *disk_mount_point(const char *label) {
struct mntent *m;
/* Yeah, we don't support native systemd unit files here for now */
goto finish;
if (!f)
goto finish;
while ((m = getmntent(f)))
break;
}
if (f)
endmntent(f);
return mp;
}
int r;
char **p;
return log_oom();
if (r < 0) {
return r;
}
if (opt_verify) {
return log_oom();
if (r < 0) {
return r;
}
log_warning("Passwords did not match, retrying.");
return -EAGAIN;
}
}
STRV_FOREACH(p, *passwords) {
char *c;
continue;
/* Pad password if necessary */
if (!(c = new(char, opt_key_size)))
return log_oom();
strncpy(c, *p, opt_key_size);
free(*p);
*p = c;
}
return 0;
}
const char *name,
const char *key_file,
char **passwords,
int r = 0;
struct crypt_params_tcrypt params = {
.keyfiles = (const char **)opt_tcrypt_keyfiles,
};
if (opt_tcrypt_hidden)
if (opt_tcrypt_system)
if (key_file) {
if (r < 0) {
return -EAGAIN;
}
} else
if (r < 0) {
return -EAGAIN;
}
return r;
}
}
const char *name,
const char *key_file,
char **passwords,
int r = 0;
bool pass_volume_key = false;
struct crypt_params_plain params = {};
const char *cipher, *cipher_mode;
if (opt_hash) {
/* plain isn't a real hash type. it just means "use no hash" */
} else
if (opt_cipher) {
size_t l;
if (!truncated_cipher)
return log_oom();
} else {
cipher = "aes";
cipher_mode = "cbc-essiv:sha256";
}
/* for CRYPT_PLAIN limit reads
* from keyfile to key length, and
* ignore keyfile-size */
/* In contrast to what the name
* crypt_setup() might suggest this
* doesn't actually format anything,
* it just configures encryption
* parameters when used for plain
* mode. */
/* hash == NULL implies the user passed "plain" */
}
if (r < 0) {
return r;
}
log_info("Set cipher %s, mode %s, key size %i bits for device %s.",
if (key_file) {
if (r < 0) {
return -EAGAIN;
}
} else {
char **p;
STRV_FOREACH(p, passwords) {
if (pass_volume_key)
else
if (r >= 0)
break;
}
}
return r;
}
static int help(void) {
printf("%s attach VOLUME SOURCEDEVICE [PASSWORD] [OPTIONS]\n"
"%s detach VOLUME\n\n"
"Attaches or detaches an encrypted block device.\n",
return 0;
}
int r = EXIT_FAILURE;
if (argc <= 1) {
help();
return EXIT_SUCCESS;
}
if (argc < 3) {
log_error("This program requires at least two arguments.");
return EXIT_FAILURE;
}
log_open();
umask(0022);
int k;
unsigned tries;
/* Arguments: systemd-cryptsetup attach VOLUME SOURCE-DEVICE [PASSWORD] [OPTIONS] */
if (argc < 4) {
log_error("attach requires at least two arguments.");
goto finish;
}
if (argc >= 5 &&
argv[4][0] &&
else
}
goto finish;
}
/* A delicious drop of snake oil */
/* If the description string is simply the
* volume name, then let's not show this
* twice */
description = NULL;
}
if (mount_point && description)
else if (mount_point)
else if (description)
if (k) {
goto finish;
}
r = EXIT_SUCCESS;
goto finish;
}
if (opt_readonly)
if (opt_discards)
if (opt_timeout > 0)
else
until = 0;
if (key_file) {
/* Ideally we'd do this on the open fd, but since this is just a
* warning it's OK to do this in two steps. */
}
if (!key_file) {
if (k == -EAGAIN)
continue;
else if (k < 0)
goto finish;
}
else
if (k >= 0)
break;
else if (k == -EAGAIN) {
continue;
} else if (k != -EPERM) {
goto finish;
}
log_warning("Invalid passphrase.");
}
log_error("Too many attempts; giving up.");
r = EXIT_FAILURE;
goto finish;
}
int k;
if (k) {
goto finish;
}
if (k < 0) {
goto finish;
}
} else {
goto finish;
}
r = EXIT_SUCCESS;
if (cd)
crypt_free(cd);
return r;
}