2016-09-15 02:29:22 +00:00
|
|
|
/**
|
|
|
|
* Author......: See docs/credits.txt
|
|
|
|
* License.....: MIT
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "common.h"
|
|
|
|
#include "types.h"
|
2016-10-09 20:41:55 +00:00
|
|
|
#include "event.h"
|
2016-10-04 09:13:33 +00:00
|
|
|
#include "hwmon.h"
|
2016-09-15 02:29:22 +00:00
|
|
|
#include "timer.h"
|
2016-10-04 09:13:33 +00:00
|
|
|
#include "hashes.h"
|
2016-09-20 11:18:47 +00:00
|
|
|
#include "thread.h"
|
2016-09-15 14:02:52 +00:00
|
|
|
#include "restore.h"
|
2016-11-21 17:30:36 +00:00
|
|
|
#include "status.h"
|
2016-09-15 02:29:22 +00:00
|
|
|
#include "monitor.h"
|
|
|
|
|
2016-10-17 11:44:07 +00:00
|
|
|
int get_runtime_left (const hashcat_ctx_t *hashcat_ctx)
|
|
|
|
{
|
|
|
|
const status_ctx_t *status_ctx = hashcat_ctx->status_ctx;
|
|
|
|
const user_options_t *user_options = hashcat_ctx->user_options;
|
|
|
|
|
|
|
|
double msec_paused = status_ctx->msec_paused;
|
|
|
|
|
|
|
|
if (status_ctx->devices_status == STATUS_PAUSED)
|
|
|
|
{
|
|
|
|
double msec_paused_tmp = hc_timer_get (status_ctx->timer_paused);
|
|
|
|
|
|
|
|
msec_paused += msec_paused_tmp;
|
|
|
|
}
|
|
|
|
|
2017-12-10 00:40:45 +00:00
|
|
|
time_t runtime_cur;
|
2016-10-17 11:44:07 +00:00
|
|
|
|
2017-12-10 00:40:45 +00:00
|
|
|
time (&runtime_cur);
|
2016-10-17 11:44:07 +00:00
|
|
|
|
2016-10-30 17:55:27 +00:00
|
|
|
const int runtime_left = (int) (status_ctx->runtime_start
|
|
|
|
+ user_options->runtime
|
|
|
|
+ (msec_paused / 1000)
|
|
|
|
- runtime_cur);
|
2016-10-17 11:44:07 +00:00
|
|
|
|
|
|
|
return runtime_left;
|
|
|
|
}
|
|
|
|
|
2016-10-09 20:41:55 +00:00
|
|
|
static int monitor (hashcat_ctx_t *hashcat_ctx)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2016-10-17 11:44:07 +00:00
|
|
|
hashes_t *hashes = hashcat_ctx->hashes;
|
|
|
|
hwmon_ctx_t *hwmon_ctx = hashcat_ctx->hwmon_ctx;
|
2019-04-25 12:45:17 +00:00
|
|
|
backend_ctx_t *backend_ctx = hashcat_ctx->backend_ctx;
|
2016-10-17 11:44:07 +00:00
|
|
|
restore_ctx_t *restore_ctx = hashcat_ctx->restore_ctx;
|
|
|
|
status_ctx_t *status_ctx = hashcat_ctx->status_ctx;
|
|
|
|
user_options_t *user_options = hashcat_ctx->user_options;
|
2016-09-22 09:56:06 +00:00
|
|
|
|
2016-11-21 17:30:36 +00:00
|
|
|
bool runtime_check = false;
|
|
|
|
bool remove_check = false;
|
|
|
|
bool status_check = false;
|
|
|
|
bool restore_check = false;
|
|
|
|
bool hwmon_check = false;
|
|
|
|
bool performance_check = false;
|
2016-09-22 09:56:06 +00:00
|
|
|
|
2017-12-01 14:19:10 +00:00
|
|
|
const int sleep_time = 1;
|
|
|
|
const double exec_low = 50.0; // in ms
|
|
|
|
const double util_low = 90.0; // in percent
|
2016-09-22 09:56:06 +00:00
|
|
|
|
|
|
|
if (user_options->runtime)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2016-09-22 09:56:06 +00:00
|
|
|
runtime_check = true;
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
|
2016-09-30 10:11:46 +00:00
|
|
|
if (restore_ctx->enabled == true)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2016-09-22 09:56:06 +00:00
|
|
|
restore_check = true;
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
|
2019-02-25 10:20:22 +00:00
|
|
|
if ((user_options->remove == true) && ((hashes->hashlist_mode == HL_MODE_FILE_PLAIN) || (hashes->hashlist_mode == HL_MODE_FILE_BINARY)))
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2016-09-22 09:56:06 +00:00
|
|
|
remove_check = true;
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
|
2016-09-22 09:56:06 +00:00
|
|
|
if (user_options->status == true)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2016-09-22 09:56:06 +00:00
|
|
|
status_check = true;
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
|
2016-09-30 10:11:46 +00:00
|
|
|
if (hwmon_ctx->enabled == true)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2016-09-22 09:56:06 +00:00
|
|
|
hwmon_check = true;
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
|
2016-11-21 17:30:36 +00:00
|
|
|
if (hwmon_ctx->enabled == true)
|
|
|
|
{
|
|
|
|
performance_check = true; // this check simply requires hwmon to work
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((runtime_check == false) && (remove_check == false) && (status_check == false) && (restore_check == false) && (hwmon_check == false) && (performance_check == false))
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2016-10-09 20:41:55 +00:00
|
|
|
return 0;
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
|
2017-02-14 15:00:10 +00:00
|
|
|
// timer
|
2016-09-22 09:56:06 +00:00
|
|
|
|
2016-11-21 17:30:36 +00:00
|
|
|
u32 slowdown_warnings = 0;
|
|
|
|
u32 performance_warnings = 0;
|
2016-09-22 09:56:06 +00:00
|
|
|
|
2016-11-21 17:30:36 +00:00
|
|
|
u32 restore_left = user_options->restore_timer;
|
|
|
|
u32 remove_left = user_options->remove_timer;
|
|
|
|
u32 status_left = user_options->status_timer;
|
2016-09-22 09:56:06 +00:00
|
|
|
|
2016-09-29 21:49:33 +00:00
|
|
|
while (status_ctx->shutdown_inner == false)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2017-09-23 20:02:34 +00:00
|
|
|
sleep (sleep_time);
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2016-09-29 21:25:29 +00:00
|
|
|
if (status_ctx->devices_status == STATUS_INIT) continue;
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2018-11-09 11:48:27 +00:00
|
|
|
if (hwmon_ctx->enabled == true)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2016-09-29 22:04:12 +00:00
|
|
|
hc_thread_mutex_lock (status_ctx->mux_hwmon);
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2019-04-30 11:38:44 +00:00
|
|
|
for (int backend_devices_idx = 0; backend_devices_idx < backend_ctx->backend_devices_cnt; backend_devices_idx++)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2019-04-30 11:38:44 +00:00
|
|
|
hc_device_param_t *device_param = &backend_ctx->devices_param[backend_devices_idx];
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2016-11-18 09:09:03 +00:00
|
|
|
if (device_param->skipped == true) continue;
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2019-04-30 11:38:44 +00:00
|
|
|
if ((backend_ctx->devices_param[backend_devices_idx].opencl_device_type & CL_DEVICE_TYPE_GPU) == 0) continue;
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2019-04-30 11:38:44 +00:00
|
|
|
const int temperature = hm_get_temperature_with_devices_idx (hashcat_ctx, backend_devices_idx);
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2018-11-09 11:48:27 +00:00
|
|
|
if (temperature > (int) user_options->hwmon_temp_abort)
|
2016-10-10 09:03:11 +00:00
|
|
|
{
|
2019-04-30 11:38:44 +00:00
|
|
|
EVENT_DATA (EVENT_MONITOR_TEMP_ABORT, &backend_devices_idx, sizeof (int));
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2018-11-09 11:48:27 +00:00
|
|
|
myabort (hashcat_ctx);
|
2016-10-10 09:03:11 +00:00
|
|
|
}
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
|
2019-04-30 11:38:44 +00:00
|
|
|
for (int backend_devices_idx = 0; backend_devices_idx < backend_ctx->backend_devices_cnt; backend_devices_idx++)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2019-04-30 11:38:44 +00:00
|
|
|
hc_device_param_t *device_param = &backend_ctx->devices_param[backend_devices_idx];
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2016-11-18 09:09:03 +00:00
|
|
|
if (device_param->skipped == true) continue;
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2019-03-04 09:29:57 +00:00
|
|
|
if (device_param->skipped_warning == true) continue;
|
|
|
|
|
2019-04-30 11:38:44 +00:00
|
|
|
const int rc_throttle = hm_get_throttle_with_devices_idx (hashcat_ctx, backend_devices_idx);
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2018-11-09 11:48:27 +00:00
|
|
|
if (rc_throttle == -1) continue;
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2018-11-09 11:48:27 +00:00
|
|
|
if (rc_throttle > 0)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2018-11-09 11:48:27 +00:00
|
|
|
slowdown_warnings++;
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2019-04-30 11:38:44 +00:00
|
|
|
if (slowdown_warnings == 1) EVENT_DATA (EVENT_MONITOR_THROTTLE1, &backend_devices_idx, sizeof (int));
|
|
|
|
if (slowdown_warnings == 2) EVENT_DATA (EVENT_MONITOR_THROTTLE2, &backend_devices_idx, sizeof (int));
|
|
|
|
if (slowdown_warnings == 3) EVENT_DATA (EVENT_MONITOR_THROTTLE3, &backend_devices_idx, sizeof (int));
|
2018-11-09 11:48:27 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (slowdown_warnings > 0) slowdown_warnings--;
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-29 22:04:12 +00:00
|
|
|
hc_thread_mutex_unlock (status_ctx->mux_hwmon);
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
|
2016-09-22 09:56:06 +00:00
|
|
|
if (restore_check == true)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
|
|
|
restore_left--;
|
|
|
|
|
2016-09-22 09:56:06 +00:00
|
|
|
if (restore_left == 0)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2021-06-04 07:48:27 +00:00
|
|
|
// Can't return from monitor for that reasons, see:
|
|
|
|
// https://github.com/hashcat/hashcat/issues/2704
|
|
|
|
//
|
|
|
|
//const int rc = cycle_restore (hashcat_ctx);
|
|
|
|
//
|
|
|
|
//if (rc == -1) return -1;
|
2016-10-09 20:41:55 +00:00
|
|
|
|
2021-06-04 07:48:27 +00:00
|
|
|
cycle_restore (hashcat_ctx);
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2016-09-22 09:56:06 +00:00
|
|
|
restore_left = user_options->restore_timer;
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-29 21:49:33 +00:00
|
|
|
if ((runtime_check == true) && (status_ctx->runtime_start > 0))
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2016-10-17 11:44:07 +00:00
|
|
|
const int runtime_left = get_runtime_left (hashcat_ctx);
|
2016-09-15 02:29:22 +00:00
|
|
|
|
|
|
|
if (runtime_left <= 0)
|
|
|
|
{
|
2016-10-17 23:24:03 +00:00
|
|
|
EVENT_DATA (EVENT_MONITOR_RUNTIME_LIMIT, NULL, 0);
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2017-01-06 07:45:40 +00:00
|
|
|
myabort_runtime (hashcat_ctx);
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-22 09:56:06 +00:00
|
|
|
if (remove_check == true)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
|
|
|
remove_left--;
|
|
|
|
|
2016-09-22 09:56:06 +00:00
|
|
|
if (remove_left == 0)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2016-09-16 15:01:18 +00:00
|
|
|
if (hashes->digests_saved != hashes->digests_done)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2016-09-16 15:01:18 +00:00
|
|
|
hashes->digests_saved = hashes->digests_done;
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2021-06-04 07:48:27 +00:00
|
|
|
// Can't return from monitor for that reasons, see:
|
|
|
|
// https://github.com/hashcat/hashcat/issues/2704
|
|
|
|
//
|
|
|
|
// const int rc = save_hash (hashcat_ctx);
|
|
|
|
//
|
|
|
|
// if (rc == -1) return -1;
|
2016-10-09 20:41:55 +00:00
|
|
|
|
2021-06-04 07:48:27 +00:00
|
|
|
save_hash (hashcat_ctx);
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
|
2016-09-22 09:56:06 +00:00
|
|
|
remove_left = user_options->remove_timer;
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-22 09:56:06 +00:00
|
|
|
if (status_check == true)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
|
|
|
status_left--;
|
|
|
|
|
2016-09-22 09:56:06 +00:00
|
|
|
if (status_left == 0)
|
2016-09-15 02:29:22 +00:00
|
|
|
{
|
2016-09-29 22:04:12 +00:00
|
|
|
hc_thread_mutex_lock (status_ctx->mux_display);
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2016-10-17 23:24:03 +00:00
|
|
|
EVENT_DATA (EVENT_MONITOR_STATUS_REFRESH, NULL, 0);
|
2016-10-14 19:38:52 +00:00
|
|
|
|
2016-09-29 22:04:12 +00:00
|
|
|
hc_thread_mutex_unlock (status_ctx->mux_display);
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2016-09-22 09:56:06 +00:00
|
|
|
status_left = user_options->status_timer;
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
}
|
2016-11-21 17:30:36 +00:00
|
|
|
|
|
|
|
if (performance_check == true)
|
|
|
|
{
|
|
|
|
int exec_cnt = 0;
|
|
|
|
int util_cnt = 0;
|
|
|
|
|
|
|
|
double exec_total = 0;
|
|
|
|
double util_total = 0;
|
|
|
|
|
|
|
|
hc_thread_mutex_lock (status_ctx->mux_hwmon);
|
|
|
|
|
2019-04-30 11:38:44 +00:00
|
|
|
for (int backend_devices_idx = 0; backend_devices_idx < backend_ctx->backend_devices_cnt; backend_devices_idx++)
|
2016-11-21 17:30:36 +00:00
|
|
|
{
|
2019-04-30 11:38:44 +00:00
|
|
|
hc_device_param_t *device_param = &backend_ctx->devices_param[backend_devices_idx];
|
2016-11-21 17:30:36 +00:00
|
|
|
|
|
|
|
if (device_param->skipped == true) continue;
|
|
|
|
|
2019-03-04 09:29:57 +00:00
|
|
|
if (device_param->skipped_warning == true) continue;
|
|
|
|
|
2016-11-21 17:30:36 +00:00
|
|
|
exec_cnt++;
|
|
|
|
|
2019-04-30 11:38:44 +00:00
|
|
|
const double exec = status_get_exec_msec_dev (hashcat_ctx, backend_devices_idx);
|
2016-11-21 17:30:36 +00:00
|
|
|
|
|
|
|
exec_total += exec;
|
|
|
|
|
2019-04-30 11:38:44 +00:00
|
|
|
const int util = hm_get_utilization_with_devices_idx (hashcat_ctx, backend_devices_idx);
|
2016-11-21 17:30:36 +00:00
|
|
|
|
|
|
|
if (util == -1) continue;
|
|
|
|
|
|
|
|
util_total += (double) util;
|
|
|
|
|
|
|
|
util_cnt++;
|
|
|
|
}
|
|
|
|
|
|
|
|
hc_thread_mutex_unlock (status_ctx->mux_hwmon);
|
|
|
|
|
|
|
|
double exec_avg = 0;
|
|
|
|
double util_avg = 0;
|
|
|
|
|
|
|
|
if (exec_cnt > 0) exec_avg = exec_total / exec_cnt;
|
|
|
|
if (util_cnt > 0) util_avg = util_total / util_cnt;
|
|
|
|
|
|
|
|
if ((exec_avg > 0) && (exec_avg < exec_low))
|
|
|
|
{
|
|
|
|
performance_warnings++;
|
|
|
|
|
|
|
|
if (performance_warnings == 10) EVENT_DATA (EVENT_MONITOR_PERFORMANCE_HINT, NULL, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((util_avg > 0) && (util_avg < util_low))
|
|
|
|
{
|
|
|
|
performance_warnings++;
|
|
|
|
|
|
|
|
if (performance_warnings == 10) EVENT_DATA (EVENT_MONITOR_PERFORMANCE_HINT, NULL, 0);
|
|
|
|
}
|
|
|
|
}
|
2018-10-17 08:55:47 +00:00
|
|
|
|
|
|
|
// stdin read timeout check
|
2018-10-31 10:37:06 +00:00
|
|
|
// note: we skip the stdin timeout check if it was disabled with stdin_timeout_abort set to 0
|
2018-10-17 08:55:47 +00:00
|
|
|
|
2018-10-31 10:37:06 +00:00
|
|
|
if (user_options->stdin_timeout_abort != 0)
|
2018-10-17 08:55:47 +00:00
|
|
|
{
|
2018-10-31 10:37:06 +00:00
|
|
|
if (status_get_progress_done (hashcat_ctx) == 0)
|
2018-10-17 08:55:47 +00:00
|
|
|
{
|
2018-10-31 10:37:06 +00:00
|
|
|
if (status_ctx->stdin_read_timeout_cnt > 0)
|
2018-10-20 16:53:14 +00:00
|
|
|
{
|
2018-10-31 10:37:06 +00:00
|
|
|
if (status_ctx->stdin_read_timeout_cnt >= user_options->stdin_timeout_abort)
|
|
|
|
{
|
|
|
|
EVENT_DATA (EVENT_MONITOR_NOINPUT_ABORT, NULL, 0);
|
2018-10-17 08:55:47 +00:00
|
|
|
|
2018-10-31 10:37:06 +00:00
|
|
|
myabort (hashcat_ctx);
|
2018-10-17 08:55:47 +00:00
|
|
|
|
2018-10-31 10:37:06 +00:00
|
|
|
status_ctx->shutdown_inner = true;
|
2018-10-17 08:55:47 +00:00
|
|
|
|
2018-10-31 10:37:06 +00:00
|
|
|
break;
|
|
|
|
}
|
2018-10-17 08:55:47 +00:00
|
|
|
|
2018-10-31 10:37:06 +00:00
|
|
|
if ((status_ctx->stdin_read_timeout_cnt % STDIN_TIMEOUT_WARN) == 0)
|
|
|
|
{
|
|
|
|
EVENT_DATA (EVENT_MONITOR_NOINPUT_HINT, NULL, 0);
|
|
|
|
}
|
2018-10-20 16:53:14 +00:00
|
|
|
}
|
2018-10-17 08:55:47 +00:00
|
|
|
}
|
|
|
|
}
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|
|
|
|
|
2016-10-04 09:03:20 +00:00
|
|
|
// final round of save_hash
|
|
|
|
|
|
|
|
if (remove_check == true)
|
|
|
|
{
|
|
|
|
if (hashes->digests_saved != hashes->digests_done)
|
|
|
|
{
|
2021-06-04 07:48:27 +00:00
|
|
|
// Can't return from monitor for that reasons, see:
|
|
|
|
// https://github.com/hashcat/hashcat/issues/2704
|
|
|
|
//
|
|
|
|
// const int rc = save_hash (hashcat_ctx);
|
|
|
|
//
|
|
|
|
// if (rc == -1) return -1;
|
|
|
|
|
|
|
|
save_hash (hashcat_ctx);
|
2016-10-04 09:03:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// final round of cycle_restore
|
|
|
|
|
|
|
|
if (restore_check == true)
|
|
|
|
{
|
2021-06-04 07:48:27 +00:00
|
|
|
// Can't return from monitor for that reasons, see:
|
|
|
|
// https://github.com/hashcat/hashcat/issues/2704
|
|
|
|
//
|
|
|
|
// const int rc = cycle_restore (hashcat_ctx);
|
|
|
|
//
|
|
|
|
// if (rc == -1) return -1;
|
|
|
|
|
|
|
|
cycle_restore (hashcat_ctx);
|
2016-10-04 09:03:20 +00:00
|
|
|
}
|
|
|
|
|
2016-10-09 20:41:55 +00:00
|
|
|
return 0;
|
2016-09-30 20:52:44 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 21:37:30 +00:00
|
|
|
HC_API_CALL void *thread_monitor (void *p)
|
2016-09-30 20:52:44 +00:00
|
|
|
{
|
|
|
|
hashcat_ctx_t *hashcat_ctx = (hashcat_ctx_t *) p;
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2016-10-09 20:41:55 +00:00
|
|
|
monitor (hashcat_ctx); // we should give back some useful returncode
|
2016-09-15 02:29:22 +00:00
|
|
|
|
2016-09-30 20:52:44 +00:00
|
|
|
return NULL;
|
2016-09-15 02:29:22 +00:00
|
|
|
}
|