2016-09-08 14:32:24 +00:00
|
|
|
/**
|
2016-09-11 20:20:15 +00:00
|
|
|
* Author......: See docs/credits.txt
|
2016-09-08 14:32:24 +00:00
|
|
|
* License.....: MIT
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "common.h"
|
|
|
|
#include "types.h"
|
|
|
|
#include "memory.h"
|
2016-10-09 20:41:55 +00:00
|
|
|
#include "event.h"
|
2016-09-28 20:52:42 +00:00
|
|
|
#include "convert.h"
|
|
|
|
#include "dictstat.h"
|
2017-08-11 09:15:43 +00:00
|
|
|
#include "rp.h"
|
2016-09-08 14:32:24 +00:00
|
|
|
#include "rp_cpu.h"
|
2016-10-31 10:28:06 +00:00
|
|
|
#include "shared.h"
|
2016-09-08 14:32:24 +00:00
|
|
|
#include "wordlist.h"
|
2022-09-25 15:42:19 +00:00
|
|
|
#include "bitops.h"
|
2020-03-10 14:42:55 +00:00
|
|
|
#include "emu_inc_hash_sha1.h"
|
2016-09-08 14:32:24 +00:00
|
|
|
|
2018-02-08 18:13:29 +00:00
|
|
|
size_t convert_from_hex (hashcat_ctx_t *hashcat_ctx, char *line_buf, const size_t line_len)
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
2019-10-09 08:49:58 +00:00
|
|
|
const hashconfig_t *hashconfig = hashcat_ctx->hashconfig;
|
2016-10-06 14:16:56 +00:00
|
|
|
const user_options_t *user_options = hashcat_ctx->user_options;
|
|
|
|
|
2016-09-08 14:32:24 +00:00
|
|
|
if (line_len & 1) return (line_len); // not in hex
|
|
|
|
|
2019-10-09 08:49:58 +00:00
|
|
|
if (hashconfig->opts_type & OPTS_TYPE_PT_HEX)
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
2016-12-20 00:46:30 +00:00
|
|
|
size_t i, j;
|
2016-09-08 14:32:24 +00:00
|
|
|
|
|
|
|
for (i = 0, j = 0; j < line_len; i += 1, j += 2)
|
|
|
|
{
|
|
|
|
line_buf[i] = hex_to_u8 ((const u8 *) &line_buf[j]);
|
|
|
|
}
|
|
|
|
|
|
|
|
memset (line_buf + i, 0, line_len - i);
|
|
|
|
|
|
|
|
return (i);
|
|
|
|
}
|
2016-11-16 12:24:41 +00:00
|
|
|
|
2017-08-16 10:07:14 +00:00
|
|
|
if (user_options->wordlist_autohex_disable == false)
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
2018-02-08 18:13:29 +00:00
|
|
|
if (is_hexify ((const u8 *) line_buf, line_len) == true)
|
2017-08-16 10:07:14 +00:00
|
|
|
{
|
2018-02-08 18:13:29 +00:00
|
|
|
const size_t new_len = exec_unhexify ((const u8 *) line_buf, line_len, (u8 *) line_buf, line_len);
|
2016-09-08 14:32:24 +00:00
|
|
|
|
2018-02-08 18:13:29 +00:00
|
|
|
return new_len;
|
2017-08-16 10:07:14 +00:00
|
|
|
}
|
2016-09-08 14:32:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return (line_len);
|
|
|
|
}
|
|
|
|
|
2019-06-26 17:06:46 +00:00
|
|
|
int load_segment (hashcat_ctx_t *hashcat_ctx, HCFILE *fp)
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
2016-10-06 14:16:56 +00:00
|
|
|
wl_data_t *wl_data = hashcat_ctx->wl_data;
|
|
|
|
|
2016-09-08 14:32:24 +00:00
|
|
|
// NOTE: use (never changing) ->incr here instead of ->avail otherwise the buffer gets bigger and bigger
|
|
|
|
|
|
|
|
wl_data->pos = 0;
|
|
|
|
|
2019-06-27 18:18:47 +00:00
|
|
|
wl_data->cnt = hc_fread (wl_data->buf, 1, wl_data->incr - 1000, fp);
|
2016-09-08 14:32:24 +00:00
|
|
|
|
|
|
|
wl_data->buf[wl_data->cnt] = 0;
|
|
|
|
|
2016-10-13 08:07:04 +00:00
|
|
|
if (wl_data->cnt == 0) return 0;
|
2016-09-08 14:32:24 +00:00
|
|
|
|
2016-10-13 08:07:04 +00:00
|
|
|
if (wl_data->buf[wl_data->cnt - 1] == '\n') return 0;
|
2016-09-08 14:32:24 +00:00
|
|
|
|
2019-06-26 17:06:46 +00:00
|
|
|
while (!hc_feof (fp))
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
|
|
|
if (wl_data->cnt == wl_data->avail)
|
|
|
|
{
|
2016-11-20 21:54:52 +00:00
|
|
|
wl_data->buf = (char *) hcrealloc (wl_data->buf, wl_data->avail, wl_data->incr);
|
2016-09-08 14:32:24 +00:00
|
|
|
|
|
|
|
wl_data->avail += wl_data->incr;
|
|
|
|
}
|
|
|
|
|
2019-06-26 17:06:46 +00:00
|
|
|
const int c = hc_fgetc (fp);
|
2016-09-08 14:32:24 +00:00
|
|
|
|
|
|
|
if (c == EOF) break;
|
|
|
|
|
|
|
|
wl_data->buf[wl_data->cnt] = (char) c;
|
|
|
|
|
|
|
|
wl_data->cnt++;
|
|
|
|
|
|
|
|
if (c == '\n') break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ensure stream ends with a newline
|
|
|
|
|
|
|
|
if (wl_data->buf[wl_data->cnt - 1] != '\n')
|
|
|
|
{
|
|
|
|
wl_data->cnt++;
|
|
|
|
|
|
|
|
wl_data->buf[wl_data->cnt - 1] = '\n';
|
|
|
|
}
|
|
|
|
|
2016-10-13 08:07:04 +00:00
|
|
|
return 0;
|
2016-09-08 14:32:24 +00:00
|
|
|
}
|
|
|
|
|
2021-12-11 18:38:30 +00:00
|
|
|
void get_next_word_lm_gen (char *buf, u64 sz, u64 *len, u64 *off, u64 cutlen)
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
|
|
|
char *ptr = buf;
|
|
|
|
|
2016-09-30 14:43:59 +00:00
|
|
|
for (u64 i = 0; i < sz; i++, ptr++)
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
|
|
|
if (*ptr >= 'a' && *ptr <= 'z') *ptr -= 0x20;
|
|
|
|
|
2021-12-11 18:38:30 +00:00
|
|
|
if (i == cutlen)
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
2022-02-20 11:12:00 +00:00
|
|
|
if (cutlen == 20) buf[i - 1]= ']'; // add ] in $HEX[] format
|
|
|
|
|
2016-09-08 14:32:24 +00:00
|
|
|
*len = i;
|
2022-02-20 11:12:00 +00:00
|
|
|
|
2021-12-11 18:38:30 +00:00
|
|
|
// but continue a loop to skip rest of the line
|
2016-09-08 14:32:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (*ptr != '\n') continue;
|
|
|
|
|
|
|
|
*off = i + 1;
|
|
|
|
|
|
|
|
if ((i > 0) && (buf[i - 1] == '\r')) i--;
|
|
|
|
|
2021-12-11 18:38:30 +00:00
|
|
|
if (i < cutlen + 1) *len = i;
|
2021-12-12 13:43:16 +00:00
|
|
|
|
2016-09-08 14:32:24 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
*off = sz;
|
2022-02-20 11:12:00 +00:00
|
|
|
|
|
|
|
if (sz < cutlen) *len = sz;
|
2021-12-11 18:38:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void get_next_word_lm_hex (char *buf, u64 sz, u64 *len, u64 *off)
|
|
|
|
{
|
2022-02-20 11:12:00 +00:00
|
|
|
// this one is called if --hex-wordlist is used
|
2021-12-11 18:38:30 +00:00
|
|
|
// we need 14 hex-digits to get 7 characters
|
2022-02-20 11:12:00 +00:00
|
|
|
// but first convert 7 chars to upper case if they are a-z
|
|
|
|
|
2021-12-12 14:11:46 +00:00
|
|
|
for (u64 i = 5; i < sz; i++)
|
|
|
|
{
|
|
|
|
if ((i & 1) == 0)
|
|
|
|
{
|
2022-02-20 11:12:00 +00:00
|
|
|
if (is_valid_hex_char (buf[i]))
|
|
|
|
if (is_valid_hex_char (buf[i + 1]))
|
|
|
|
{
|
|
|
|
if (buf[i] == '6')
|
|
|
|
if (buf[i+1] > '0')
|
|
|
|
buf[i] = '4';
|
|
|
|
if (buf[i] == '7')
|
|
|
|
if (buf[i+1] < 'B')
|
|
|
|
buf[i] = '5';
|
|
|
|
}
|
2021-12-12 14:11:46 +00:00
|
|
|
}
|
2022-02-20 11:12:00 +00:00
|
|
|
|
2021-12-12 14:11:46 +00:00
|
|
|
if (i == 12) break; // stop when 7 chars are converted
|
|
|
|
}
|
2022-02-20 11:12:00 +00:00
|
|
|
|
2021-12-12 14:11:46 +00:00
|
|
|
// call generic next_word
|
2022-02-20 11:12:00 +00:00
|
|
|
|
|
|
|
get_next_word_lm_gen (buf, sz, len, off, 14);
|
2021-12-11 18:38:30 +00:00
|
|
|
}
|
|
|
|
|
2021-12-11 20:14:45 +00:00
|
|
|
void get_next_word_lm_hex_or_text (char *buf, u64 sz, u64 *len, u64 *off)
|
2021-12-11 18:38:30 +00:00
|
|
|
{
|
|
|
|
// check if not $HEX[..] format
|
|
|
|
bool hex = true;
|
2022-02-20 11:12:00 +00:00
|
|
|
|
|
|
|
if (sz < 8) hex = false;
|
|
|
|
|
|
|
|
if (hex && (buf[0] != '$')) hex = false;
|
|
|
|
if (hex && (buf[1] != 'H')) hex = false;
|
|
|
|
if (hex && (buf[2] != 'E')) hex = false;
|
|
|
|
if (hex && (buf[3] != 'X')) hex = false;
|
|
|
|
if (hex && (buf[4] != '[')) hex = false;
|
|
|
|
|
|
|
|
if (hex)
|
|
|
|
{
|
|
|
|
char *ptr = buf + 5; // starting after '['
|
|
|
|
|
2021-12-12 13:27:10 +00:00
|
|
|
for (u64 i = 5; i < sz; i++, ptr++)
|
2021-12-11 18:38:30 +00:00
|
|
|
{
|
|
|
|
if (*ptr == ']')
|
|
|
|
{
|
2022-02-20 11:12:00 +00:00
|
|
|
if ((i & 1) == 0) hex = false; // not even number of characters
|
2021-12-12 13:27:10 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-02-20 11:12:00 +00:00
|
|
|
if (is_valid_hex_char (*ptr) == false)
|
2021-12-12 13:27:10 +00:00
|
|
|
{
|
|
|
|
hex = false;
|
2021-12-11 18:38:30 +00:00
|
|
|
break;
|
2021-12-12 13:27:10 +00:00
|
|
|
}
|
2021-12-12 13:43:16 +00:00
|
|
|
// upcase character if it is a letter 'a-z'
|
2021-12-12 13:27:10 +00:00
|
|
|
if ((i & 1) == 1) // if first hex-char
|
2022-02-20 11:04:56 +00:00
|
|
|
{
|
2022-02-20 11:12:00 +00:00
|
|
|
if (is_valid_hex_char (buf[i + 1]))
|
2021-12-12 14:11:46 +00:00
|
|
|
{
|
2021-12-12 13:27:10 +00:00
|
|
|
if (buf[i] == '6')
|
2022-02-20 11:12:00 +00:00
|
|
|
if (buf[i + 1] > '0')
|
2021-12-12 13:27:10 +00:00
|
|
|
buf[i] = '4';
|
|
|
|
if (buf[i] == '7')
|
2022-02-20 11:12:00 +00:00
|
|
|
if (buf[i + 1] < 'B')
|
2021-12-12 13:27:10 +00:00
|
|
|
buf[i] = '5';
|
|
|
|
}
|
|
|
|
}
|
2021-12-11 18:38:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (hex)
|
|
|
|
{
|
|
|
|
//$HEX[] format so we need max 14 hex-digits + 6 chars '$HEX[]'
|
2022-02-20 11:12:00 +00:00
|
|
|
get_next_word_lm_gen (buf, sz, len, off, 20);
|
2021-12-11 18:38:30 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// threat it as normal string
|
2022-02-20 11:12:00 +00:00
|
|
|
get_next_word_lm_gen (buf, sz, len, off, 7);
|
2021-12-11 18:38:30 +00:00
|
|
|
}
|
2016-09-08 14:32:24 +00:00
|
|
|
}
|
|
|
|
|
2021-12-11 20:14:45 +00:00
|
|
|
void get_next_word_lm_text (char *buf, u64 sz, u64 *len, u64 *off)
|
|
|
|
{
|
2022-02-20 11:12:00 +00:00
|
|
|
get_next_word_lm_gen (buf, sz, len, off, 7);
|
2021-12-11 20:14:45 +00:00
|
|
|
}
|
|
|
|
|
2016-09-30 14:43:59 +00:00
|
|
|
void get_next_word_uc (char *buf, u64 sz, u64 *len, u64 *off)
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
|
|
|
char *ptr = buf;
|
|
|
|
|
2016-09-30 14:43:59 +00:00
|
|
|
for (u64 i = 0; i < sz; i++, ptr++)
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
|
|
|
if (*ptr >= 'a' && *ptr <= 'z') *ptr -= 0x20;
|
|
|
|
|
|
|
|
if (*ptr != '\n') continue;
|
|
|
|
|
|
|
|
*off = i + 1;
|
|
|
|
|
|
|
|
if ((i > 0) && (buf[i - 1] == '\r')) i--;
|
|
|
|
|
|
|
|
*len = i;
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
*off = sz;
|
|
|
|
*len = sz;
|
|
|
|
}
|
|
|
|
|
2016-09-30 14:43:59 +00:00
|
|
|
void get_next_word_std (char *buf, u64 sz, u64 *len, u64 *off)
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
|
|
|
char *ptr = buf;
|
|
|
|
|
2016-09-30 14:43:59 +00:00
|
|
|
for (u64 i = 0; i < sz; i++, ptr++)
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
|
|
|
if (*ptr != '\n') continue;
|
|
|
|
|
|
|
|
*off = i + 1;
|
|
|
|
|
|
|
|
if ((i > 0) && (buf[i - 1] == '\r')) i--;
|
|
|
|
|
|
|
|
*len = i;
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
*off = sz;
|
|
|
|
*len = sz;
|
|
|
|
}
|
|
|
|
|
2019-06-26 17:06:46 +00:00
|
|
|
void get_next_word (hashcat_ctx_t *hashcat_ctx, HCFILE *fp, char **out_buf, u32 *out_len)
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
2016-10-06 14:16:56 +00:00
|
|
|
user_options_t *user_options = hashcat_ctx->user_options;
|
|
|
|
user_options_extra_t *user_options_extra = hashcat_ctx->user_options_extra;
|
|
|
|
wl_data_t *wl_data = hashcat_ctx->wl_data;
|
|
|
|
|
2016-09-08 14:32:24 +00:00
|
|
|
while (wl_data->pos < wl_data->cnt)
|
|
|
|
{
|
2016-09-30 14:43:59 +00:00
|
|
|
u64 off;
|
|
|
|
u64 len;
|
2016-09-08 14:32:24 +00:00
|
|
|
|
|
|
|
char *ptr = wl_data->buf + wl_data->pos;
|
|
|
|
|
2016-09-22 09:35:08 +00:00
|
|
|
wl_data->func (ptr, wl_data->cnt - wl_data->pos, &len, &off);
|
2016-09-08 14:32:24 +00:00
|
|
|
|
|
|
|
wl_data->pos += off;
|
|
|
|
|
2019-10-08 10:33:11 +00:00
|
|
|
// do the on-the-fly hex decode using original buffer
|
|
|
|
// this is safe as length only decreases in size
|
|
|
|
|
|
|
|
len = (u32) convert_from_hex (hashcat_ctx, ptr, len);
|
|
|
|
|
2017-04-14 14:36:28 +00:00
|
|
|
// do the on-the-fly encoding
|
2019-10-08 10:33:11 +00:00
|
|
|
// needs to write into new buffer because size case both decrease and increase
|
2017-04-14 14:36:28 +00:00
|
|
|
|
|
|
|
if (wl_data->iconv_enabled == true)
|
|
|
|
{
|
|
|
|
char *iconv_ptr = wl_data->iconv_tmp;
|
|
|
|
size_t iconv_sz = HCBUFSIZ_TINY;
|
|
|
|
|
|
|
|
size_t ptr_len = len;
|
|
|
|
|
|
|
|
const size_t iconv_rc = iconv (wl_data->iconv_ctx, &ptr, &ptr_len, &iconv_ptr, &iconv_sz);
|
|
|
|
|
2017-06-29 10:19:05 +00:00
|
|
|
if (iconv_rc == (size_t) -1) continue;
|
|
|
|
|
|
|
|
ptr = wl_data->iconv_tmp;
|
|
|
|
len = HCBUFSIZ_TINY - iconv_sz;
|
2017-04-14 14:36:28 +00:00
|
|
|
}
|
|
|
|
|
2019-10-08 10:33:11 +00:00
|
|
|
// this is only a test for length, not writing into output buffer
|
|
|
|
|
2016-09-22 10:15:13 +00:00
|
|
|
if (run_rule_engine (user_options_extra->rule_len_l, user_options->rule_buf_l))
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
2017-08-11 09:15:43 +00:00
|
|
|
if (len >= RP_PASSWORD_SIZE) continue;
|
2016-09-08 14:32:24 +00:00
|
|
|
|
2017-08-11 09:15:43 +00:00
|
|
|
char rule_buf_out[RP_PASSWORD_SIZE];
|
2016-11-16 16:10:45 +00:00
|
|
|
|
2017-06-29 10:19:05 +00:00
|
|
|
memset (rule_buf_out, 0, sizeof (rule_buf_out));
|
2016-09-08 14:32:24 +00:00
|
|
|
|
2018-02-08 18:13:29 +00:00
|
|
|
const int rule_len_out = _old_apply_rule (user_options->rule_buf_l, user_options_extra->rule_len_l, ptr, (u32) len, rule_buf_out);
|
2016-09-08 14:32:24 +00:00
|
|
|
|
2017-06-29 10:19:05 +00:00
|
|
|
if (rule_len_out < 0) continue;
|
2016-09-08 14:32:24 +00:00
|
|
|
}
|
|
|
|
|
2019-02-26 20:20:07 +00:00
|
|
|
if (len > PW_MAX) continue;
|
2017-06-29 10:19:05 +00:00
|
|
|
|
2016-09-08 14:32:24 +00:00
|
|
|
*out_buf = ptr;
|
2018-02-08 18:13:29 +00:00
|
|
|
*out_len = (u32) len;
|
2016-09-08 14:32:24 +00:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-06-26 17:06:46 +00:00
|
|
|
if (hc_feof (fp))
|
2016-09-08 14:32:24 +00:00
|
|
|
{
|
|
|
|
fprintf (stderr, "BUG feof()!!\n");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-06-26 17:06:46 +00:00
|
|
|
load_segment (hashcat_ctx, fp);
|
2016-09-08 14:32:24 +00:00
|
|
|
|
2019-06-26 17:06:46 +00:00
|
|
|
get_next_word (hashcat_ctx, fp, out_buf, out_len);
|
2016-09-08 14:32:24 +00:00
|
|
|
}
|
|
|
|
|
2018-09-01 10:31:17 +00:00
|
|
|
void pw_pre_add (hc_device_param_t *device_param, const u8 *pw_buf, const int pw_len, const u8 *base_buf, const int base_len, const int rule_idx)
|
|
|
|
{
|
|
|
|
if (device_param->pws_pre_cnt < device_param->kernel_power)
|
|
|
|
{
|
|
|
|
pw_pre_t *pw_pre = device_param->pws_pre_buf + device_param->pws_pre_cnt;
|
|
|
|
|
|
|
|
memcpy (pw_pre->pw_buf, pw_buf, pw_len);
|
|
|
|
|
|
|
|
pw_pre->pw_len = pw_len;
|
|
|
|
|
|
|
|
if (base_buf != NULL)
|
|
|
|
{
|
|
|
|
memcpy (pw_pre->base_buf, base_buf, base_len);
|
|
|
|
|
|
|
|
pw_pre->base_len = base_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
pw_pre->rule_idx = rule_idx;
|
|
|
|
|
|
|
|
device_param->pws_pre_cnt++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf (stdout, "BUG pw_pre_add()!!\n");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void pw_base_add (hc_device_param_t *device_param, pw_pre_t *pw_pre)
|
|
|
|
{
|
|
|
|
if (device_param->pws_base_cnt < device_param->kernel_power)
|
|
|
|
{
|
|
|
|
memcpy (device_param->pws_base_buf + device_param->pws_base_cnt, pw_pre, sizeof (pw_pre_t));
|
|
|
|
|
|
|
|
device_param->pws_base_cnt++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf (stderr, "BUG pw_base_add()!!\n");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-08 14:32:24 +00:00
|
|
|
void pw_add (hc_device_param_t *device_param, const u8 *pw_buf, const int pw_len)
|
|
|
|
{
|
2017-08-22 13:19:25 +00:00
|
|
|
if (device_param->pws_cnt < device_param->kernel_power)
|
|
|
|
{
|
2018-02-05 16:18:58 +00:00
|
|
|
pw_idx_t *pw_idx = device_param->pws_idx + device_param->pws_cnt;
|
2016-09-08 14:32:24 +00:00
|
|
|
|
2018-02-05 16:18:58 +00:00
|
|
|
const u32 pw_len4 = (pw_len + 3) & ~3; // round up to multiple of 4
|
2016-09-08 14:32:24 +00:00
|
|
|
|
2018-02-05 16:18:58 +00:00
|
|
|
const u32 pw_len4_cnt = pw_len4 / 4;
|
2016-09-08 14:32:24 +00:00
|
|
|
|
2018-02-05 16:18:58 +00:00
|
|
|
pw_idx->cnt = pw_len4_cnt;
|
|
|
|
pw_idx->len = pw_len;
|
2016-09-08 14:32:24 +00:00
|
|
|
|
2018-02-05 16:18:58 +00:00
|
|
|
u8 *dst = (u8 *) (device_param->pws_comp + pw_idx->off);
|
|
|
|
|
|
|
|
memcpy (dst, pw_buf, pw_len);
|
|
|
|
|
|
|
|
memset (dst + pw_len, 0, pw_len4 - pw_len);
|
|
|
|
|
|
|
|
// prepare next element
|
|
|
|
|
|
|
|
pw_idx_t *pw_idx_next = pw_idx + 1;
|
|
|
|
|
|
|
|
pw_idx_next->off = pw_idx->off + pw_idx->cnt;
|
2016-09-08 14:32:24 +00:00
|
|
|
|
|
|
|
device_param->pws_cnt++;
|
2017-08-22 13:19:25 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf (stderr, "BUG pw_add()!!\n");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
2016-09-08 14:32:24 +00:00
|
|
|
}
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2019-06-26 17:06:46 +00:00
|
|
|
int count_words (hashcat_ctx_t *hashcat_ctx, HCFILE *fp, const char *dictfile, u64 *result)
|
2016-09-12 10:59:40 +00:00
|
|
|
{
|
2016-10-06 14:16:56 +00:00
|
|
|
combinator_ctx_t *combinator_ctx = hashcat_ctx->combinator_ctx;
|
2017-07-18 12:45:15 +00:00
|
|
|
hashconfig_t *hashconfig = hashcat_ctx->hashconfig;
|
2016-10-06 14:16:56 +00:00
|
|
|
straight_ctx_t *straight_ctx = hashcat_ctx->straight_ctx;
|
2017-06-30 14:51:57 +00:00
|
|
|
mask_ctx_t *mask_ctx = hashcat_ctx->mask_ctx;
|
2016-10-06 14:16:56 +00:00
|
|
|
user_options_extra_t *user_options_extra = hashcat_ctx->user_options_extra;
|
|
|
|
user_options_t *user_options = hashcat_ctx->user_options;
|
|
|
|
wl_data_t *wl_data = hashcat_ctx->wl_data;
|
|
|
|
|
2016-09-30 20:52:44 +00:00
|
|
|
//hc_signal (NULL);
|
2016-09-12 10:59:40 +00:00
|
|
|
|
|
|
|
dictstat_t d;
|
|
|
|
|
2020-03-10 14:42:55 +00:00
|
|
|
memset (&d, 0, sizeof (d));
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2021-08-21 11:29:10 +00:00
|
|
|
if (hc_fstat (fp, &d.stat))
|
2017-02-22 15:33:23 +00:00
|
|
|
{
|
|
|
|
*result = 0;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2016-09-12 10:59:40 +00:00
|
|
|
|
|
|
|
d.stat.st_mode = 0;
|
|
|
|
d.stat.st_nlink = 0;
|
|
|
|
d.stat.st_uid = 0;
|
|
|
|
d.stat.st_gid = 0;
|
|
|
|
d.stat.st_rdev = 0;
|
|
|
|
d.stat.st_atime = 0;
|
|
|
|
|
2017-10-04 10:23:37 +00:00
|
|
|
#if defined (STAT_NANOSECONDS_ACCESS_TIME)
|
|
|
|
d.stat.STAT_NANOSECONDS_ACCESS_TIME = 0;
|
2017-09-29 10:13:29 +00:00
|
|
|
#endif
|
|
|
|
|
2016-09-12 10:59:40 +00:00
|
|
|
#if defined (_POSIX)
|
|
|
|
d.stat.st_blksize = 0;
|
|
|
|
d.stat.st_blocks = 0;
|
|
|
|
#endif
|
|
|
|
|
2017-04-14 14:36:28 +00:00
|
|
|
memset (d.encoding_from, 0, sizeof (d.encoding_from));
|
|
|
|
memset (d.encoding_to, 0, sizeof (d.encoding_to));
|
|
|
|
|
2019-01-20 22:02:12 +00:00
|
|
|
strncpy (d.encoding_from, user_options->encoding_from, sizeof (d.encoding_from) - 1);
|
|
|
|
strncpy (d.encoding_to, user_options->encoding_to, sizeof (d.encoding_to) - 1);
|
2017-04-14 14:36:28 +00:00
|
|
|
|
2017-02-22 15:33:23 +00:00
|
|
|
if (d.stat.st_size == 0)
|
|
|
|
{
|
|
|
|
*result = 0;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2020-03-10 14:42:55 +00:00
|
|
|
const size_t dictfile_len = strlen (dictfile);
|
|
|
|
|
|
|
|
u32 *dictfile_padded = (u32 *) hcmalloc (dictfile_len + 64); // padding required for sha1_update()
|
|
|
|
|
2020-03-10 14:49:02 +00:00
|
|
|
memcpy (dictfile_padded, dictfile, dictfile_len);
|
|
|
|
|
2022-09-25 15:42:19 +00:00
|
|
|
for (size_t i = 0, j = 0; i < dictfile_len; i += 4, j += 1)
|
|
|
|
{
|
|
|
|
dictfile_padded[j] = byte_swap_32 (dictfile_padded[j]);
|
|
|
|
}
|
|
|
|
|
2020-03-10 14:42:55 +00:00
|
|
|
sha1_ctx_t sha1_ctx;
|
|
|
|
sha1_init (&sha1_ctx);
|
|
|
|
sha1_update (&sha1_ctx, dictfile_padded, dictfile_len);
|
|
|
|
sha1_final (&sha1_ctx);
|
|
|
|
|
2022-09-25 15:42:19 +00:00
|
|
|
sha1_ctx.h[0] = byte_swap_32 (sha1_ctx.h[0]);
|
|
|
|
sha1_ctx.h[1] = byte_swap_32 (sha1_ctx.h[1]);
|
|
|
|
sha1_ctx.h[2] = byte_swap_32 (sha1_ctx.h[2]);
|
|
|
|
sha1_ctx.h[3] = byte_swap_32 (sha1_ctx.h[3]);
|
|
|
|
sha1_ctx.h[4] = byte_swap_32 (sha1_ctx.h[4]);
|
|
|
|
|
2020-03-10 14:42:55 +00:00
|
|
|
hcfree (dictfile_padded);
|
|
|
|
|
|
|
|
memcpy (d.hash_filename, sha1_ctx.h, 16);
|
|
|
|
|
2016-10-06 14:55:17 +00:00
|
|
|
const u64 cached_cnt = dictstat_find (hashcat_ctx, &d);
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2016-09-22 10:15:13 +00:00
|
|
|
if (run_rule_engine (user_options_extra->rule_len_l, user_options->rule_buf_l) == 0)
|
2016-09-12 10:59:40 +00:00
|
|
|
{
|
|
|
|
if (cached_cnt)
|
|
|
|
{
|
|
|
|
u64 keyspace = cached_cnt;
|
|
|
|
|
2016-09-22 13:50:49 +00:00
|
|
|
if (user_options_extra->attack_kern == ATTACK_KERN_STRAIGHT)
|
2016-09-12 10:59:40 +00:00
|
|
|
{
|
2017-02-23 09:39:17 +00:00
|
|
|
if (overflow_check_u64_mul (keyspace, straight_ctx->kernel_rules_cnt) == false) return -1;
|
2017-02-22 15:33:23 +00:00
|
|
|
|
2016-09-27 11:13:07 +00:00
|
|
|
keyspace *= straight_ctx->kernel_rules_cnt;
|
2016-09-12 10:59:40 +00:00
|
|
|
}
|
2016-09-22 13:50:49 +00:00
|
|
|
else if (user_options_extra->attack_kern == ATTACK_KERN_COMBI)
|
2016-09-12 10:59:40 +00:00
|
|
|
{
|
2017-07-18 12:45:15 +00:00
|
|
|
if (((hashconfig->opti_type & OPTI_TYPE_OPTIMIZED_KERNEL) == 0) && (user_options->attack_mode == ATTACK_MODE_HYBRID2))
|
2017-06-30 14:51:57 +00:00
|
|
|
{
|
|
|
|
if (overflow_check_u64_mul (keyspace, mask_ctx->bfs_cnt) == false) return -1;
|
|
|
|
|
|
|
|
keyspace *= mask_ctx->bfs_cnt;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (overflow_check_u64_mul (keyspace, combinator_ctx->combs_cnt) == false) return -1;
|
|
|
|
|
|
|
|
keyspace *= combinator_ctx->combs_cnt;
|
|
|
|
}
|
2016-09-12 10:59:40 +00:00
|
|
|
}
|
|
|
|
|
2016-10-23 12:06:52 +00:00
|
|
|
cache_hit_t cache_hit;
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2017-11-14 04:46:26 +00:00
|
|
|
cache_hit.dictfile = dictfile;
|
2016-10-31 09:20:25 +00:00
|
|
|
cache_hit.stat.st_size = d.stat.st_size;
|
|
|
|
cache_hit.cached_cnt = cached_cnt;
|
|
|
|
cache_hit.keyspace = keyspace;
|
2016-10-23 12:06:52 +00:00
|
|
|
|
|
|
|
EVENT_DATA (EVENT_WORDLIST_CACHE_HIT, &cache_hit, sizeof (cache_hit));
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2017-02-22 15:33:23 +00:00
|
|
|
*result = keyspace;
|
|
|
|
|
|
|
|
return 0;
|
2016-09-12 10:59:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-10 00:40:45 +00:00
|
|
|
time_t rt_start;
|
2017-05-23 08:07:04 +00:00
|
|
|
|
2017-12-10 00:40:45 +00:00
|
|
|
time (&rt_start);
|
2017-05-23 08:07:04 +00:00
|
|
|
|
2017-12-10 00:40:45 +00:00
|
|
|
time_t now = 0;
|
|
|
|
time_t prev = 0;
|
2016-09-12 10:59:40 +00:00
|
|
|
|
|
|
|
u64 comp = 0;
|
|
|
|
u64 cnt = 0;
|
|
|
|
u64 cnt2 = 0;
|
|
|
|
|
2019-06-26 17:06:46 +00:00
|
|
|
while (!hc_feof (fp))
|
2016-09-12 10:59:40 +00:00
|
|
|
{
|
2019-06-26 17:06:46 +00:00
|
|
|
load_segment (hashcat_ctx, fp);
|
2016-09-12 10:59:40 +00:00
|
|
|
|
|
|
|
comp += wl_data->cnt;
|
|
|
|
|
2016-09-30 14:43:59 +00:00
|
|
|
u64 i = 0;
|
2016-09-12 10:59:40 +00:00
|
|
|
|
|
|
|
while (i < wl_data->cnt)
|
|
|
|
{
|
2016-09-30 14:43:59 +00:00
|
|
|
u64 len;
|
|
|
|
u64 off;
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2017-04-14 14:36:28 +00:00
|
|
|
char *ptr = wl_data->buf + i;
|
|
|
|
|
|
|
|
wl_data->func (ptr, wl_data->cnt - i, &len, &off);
|
|
|
|
|
2017-06-29 10:19:05 +00:00
|
|
|
i += off;
|
|
|
|
|
2019-10-08 10:33:11 +00:00
|
|
|
// do the on-the-fly hex decode using original buffer
|
|
|
|
// this is safe as length only decreases in size
|
|
|
|
|
|
|
|
len = (u32) convert_from_hex (hashcat_ctx, ptr, len);
|
|
|
|
|
2017-04-14 14:36:28 +00:00
|
|
|
// do the on-the-fly encoding
|
|
|
|
|
|
|
|
if (wl_data->iconv_enabled == true)
|
|
|
|
{
|
|
|
|
char *iconv_ptr = wl_data->iconv_tmp;
|
|
|
|
size_t iconv_sz = HCBUFSIZ_TINY;
|
|
|
|
|
|
|
|
size_t ptr_len = len;
|
|
|
|
|
|
|
|
const size_t iconv_rc = iconv (wl_data->iconv_ctx, &ptr, &ptr_len, &iconv_ptr, &iconv_sz);
|
|
|
|
|
2017-06-29 10:19:05 +00:00
|
|
|
if (iconv_rc == (size_t) -1) continue;
|
|
|
|
|
|
|
|
ptr = wl_data->iconv_tmp;
|
|
|
|
len = HCBUFSIZ_TINY - iconv_sz;
|
2017-04-14 14:36:28 +00:00
|
|
|
}
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2016-09-22 10:15:13 +00:00
|
|
|
if (run_rule_engine (user_options_extra->rule_len_l, user_options->rule_buf_l))
|
2016-09-12 10:59:40 +00:00
|
|
|
{
|
2017-08-11 09:15:43 +00:00
|
|
|
if (len >= RP_PASSWORD_SIZE) continue;
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2017-08-11 09:15:43 +00:00
|
|
|
char rule_buf_out[RP_PASSWORD_SIZE];
|
2017-02-22 15:33:23 +00:00
|
|
|
|
2017-06-29 10:19:05 +00:00
|
|
|
memset (rule_buf_out, 0, sizeof (rule_buf_out));
|
2017-02-22 15:33:23 +00:00
|
|
|
|
2018-02-08 18:13:29 +00:00
|
|
|
const int rule_len_out = _old_apply_rule (user_options->rule_buf_l, user_options_extra->rule_len_l, ptr, (u32) len, rule_buf_out);
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2017-06-29 10:19:05 +00:00
|
|
|
if (rule_len_out < 0) continue;
|
2016-09-12 10:59:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cnt2++;
|
2017-06-29 10:19:05 +00:00
|
|
|
|
2019-02-26 20:20:07 +00:00
|
|
|
if (len > PW_MAX) continue;
|
2017-06-29 10:19:05 +00:00
|
|
|
|
|
|
|
d.cnt++;
|
|
|
|
|
|
|
|
if (user_options_extra->attack_kern == ATTACK_KERN_STRAIGHT)
|
|
|
|
{
|
|
|
|
if (overflow_check_u64_add (cnt, straight_ctx->kernel_rules_cnt) == false) return -1;
|
|
|
|
|
|
|
|
cnt += straight_ctx->kernel_rules_cnt;
|
|
|
|
}
|
|
|
|
else if (user_options_extra->attack_kern == ATTACK_KERN_COMBI)
|
|
|
|
{
|
2017-07-18 12:45:15 +00:00
|
|
|
if (((hashconfig->opti_type & OPTI_TYPE_OPTIMIZED_KERNEL) == 0) && (user_options->attack_mode == ATTACK_MODE_HYBRID2))
|
2017-06-30 14:51:57 +00:00
|
|
|
{
|
|
|
|
if (overflow_check_u64_add (cnt, mask_ctx->bfs_cnt) == false) return -1;
|
|
|
|
|
|
|
|
cnt += mask_ctx->bfs_cnt;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (overflow_check_u64_add (cnt, combinator_ctx->combs_cnt) == false) return -1;
|
|
|
|
|
|
|
|
cnt += combinator_ctx->combs_cnt;
|
|
|
|
}
|
2017-06-29 10:19:05 +00:00
|
|
|
}
|
2016-09-12 10:59:40 +00:00
|
|
|
}
|
|
|
|
|
2017-12-10 00:40:45 +00:00
|
|
|
time (&now);
|
2016-09-12 10:59:40 +00:00
|
|
|
|
|
|
|
if ((now - prev) == 0) continue;
|
|
|
|
|
2017-12-10 00:40:45 +00:00
|
|
|
time (&prev);
|
2016-10-23 12:49:40 +00:00
|
|
|
|
2016-10-23 12:06:52 +00:00
|
|
|
double percent = ((double) comp / (double) d.stat.st_size) * 100;
|
|
|
|
|
2016-10-26 12:42:19 +00:00
|
|
|
if (percent < 100)
|
|
|
|
{
|
|
|
|
cache_generate_t cache_generate;
|
2016-10-23 12:06:52 +00:00
|
|
|
|
2017-11-14 04:46:26 +00:00
|
|
|
cache_generate.dictfile = dictfile;
|
2016-10-26 12:42:19 +00:00
|
|
|
cache_generate.comp = comp;
|
|
|
|
cache_generate.percent = percent;
|
|
|
|
cache_generate.cnt = cnt;
|
|
|
|
cache_generate.cnt2 = cnt2;
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2016-10-26 12:42:19 +00:00
|
|
|
EVENT_DATA (EVENT_WORDLIST_CACHE_GENERATE, &cache_generate, sizeof (cache_generate));
|
|
|
|
}
|
2016-09-12 10:59:40 +00:00
|
|
|
}
|
|
|
|
|
2017-12-10 00:40:45 +00:00
|
|
|
time_t rt_stop;
|
2017-05-23 08:07:04 +00:00
|
|
|
|
2017-12-10 00:40:45 +00:00
|
|
|
time (&rt_stop);
|
2017-05-23 08:07:04 +00:00
|
|
|
|
2016-10-23 12:06:52 +00:00
|
|
|
cache_generate_t cache_generate;
|
|
|
|
|
2017-11-14 04:46:26 +00:00
|
|
|
cache_generate.dictfile = dictfile;
|
2016-10-23 12:06:52 +00:00
|
|
|
cache_generate.comp = comp;
|
|
|
|
cache_generate.percent = 100;
|
|
|
|
cache_generate.cnt = cnt;
|
|
|
|
cache_generate.cnt2 = cnt2;
|
2017-05-23 08:07:04 +00:00
|
|
|
cache_generate.runtime = rt_stop - rt_start;
|
2016-10-23 12:06:52 +00:00
|
|
|
|
|
|
|
EVENT_DATA (EVENT_WORDLIST_CACHE_GENERATE, &cache_generate, sizeof (cache_generate));
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2016-10-06 14:55:17 +00:00
|
|
|
dictstat_append (hashcat_ctx, &d);
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2016-09-30 20:52:44 +00:00
|
|
|
//hc_signal (sigHandler_default);
|
2016-09-12 10:59:40 +00:00
|
|
|
|
2017-02-22 15:33:23 +00:00
|
|
|
*result = cnt;
|
|
|
|
|
|
|
|
return 0;
|
2016-09-12 10:59:40 +00:00
|
|
|
}
|
2016-09-22 09:35:08 +00:00
|
|
|
|
2016-10-13 08:07:04 +00:00
|
|
|
int wl_data_init (hashcat_ctx_t *hashcat_ctx)
|
2016-09-22 09:35:08 +00:00
|
|
|
{
|
2022-02-13 11:33:11 +00:00
|
|
|
wl_data_t *wl_data = hashcat_ctx->wl_data;
|
2016-10-06 14:16:56 +00:00
|
|
|
hashconfig_t *hashconfig = hashcat_ctx->hashconfig;
|
|
|
|
user_options_t *user_options = hashcat_ctx->user_options;
|
|
|
|
|
2016-09-30 11:02:15 +00:00
|
|
|
wl_data->enabled = false;
|
|
|
|
|
2023-05-01 15:38:42 +00:00
|
|
|
if (user_options->usage > 0) return 0;
|
|
|
|
if (user_options->backend_info > 0) return 0;
|
|
|
|
|
2022-02-13 11:33:11 +00:00
|
|
|
if (user_options->benchmark == true) return 0;
|
|
|
|
if (user_options->hash_info == true) return 0;
|
|
|
|
if (user_options->left == true) return 0;
|
|
|
|
if (user_options->version == true) return 0;
|
2016-09-30 11:02:15 +00:00
|
|
|
|
|
|
|
wl_data->enabled = true;
|
|
|
|
|
2023-05-01 15:38:42 +00:00
|
|
|
wl_data->buf = (char *) hcmalloc (user_options->segment_size);
|
|
|
|
wl_data->avail = user_options->segment_size;
|
|
|
|
wl_data->incr = user_options->segment_size;
|
|
|
|
wl_data->cnt = 0;
|
|
|
|
wl_data->pos = 0;
|
2016-09-22 09:35:08 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* choose dictionary parser
|
|
|
|
*/
|
|
|
|
|
|
|
|
wl_data->func = get_next_word_std;
|
|
|
|
|
|
|
|
if (hashconfig->opts_type & OPTS_TYPE_PT_UPPER)
|
|
|
|
{
|
|
|
|
wl_data->func = get_next_word_uc;
|
|
|
|
}
|
|
|
|
|
2018-12-19 08:51:25 +00:00
|
|
|
if (hashconfig->opts_type & OPTS_TYPE_PT_LM)
|
2016-09-22 09:35:08 +00:00
|
|
|
{
|
2022-02-20 11:12:00 +00:00
|
|
|
if (hashconfig->opts_type & OPTS_TYPE_PT_HEX)
|
|
|
|
{
|
2021-12-11 20:14:45 +00:00
|
|
|
wl_data->func = get_next_word_lm_hex; // all hex in file
|
2021-12-11 18:38:30 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-12-11 20:14:45 +00:00
|
|
|
if (user_options->wordlist_autohex_disable == false)
|
|
|
|
{
|
|
|
|
wl_data->func = get_next_word_lm_hex_or_text; // might be $HEX[] notation
|
2022-02-20 11:12:00 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-09-07 20:52:06 +00:00
|
|
|
wl_data->func = get_next_word_lm_text; // treat as normal text
|
2021-12-11 20:14:45 +00:00
|
|
|
}
|
2021-12-11 18:38:30 +00:00
|
|
|
}
|
2016-09-22 09:35:08 +00:00
|
|
|
}
|
2016-10-13 08:07:04 +00:00
|
|
|
|
2017-04-14 14:36:28 +00:00
|
|
|
/**
|
|
|
|
* iconv
|
|
|
|
*/
|
|
|
|
|
2017-11-05 08:52:29 +00:00
|
|
|
if (strcmp (user_options->encoding_from, user_options->encoding_to) != 0)
|
2017-04-14 14:36:28 +00:00
|
|
|
{
|
|
|
|
wl_data->iconv_enabled = true;
|
|
|
|
|
|
|
|
wl_data->iconv_ctx = iconv_open (user_options->encoding_to, user_options->encoding_from);
|
|
|
|
|
|
|
|
if (wl_data->iconv_ctx == (iconv_t) -1) return -1;
|
|
|
|
|
|
|
|
wl_data->iconv_tmp = (char *) hcmalloc (HCBUFSIZ_TINY);
|
|
|
|
}
|
|
|
|
|
2016-10-13 08:07:04 +00:00
|
|
|
return 0;
|
2016-09-22 09:35:08 +00:00
|
|
|
}
|
|
|
|
|
2016-10-06 14:16:56 +00:00
|
|
|
void wl_data_destroy (hashcat_ctx_t *hashcat_ctx)
|
2016-09-22 09:35:08 +00:00
|
|
|
{
|
2016-10-06 14:16:56 +00:00
|
|
|
wl_data_t *wl_data = hashcat_ctx->wl_data;
|
|
|
|
|
2016-10-01 22:00:21 +00:00
|
|
|
if (wl_data->enabled == false) return;
|
2016-09-30 11:02:15 +00:00
|
|
|
|
2016-10-10 09:03:11 +00:00
|
|
|
hcfree (wl_data->buf);
|
2016-09-22 09:35:08 +00:00
|
|
|
|
2017-04-14 14:36:28 +00:00
|
|
|
if (wl_data->iconv_enabled == true)
|
|
|
|
{
|
|
|
|
iconv_close (wl_data->iconv_ctx);
|
|
|
|
|
|
|
|
wl_data->iconv_enabled = false;
|
|
|
|
|
|
|
|
hcfree (wl_data->iconv_tmp);
|
|
|
|
}
|
|
|
|
|
2016-10-01 22:00:21 +00:00
|
|
|
memset (wl_data, 0, sizeof (wl_data_t));
|
2016-09-22 09:35:08 +00:00
|
|
|
}
|