mirror of
https://github.com/hashcat/hashcat.git
synced 2025-01-08 23:01:14 +00:00
505 lines
13 KiB
C
505 lines
13 KiB
C
/**
|
|
* Author......: See docs/credits.txt
|
|
* License.....: MIT
|
|
*/
|
|
|
|
#include "common.h"
|
|
#include "types.h"
|
|
#include "event.h"
|
|
#include "memory.h"
|
|
#include "filehandling.h"
|
|
#include "folder.h"
|
|
#include "shared.h"
|
|
#include "tuningdb.h"
|
|
|
|
int sort_by_tuning_db_alias (const void *v1, const void *v2)
|
|
{
|
|
const tuning_db_alias_t *t1 = (const tuning_db_alias_t *) v1;
|
|
const tuning_db_alias_t *t2 = (const tuning_db_alias_t *) v2;
|
|
|
|
const int res1 = strcmp (t1->device_name, t2->device_name);
|
|
|
|
if (res1 != 0) return (res1);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int sort_by_tuning_db_entry (const void *v1, const void *v2)
|
|
{
|
|
const tuning_db_entry_t *t1 = (const tuning_db_entry_t *) v1;
|
|
const tuning_db_entry_t *t2 = (const tuning_db_entry_t *) v2;
|
|
|
|
const int res1 = strcmp (t1->device_name, t2->device_name);
|
|
|
|
if (res1 != 0) return (res1);
|
|
|
|
const int res2 = t1->attack_mode
|
|
- t2->attack_mode;
|
|
|
|
if (res2 != 0) return (res2);
|
|
|
|
const int res3 = t1->hash_mode
|
|
- t2->hash_mode;
|
|
|
|
if (res3 != 0) return (res3);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int tuning_db_init (hashcat_ctx_t *hashcat_ctx)
|
|
{
|
|
tuning_db_t *tuning_db = hashcat_ctx->tuning_db;
|
|
user_options_t *user_options = hashcat_ctx->user_options;
|
|
folder_config_t *folder_config = hashcat_ctx->folder_config;
|
|
|
|
tuning_db->enabled = false;
|
|
|
|
if (user_options->hash_info == true) return 0;
|
|
if (user_options->keyspace == true) return 0;
|
|
if (user_options->left == true) return 0;
|
|
if (user_options->show == true) return 0;
|
|
if (user_options->usage == true) return 0;
|
|
if (user_options->version == true) return 0;
|
|
if (user_options->identify == true) return 0;
|
|
if (user_options->backend_info > 0) return 0;
|
|
|
|
tuning_db->enabled = true;
|
|
|
|
char *tuning_db_folder = NULL;
|
|
|
|
hc_asprintf (&tuning_db_folder, "%s/tunings", folder_config->shared_dir);
|
|
|
|
char **tuning_db_files = scan_directory (tuning_db_folder);
|
|
|
|
for (int i = 0; tuning_db_files[i] != NULL; i++)
|
|
{
|
|
char *tuning_db_file = tuning_db_files[i];
|
|
|
|
const size_t suflen = strlen (TUNING_DB_SUFFIX);
|
|
|
|
const size_t dblen = strlen (tuning_db_file);
|
|
|
|
if (dblen < suflen) continue; // make sure to not do any out-of-boundary reads
|
|
|
|
if (memcmp (tuning_db_file + dblen - suflen, TUNING_DB_SUFFIX, suflen) != 0) continue;
|
|
|
|
HCFILE fp;
|
|
|
|
if (hc_fopen (&fp, tuning_db_file, "rb") == false)
|
|
{
|
|
event_log_error (hashcat_ctx, "%s: %s", tuning_db_file, strerror (errno));
|
|
|
|
return -1;
|
|
}
|
|
|
|
hcfree (tuning_db_file);
|
|
|
|
int line_num = 0;
|
|
|
|
char *buf = (char *) hcmalloc (HCBUFSIZ_LARGE);
|
|
|
|
while (!hc_feof (&fp))
|
|
{
|
|
char *line_buf = hc_fgets (buf, HCBUFSIZ_LARGE - 1, &fp);
|
|
|
|
if (line_buf == NULL) break;
|
|
|
|
line_num++;
|
|
|
|
const size_t line_len = in_superchop (line_buf);
|
|
|
|
if (line_len == 0) continue;
|
|
|
|
if (line_buf[0] == '#') continue;
|
|
|
|
tuning_db_process_line (hashcat_ctx, line_buf, line_num);
|
|
}
|
|
|
|
hcfree (buf);
|
|
|
|
hc_fclose (&fp);
|
|
}
|
|
|
|
hcfree (tuning_db_files);
|
|
|
|
// todo: print loaded 'cnt' message
|
|
|
|
// sort the database
|
|
|
|
qsort (tuning_db->alias_buf, tuning_db->alias_cnt, sizeof (tuning_db_alias_t), sort_by_tuning_db_alias);
|
|
qsort (tuning_db->entry_buf, tuning_db->entry_cnt, sizeof (tuning_db_entry_t), sort_by_tuning_db_entry);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void tuning_db_destroy (hashcat_ctx_t *hashcat_ctx)
|
|
{
|
|
tuning_db_t *tuning_db = hashcat_ctx->tuning_db;
|
|
|
|
if (tuning_db->enabled == false) return;
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < tuning_db->alias_cnt; i++)
|
|
{
|
|
tuning_db_alias_t *alias = &tuning_db->alias_buf[i];
|
|
|
|
hcfree (alias->device_name);
|
|
hcfree (alias->alias_name);
|
|
}
|
|
|
|
for (i = 0; i < tuning_db->entry_cnt; i++)
|
|
{
|
|
tuning_db_entry_t *entry = &tuning_db->entry_buf[i];
|
|
|
|
hcfree ((void *)entry->device_name);
|
|
}
|
|
|
|
hcfree (tuning_db->alias_buf);
|
|
hcfree (tuning_db->entry_buf);
|
|
|
|
memset (tuning_db, 0, sizeof (tuning_db_t));
|
|
}
|
|
|
|
bool tuning_db_process_line (hashcat_ctx_t *hashcat_ctx, const char *line_buf, const int line_num)
|
|
{
|
|
tuning_db_t *tuning_db = hashcat_ctx->tuning_db;
|
|
user_options_extra_t *user_options_extra = hashcat_ctx->user_options_extra;
|
|
|
|
#define ADD_DB_ENTRIES 1
|
|
|
|
if (tuning_db->alias_cnt == tuning_db->alias_alloc)
|
|
{
|
|
tuning_db->alias_buf = (tuning_db_alias_t *) hcrealloc (tuning_db->alias_buf, tuning_db->alias_alloc * sizeof (tuning_db_alias_t), ADD_DB_ENTRIES * sizeof (tuning_db_alias_t));
|
|
tuning_db->alias_alloc += ADD_DB_ENTRIES;
|
|
}
|
|
|
|
if (tuning_db->entry_cnt == tuning_db->entry_alloc)
|
|
{
|
|
tuning_db->entry_buf = (tuning_db_entry_t *) hcrealloc (tuning_db->entry_buf, tuning_db->entry_alloc * sizeof (tuning_db_entry_t), ADD_DB_ENTRIES * sizeof (tuning_db_entry_t));
|
|
tuning_db->entry_alloc += ADD_DB_ENTRIES;
|
|
}
|
|
|
|
char *buf = hcstrdup (line_buf);
|
|
|
|
char *token_ptr[7] = { NULL };
|
|
|
|
int token_cnt = 0;
|
|
|
|
char *saveptr = NULL;
|
|
|
|
char *next = strtok_r (buf, "\t ", &saveptr);
|
|
|
|
token_ptr[token_cnt] = next;
|
|
|
|
token_cnt++;
|
|
|
|
while ((next = strtok_r ((char *) NULL, "\t ", &saveptr)) != NULL)
|
|
{
|
|
token_ptr[token_cnt] = next;
|
|
|
|
token_cnt++;
|
|
}
|
|
|
|
if (token_cnt == 2)
|
|
{
|
|
char *device_name = token_ptr[0];
|
|
char *alias_name = token_ptr[1];
|
|
|
|
tuning_db_alias_t *alias = &tuning_db->alias_buf[tuning_db->alias_cnt];
|
|
|
|
alias->device_name = hcstrdup (device_name);
|
|
alias->alias_name = hcstrdup (alias_name);
|
|
|
|
tuning_db->alias_cnt++;
|
|
}
|
|
else if (token_cnt == 6)
|
|
{
|
|
if ((token_ptr[1][0] != '0') &&
|
|
(token_ptr[1][0] != '1') &&
|
|
(token_ptr[1][0] != '3') &&
|
|
(token_ptr[1][0] != '9') &&
|
|
(token_ptr[1][0] != '*'))
|
|
{
|
|
event_log_warning (hashcat_ctx, "Tuning-db: Invalid attack_mode '%c' in Line '%d'", token_ptr[1][0], line_num);
|
|
|
|
hcfree (buf);
|
|
|
|
return false;
|
|
}
|
|
|
|
if ((token_ptr[3][0] != '1') &&
|
|
(token_ptr[3][0] != '2') &&
|
|
(token_ptr[3][0] != '4') &&
|
|
(token_ptr[3][0] != '8') &&
|
|
(token_ptr[3][0] != 'N'))
|
|
{
|
|
event_log_warning (hashcat_ctx, "Tuning-db: Invalid vector_width '%c' in Line '%d'", token_ptr[3][0], line_num);
|
|
|
|
hcfree (buf);
|
|
|
|
return false;
|
|
}
|
|
|
|
char *device_name = token_ptr[0];
|
|
|
|
int hash_mode = -1;
|
|
int attack_mode = -1;
|
|
int vector_width = -1;
|
|
int kernel_accel = -1;
|
|
int kernel_loops = -1;
|
|
|
|
if (token_ptr[1][0] != '*') attack_mode = (int) strtol (token_ptr[1], NULL, 10);
|
|
if (token_ptr[2][0] != '*') hash_mode = (int) strtol (token_ptr[2], NULL, 10);
|
|
if (token_ptr[3][0] != 'N') vector_width = (int) strtol (token_ptr[3], NULL, 10);
|
|
|
|
if (token_ptr[4][0] == 'A')
|
|
{
|
|
kernel_accel = 0;
|
|
}
|
|
else if (token_ptr[4][0] == 'M')
|
|
{
|
|
kernel_accel = 1024;
|
|
}
|
|
else if (token_ptr[4][0] == 'N')
|
|
{
|
|
kernel_accel = -1;
|
|
}
|
|
else
|
|
{
|
|
kernel_accel = (int) strtol (token_ptr[4], NULL, 10);
|
|
|
|
if ((kernel_accel < 1) || (kernel_accel > 1024))
|
|
{
|
|
event_log_warning (hashcat_ctx, "Tuning-db: Invalid kernel_accel '%d' in Line '%d'", kernel_accel, line_num);
|
|
|
|
hcfree (buf);
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (token_ptr[5][0] == 'A')
|
|
{
|
|
kernel_loops = 0;
|
|
}
|
|
else if (token_ptr[5][0] == 'M')
|
|
{
|
|
if (user_options_extra->attack_kern == ATTACK_KERN_STRAIGHT)
|
|
{
|
|
kernel_loops = KERNEL_RULES;
|
|
}
|
|
else if (user_options_extra->attack_kern == ATTACK_KERN_COMBI)
|
|
{
|
|
kernel_loops = KERNEL_COMBS;
|
|
}
|
|
else if (user_options_extra->attack_kern == ATTACK_KERN_BF)
|
|
{
|
|
kernel_loops = KERNEL_BFS;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
kernel_loops = (int) strtol (token_ptr[5], NULL, 10);
|
|
|
|
if (kernel_loops < 1)
|
|
{
|
|
event_log_warning (hashcat_ctx, "Tuning-db: Invalid kernel_loops '%d' in Line '%d'", kernel_loops, line_num);
|
|
|
|
hcfree (buf);
|
|
|
|
return false;
|
|
}
|
|
|
|
if ((user_options_extra->attack_kern == ATTACK_KERN_STRAIGHT) && (kernel_loops > KERNEL_RULES))
|
|
{
|
|
event_log_warning (hashcat_ctx, "Tuning-db: Invalid kernel_loops '%d' in Line '%d'", kernel_loops, line_num);
|
|
|
|
hcfree (buf);
|
|
|
|
return false;
|
|
}
|
|
|
|
if ((user_options_extra->attack_kern == ATTACK_KERN_COMBI) && (kernel_loops > KERNEL_COMBS))
|
|
{
|
|
event_log_warning (hashcat_ctx, "Tuning-db: Invalid kernel_loops '%d' in Line '%d'", kernel_loops, line_num);
|
|
|
|
hcfree (buf);
|
|
|
|
return false;
|
|
}
|
|
|
|
if ((user_options_extra->attack_kern == ATTACK_KERN_BF) && (kernel_loops > KERNEL_BFS))
|
|
{
|
|
event_log_warning (hashcat_ctx, "Tuning-db: Invalid kernel_loops '%d' in Line '%d'", kernel_loops, line_num);
|
|
|
|
hcfree (buf);
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
tuning_db_entry_t *entry = &tuning_db->entry_buf[tuning_db->entry_cnt];
|
|
|
|
entry->device_name = hcstrdup (device_name);
|
|
entry->attack_mode = attack_mode;
|
|
entry->hash_mode = hash_mode;
|
|
entry->vector_width = vector_width;
|
|
entry->kernel_accel = kernel_accel;
|
|
entry->kernel_loops = kernel_loops;
|
|
|
|
tuning_db->entry_cnt++;
|
|
}
|
|
else
|
|
{
|
|
event_log_warning (hashcat_ctx, "Tuning-db: Invalid number of token in Line '%d'", line_num);
|
|
|
|
hcfree (buf);
|
|
|
|
return false;
|
|
}
|
|
|
|
hcfree (buf);
|
|
|
|
return true;
|
|
}
|
|
|
|
tuning_db_entry_t *tuning_db_search_real (hashcat_ctx_t *hashcat_ctx, const char *device_name, const cl_device_type device_type, int attack_mode, const int hash_mode)
|
|
{
|
|
tuning_db_t *tuning_db = hashcat_ctx->tuning_db;
|
|
|
|
static tuning_db_entry_t s;
|
|
|
|
// first we need to convert all spaces in the device_name to underscore
|
|
|
|
char *device_name_nospace = hcstrdup (device_name);
|
|
|
|
const size_t device_name_length = strlen (device_name_nospace);
|
|
|
|
size_t i;
|
|
|
|
for (i = 0; i < device_name_length; i++)
|
|
{
|
|
if (device_name_nospace[i] == ' ') device_name_nospace[i] = '_';
|
|
}
|
|
|
|
// find out if there's an alias configured
|
|
|
|
char *device_name_nospace2 = hcstrdup (device_name_nospace);
|
|
|
|
tuning_db_alias_t a;
|
|
|
|
a.device_name = device_name_nospace2;
|
|
|
|
char *alias_name = NULL;
|
|
|
|
for (i = device_name_length; i >= 1; i--)
|
|
{
|
|
device_name_nospace2[i] = 0;
|
|
|
|
tuning_db_alias_t *alias = (tuning_db_alias_t *) bsearch (&a, tuning_db->alias_buf, tuning_db->alias_cnt, sizeof (tuning_db_alias_t), sort_by_tuning_db_alias);
|
|
|
|
if (alias == NULL) continue;
|
|
|
|
alias_name = alias->alias_name;
|
|
|
|
break;
|
|
}
|
|
|
|
hcfree (device_name_nospace2);
|
|
|
|
// attack-mode 6 and 7 are attack-mode 1 basically
|
|
|
|
if (attack_mode == 6) attack_mode = 1;
|
|
if (attack_mode == 7) attack_mode = 1;
|
|
|
|
// bsearch is not ideal but fast enough
|
|
|
|
s.device_name = device_name_nospace;
|
|
s.attack_mode = attack_mode;
|
|
s.hash_mode = hash_mode;
|
|
|
|
tuning_db_entry_t *entry = NULL;
|
|
|
|
// this will produce all 2^3 combinations required
|
|
|
|
for (i = 0; i < 8; i++)
|
|
{
|
|
s.device_name = (i & 1) ? "*" : device_name_nospace;
|
|
s.attack_mode = (i & 2) ? -1 : attack_mode;
|
|
s.hash_mode = (i & 4) ? -1 : hash_mode;
|
|
|
|
entry = (tuning_db_entry_t *) bsearch (&s, tuning_db->entry_buf, tuning_db->entry_cnt, sizeof (tuning_db_entry_t), sort_by_tuning_db_entry);
|
|
|
|
if (entry != NULL) break;
|
|
|
|
// in non-wildcard mode do some additional checks:
|
|
|
|
if ((i & 1) == 0)
|
|
{
|
|
// in case we have an alias-name
|
|
|
|
if (alias_name != NULL)
|
|
{
|
|
s.device_name = alias_name;
|
|
|
|
entry = (tuning_db_entry_t *) bsearch (&s, tuning_db->entry_buf, tuning_db->entry_cnt, sizeof (tuning_db_entry_t), sort_by_tuning_db_entry);
|
|
|
|
if (entry != NULL) break;
|
|
}
|
|
|
|
// or by device type
|
|
|
|
if (device_type & CL_DEVICE_TYPE_CPU)
|
|
{
|
|
s.device_name = "DEVICE_TYPE_CPU";
|
|
}
|
|
else if (device_type & CL_DEVICE_TYPE_GPU)
|
|
{
|
|
s.device_name = "DEVICE_TYPE_GPU";
|
|
}
|
|
else if (device_type & CL_DEVICE_TYPE_ACCELERATOR)
|
|
{
|
|
s.device_name = "DEVICE_TYPE_ACCELERATOR";
|
|
}
|
|
|
|
entry = (tuning_db_entry_t *) bsearch (&s, tuning_db->entry_buf, tuning_db->entry_cnt, sizeof (tuning_db_entry_t), sort_by_tuning_db_entry);
|
|
|
|
if (entry != NULL) break;
|
|
}
|
|
}
|
|
|
|
// free converted device_name
|
|
|
|
hcfree (device_name_nospace);
|
|
|
|
return entry;
|
|
}
|
|
|
|
tuning_db_entry_t *tuning_db_search (hashcat_ctx_t *hashcat_ctx, const char *device_name, const cl_device_type device_type, int attack_mode, const int hash_mode)
|
|
{
|
|
tuning_db_entry_t *entry = NULL;
|
|
|
|
const char *NV_prefix = (const char *) "NVIDIA ";
|
|
|
|
if (strncmp (device_name, NV_prefix, strlen (NV_prefix)) == 0)
|
|
{
|
|
entry = tuning_db_search_real (hashcat_ctx, device_name + strlen (NV_prefix), device_type, attack_mode, hash_mode);
|
|
|
|
if (entry) return entry;
|
|
}
|
|
|
|
const char *AMD_prefix = (const char *) "AMD ";
|
|
|
|
if (strncmp (device_name, AMD_prefix, strlen (AMD_prefix)) == 0)
|
|
{
|
|
entry = tuning_db_search_real (hashcat_ctx, device_name + strlen (AMD_prefix), device_type, attack_mode, hash_mode);
|
|
|
|
if (entry) return entry;
|
|
}
|
|
|
|
entry = tuning_db_search_real (hashcat_ctx, device_name, device_type, attack_mode, hash_mode);
|
|
|
|
return entry;
|
|
}
|