2007-10-07 11:44:02 +00:00
|
|
|
/*
|
|
|
|
* salt.c - generate a random salt string for crypt()
|
|
|
|
*
|
|
|
|
* Written by Marek Michalkiewicz <marekm@i17linuxb.ists.pwr.wroc.pl>,
|
|
|
|
* public domain.
|
2007-11-16 12:36:21 +00:00
|
|
|
*
|
|
|
|
* l64a was Written by J.T. Conklin <jtc@netbsd.org>. Public domain.
|
2007-10-07 11:44:02 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
2007-11-10 23:46:11 +00:00
|
|
|
#ident "$Id$"
|
2007-10-07 11:47:01 +00:00
|
|
|
|
2007-10-07 11:46:07 +00:00
|
|
|
#include <sys/time.h>
|
|
|
|
#include <stdlib.h>
|
2007-11-23 20:51:43 +00:00
|
|
|
#include <assert.h>
|
2007-10-07 11:44:02 +00:00
|
|
|
#include "prototypes.h"
|
|
|
|
#include "defines.h"
|
|
|
|
#include "getdef.h"
|
2007-11-16 12:36:21 +00:00
|
|
|
|
2008-01-06 14:50:26 +00:00
|
|
|
/* local function prototypes */
|
2007-11-16 12:36:21 +00:00
|
|
|
#ifndef HAVE_L64A
|
2008-01-06 14:50:26 +00:00
|
|
|
char *l64a(long value);
|
|
|
|
#endif
|
|
|
|
static char *gensalt (unsigned int salt_size);
|
|
|
|
#ifdef USE_SHA_CRYPT
|
|
|
|
static unsigned int SHA_salt_size (void);
|
|
|
|
static const char *SHA_salt_rounds (int *prefered_rounds);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef HAVE_L64A
|
|
|
|
static char *l64a(long value)
|
2007-11-16 12:36:21 +00:00
|
|
|
{
|
|
|
|
static char buf[8];
|
|
|
|
char *s = buf;
|
|
|
|
int digit;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (value < 0) {
|
|
|
|
errno = EINVAL;
|
|
|
|
return(NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; value != 0 && i < 6; i++) {
|
|
|
|
digit = value & 0x3f;
|
|
|
|
|
2007-11-24 00:00:12 +00:00
|
|
|
if (digit < 2)
|
2007-11-16 12:36:21 +00:00
|
|
|
*s = digit + '.';
|
|
|
|
else if (digit < 12)
|
|
|
|
*s = digit + '0' - 2;
|
|
|
|
else if (digit < 38)
|
|
|
|
*s = digit + 'A' - 12;
|
|
|
|
else
|
|
|
|
*s = digit + 'a' - 38;
|
|
|
|
|
|
|
|
value >>= 6;
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
|
|
|
|
*s = '\0';
|
|
|
|
|
|
|
|
return(buf);
|
|
|
|
}
|
|
|
|
#endif /* !HAVE_L64A */
|
|
|
|
|
2007-10-07 11:44:02 +00:00
|
|
|
/*
|
2007-11-20 00:05:54 +00:00
|
|
|
* Add the salt prefix.
|
2007-10-07 11:44:02 +00:00
|
|
|
*/
|
2007-11-23 20:51:43 +00:00
|
|
|
#define MAGNUM(array,ch) (array)[0]=(array)[2]='$',(array)[1]=(ch),(array)[3]='\0'
|
2007-11-19 22:14:19 +00:00
|
|
|
|
* configure.in: New configure option: --with-sha-crypt enabled by
default. Keeping the feature enabled is safe. Disabling it permits
to disable the references to the SHA256 and SHA512 password
encryption algorithms from the usage help and manuals (in addition
to the support for these algorithms in the code).
* libmisc/obscure.c, libmisc/salt.c, src/newusers.c,
src/chpasswd.c, src/chgpasswd.c, src/passwd.c: ENCRYPT_METHOD is
always supported in login.defs. Remove the ENCRYPTMETHOD_SELECT
preprocessor condition.
* libmisc/obscure.c, libmisc/salt.c, src/newusers.c,
src/chpasswd.c, src/chgpasswd.c, src/passwd.c: Disable SHA256 and
SHA512 if USE_SHA_CRYPT is not defined (this corresponds to a
subset of the ENCRYPTMETHOD_SELECT sections).
2007-11-24 13:08:08 +00:00
|
|
|
#ifdef USE_SHA_CRYPT
|
2007-11-20 00:05:54 +00:00
|
|
|
/*
|
|
|
|
* Return the salt size.
|
|
|
|
* The size of the salt string is between 8 and 16 bytes for the SHA crypt
|
|
|
|
* methods.
|
|
|
|
*/
|
|
|
|
static unsigned int SHA_salt_size (void)
|
|
|
|
{
|
2008-01-06 14:50:26 +00:00
|
|
|
double rand_rounds = 9 * random ();
|
|
|
|
rand_rounds /= RAND_MAX;
|
|
|
|
return 8 + rand_rounds;
|
2007-11-20 00:05:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* ! Arguments evaluated twice ! */
|
|
|
|
#define MAX(x,y) ((x) > (y) ? (x) : (y))
|
|
|
|
#define MIN(x,y) ((x) < (y) ? (x) : (y))
|
|
|
|
|
|
|
|
/* Default number of rounds if not explicitly specified. */
|
|
|
|
#define ROUNDS_DEFAULT 5000
|
|
|
|
/* Minimum number of rounds. */
|
|
|
|
#define ROUNDS_MIN 1000
|
|
|
|
/* Maximum number of rounds. */
|
|
|
|
#define ROUNDS_MAX 999999999
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return a salt prefix specifying the rounds number for the SHA crypt methods.
|
|
|
|
*/
|
2008-01-06 14:50:26 +00:00
|
|
|
static const char *SHA_salt_rounds (int *prefered_rounds)
|
2007-11-20 00:05:54 +00:00
|
|
|
{
|
2007-11-20 09:33:52 +00:00
|
|
|
static char rounds_prefix[18];
|
2007-11-20 00:05:54 +00:00
|
|
|
long rounds;
|
|
|
|
|
2007-11-20 09:33:52 +00:00
|
|
|
if (NULL == prefered_rounds) {
|
|
|
|
long min_rounds = getdef_long ("SHA_CRYPT_MIN_ROUNDS", -1);
|
|
|
|
long max_rounds = getdef_long ("SHA_CRYPT_MAX_ROUNDS", -1);
|
2008-01-06 14:50:26 +00:00
|
|
|
double rand_rounds;
|
2007-11-20 00:05:54 +00:00
|
|
|
|
2007-11-20 09:33:52 +00:00
|
|
|
if (-1 == min_rounds && -1 == max_rounds)
|
|
|
|
return "";
|
2007-11-20 00:05:54 +00:00
|
|
|
|
2007-11-20 09:33:52 +00:00
|
|
|
if (-1 == min_rounds)
|
|
|
|
min_rounds = max_rounds;
|
2007-11-20 00:05:54 +00:00
|
|
|
|
2007-11-20 09:33:52 +00:00
|
|
|
if (-1 == max_rounds)
|
|
|
|
max_rounds = min_rounds;
|
2007-11-20 00:05:54 +00:00
|
|
|
|
2007-11-20 09:33:52 +00:00
|
|
|
if (min_rounds > max_rounds)
|
|
|
|
max_rounds = min_rounds;
|
|
|
|
|
|
|
|
srand (time (NULL));
|
2008-01-06 14:50:26 +00:00
|
|
|
rand_rounds = (max_rounds-min_rounds+1) * random ();
|
|
|
|
rand_rounds /= RAND_MAX;
|
|
|
|
rounds = min_rounds + rand_rounds;
|
2007-11-20 09:33:52 +00:00
|
|
|
} else if (0 == *prefered_rounds)
|
|
|
|
return "";
|
2007-11-20 20:00:16 +00:00
|
|
|
else
|
|
|
|
rounds = *prefered_rounds;
|
2007-11-20 00:05:54 +00:00
|
|
|
|
|
|
|
/* Sanity checks. The libc should also check this, but this
|
|
|
|
* protects against a rounds_prefix overflow. */
|
|
|
|
if (rounds < ROUNDS_MIN)
|
|
|
|
rounds = ROUNDS_MIN;
|
|
|
|
|
|
|
|
if (rounds > ROUNDS_MAX)
|
|
|
|
rounds = ROUNDS_MAX;
|
|
|
|
|
|
|
|
snprintf (rounds_prefix, 18, "rounds=%ld$", rounds);
|
|
|
|
|
|
|
|
/* Sanity checks. That should not be necessary. */
|
|
|
|
rounds_prefix[17] = '\0';
|
|
|
|
if ('$' != rounds_prefix[16])
|
|
|
|
rounds_prefix[17] = '$';
|
|
|
|
|
|
|
|
return rounds_prefix;
|
|
|
|
}
|
2007-11-20 09:20:34 +00:00
|
|
|
#endif
|
2007-11-20 00:05:54 +00:00
|
|
|
|
2007-11-23 20:51:43 +00:00
|
|
|
/*
|
|
|
|
* Generate salt of size salt_size.
|
|
|
|
*/
|
|
|
|
#define MAX_SALT_SIZE 16
|
|
|
|
#define MIN_SALT_SIZE 8
|
|
|
|
|
2008-01-06 14:50:26 +00:00
|
|
|
static char *gensalt (unsigned int salt_size)
|
|
|
|
{
|
2007-11-24 00:00:12 +00:00
|
|
|
static char salt[32];
|
|
|
|
|
|
|
|
salt[0] = '\0';
|
|
|
|
|
2007-11-23 21:04:43 +00:00
|
|
|
assert (salt_size >= MIN_SALT_SIZE &&
|
|
|
|
salt_size <= MAX_SALT_SIZE);
|
|
|
|
srandom ((unsigned int)time(NULL));
|
2007-11-24 00:00:12 +00:00
|
|
|
strcat (salt, l64a (random()));
|
|
|
|
do {
|
|
|
|
strcat (salt, l64a (random()));
|
|
|
|
} while (strlen (salt) < salt_size);
|
|
|
|
salt[salt_size] = '\0';
|
|
|
|
|
|
|
|
return salt;
|
2007-11-23 20:51:43 +00:00
|
|
|
}
|
|
|
|
|
2007-11-20 00:05:54 +00:00
|
|
|
/*
|
|
|
|
* Generate 8 base64 ASCII characters of random salt. If MD5_CRYPT_ENAB
|
|
|
|
* in /etc/login.defs is "yes", the salt string will be prefixed by "$1$"
|
|
|
|
* (magic) and pw_encrypt() will execute the MD5-based FreeBSD-compatible
|
|
|
|
* version of crypt() instead of the standard one.
|
|
|
|
* Other methods can be set with ENCRYPT_METHOD
|
2007-11-20 09:33:52 +00:00
|
|
|
*
|
|
|
|
* The method can be forced with the meth parameter.
|
|
|
|
* If NULL, the method will be defined according to the MD5_CRYPT_ENAB and
|
|
|
|
* ENCRYPT_METHOD login.defs variables.
|
|
|
|
*
|
|
|
|
* If meth is specified, an additional parameter can be provided.
|
|
|
|
* * For the SHA256 and SHA512 method, this specifies the number of rounds
|
|
|
|
* (if not NULL).
|
2007-11-20 00:05:54 +00:00
|
|
|
*/
|
2008-01-06 14:50:26 +00:00
|
|
|
char *crypt_make_salt (const char *meth, void *arg)
|
2007-10-07 11:44:02 +00:00
|
|
|
{
|
2007-11-20 00:05:54 +00:00
|
|
|
/* Max result size for the SHA methods:
|
|
|
|
* +3 $5$
|
|
|
|
* +17 rounds=999999999$
|
|
|
|
* +16 salt
|
|
|
|
* +1 \0
|
|
|
|
*/
|
2007-10-07 11:44:02 +00:00
|
|
|
static char result[40];
|
2007-11-23 20:51:43 +00:00
|
|
|
size_t salt_len = 8;
|
2008-01-06 14:50:26 +00:00
|
|
|
const char *method;
|
2007-10-07 11:44:02 +00:00
|
|
|
|
2007-11-19 22:34:48 +00:00
|
|
|
result[0] = '\0';
|
|
|
|
|
2007-11-20 09:33:52 +00:00
|
|
|
if (NULL != meth)
|
|
|
|
method = meth;
|
2007-11-23 23:57:47 +00:00
|
|
|
else {
|
2007-11-20 09:33:52 +00:00
|
|
|
if ((method = getdef_str ("ENCRYPT_METHOD")) == NULL)
|
2007-11-23 23:57:47 +00:00
|
|
|
method = getdef_bool ("MD5_CRYPT_ENAB") ? "MD5" : "DES";
|
|
|
|
}
|
2007-11-20 09:33:52 +00:00
|
|
|
|
2007-11-20 20:00:16 +00:00
|
|
|
if (!strcmp (method, "MD5")) {
|
2007-11-20 09:33:52 +00:00
|
|
|
MAGNUM(result, '1');
|
* configure.in: New configure option: --with-sha-crypt enabled by
default. Keeping the feature enabled is safe. Disabling it permits
to disable the references to the SHA256 and SHA512 password
encryption algorithms from the usage help and manuals (in addition
to the support for these algorithms in the code).
* libmisc/obscure.c, libmisc/salt.c, src/newusers.c,
src/chpasswd.c, src/chgpasswd.c, src/passwd.c: ENCRYPT_METHOD is
always supported in login.defs. Remove the ENCRYPTMETHOD_SELECT
preprocessor condition.
* libmisc/obscure.c, libmisc/salt.c, src/newusers.c,
src/chpasswd.c, src/chgpasswd.c, src/passwd.c: Disable SHA256 and
SHA512 if USE_SHA_CRYPT is not defined (this corresponds to a
subset of the ENCRYPTMETHOD_SELECT sections).
2007-11-24 13:08:08 +00:00
|
|
|
#ifdef USE_SHA_CRYPT
|
2007-11-20 20:00:16 +00:00
|
|
|
} else if (!strcmp (method, "SHA256")) {
|
2007-11-20 09:33:52 +00:00
|
|
|
MAGNUM(result, '5');
|
|
|
|
strcat(result, SHA_salt_rounds((int *)arg));
|
2007-11-23 20:51:43 +00:00
|
|
|
salt_len = SHA_salt_size();
|
2007-11-20 20:00:16 +00:00
|
|
|
} else if (!strcmp (method, "SHA512")) {
|
2007-11-20 09:33:52 +00:00
|
|
|
MAGNUM(result, '6');
|
|
|
|
strcat(result, SHA_salt_rounds((int *)arg));
|
2007-11-23 20:51:43 +00:00
|
|
|
salt_len = SHA_salt_size();
|
2007-11-20 09:33:52 +00:00
|
|
|
#endif
|
2007-11-20 20:00:16 +00:00
|
|
|
} else if (0 != strcmp (method, "DES")) {
|
2007-11-20 09:33:52 +00:00
|
|
|
fprintf (stderr,
|
|
|
|
_("Invalid ENCRYPT_METHOD value: '%s'.\n"
|
|
|
|
"Defaulting to DES.\n"),
|
|
|
|
method);
|
|
|
|
result[0] = '\0';
|
2007-11-19 22:14:19 +00:00
|
|
|
}
|
2007-11-20 00:05:54 +00:00
|
|
|
|
2007-10-07 11:44:02 +00:00
|
|
|
/*
|
2007-11-20 00:05:54 +00:00
|
|
|
* Concatenate a pseudo random salt.
|
2007-10-07 11:44:02 +00:00
|
|
|
*/
|
2007-11-23 20:51:43 +00:00
|
|
|
assert (sizeof (result) > strlen (result) + salt_len);
|
|
|
|
strncat (result, gensalt (salt_len),
|
|
|
|
sizeof (result) - strlen (result) - 1);
|
2007-10-07 11:44:02 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2008-01-06 14:50:26 +00:00
|
|
|
|