/** * Author......: See docs/credits.txt * License.....: MIT */ #include "common.h" #include "types.h" #include "memory.h" #include "event.h" #include "shared.h" #include "folder.h" #if defined (__APPLE__) #include "event.h" #endif int sort_by_stringptr (const void *p1, const void *p2) { const char **s1 = (const char **) p1; const char **s2 = (const char **) p2; return strcmp (*s1, *s2); } static int get_exec_path (char *exec_path, const size_t exec_path_sz) { #if defined (__linux__) || defined (__CYGWIN__) char *tmp; hc_asprintf (&tmp, "/proc/%d/exe", getpid ()); const ssize_t len = readlink (tmp, exec_path, exec_path_sz - 1); hcfree (tmp); if (len == -1) return -1; #elif defined (_WIN) const DWORD len = GetModuleFileName (NULL, exec_path, exec_path_sz - 1); if (len == 0) return -1; #elif defined (__APPLE__) u32 size = (u32) exec_path_sz; if (_NSGetExecutablePath (exec_path, &size) != 0) return -1; const size_t len = strlen (exec_path); #elif defined (__FreeBSD__) #include int mib[4]; mib[0] = CTL_KERN; mib[1] = KERN_PROC; mib[2] = KERN_PROC_PATHNAME; mib[3] = -1; size_t size = exec_path_sz; sysctl (mib, 4, exec_path, &size, NULL, 0); const size_t len = strlen (exec_path); #else #error Your Operating System is not supported or detected #endif exec_path[len] = 0; return 0; } static void get_install_dir (char *install_dir, const char *exec_path) { strncpy (install_dir, exec_path, HCBUFSIZ_TINY - 1); char *last_slash = NULL; if ((last_slash = strrchr (install_dir, '/')) != NULL) { *last_slash = 0; } else if ((last_slash = strrchr (install_dir, '\\')) != NULL) { *last_slash = 0; } else { install_dir[0] = '.'; install_dir[1] = 0; } } #if defined (_POSIX) static void get_profile_dir (char *profile_dir, const char *home_dir) { snprintf (profile_dir, HCBUFSIZ_TINY - 1, "%s/%s", home_dir, DOT_HASHCAT); } static void get_session_dir (char *session_dir, const char *profile_dir) { snprintf (session_dir, HCBUFSIZ_TINY - 1, "%s/%s", profile_dir, SESSIONS_FOLDER); } #endif int count_dictionaries (char **dictionary_files) { if (dictionary_files == NULL) return 0; int cnt = 0; for (int d = 0; dictionary_files[d] != NULL; d++) { cnt++; } return (cnt); } char *first_file_in_directory (const char *path) { DIR *d = NULL; if ((d = opendir (path)) != NULL) { char *first_file = NULL; #if 0 struct dirent e; for (;;) { memset (&e, 0, sizeof (e)); struct dirent *de = NULL; if (readdir_r (d, &e, &de) != 0) break; if (de == NULL) break; #else struct dirent *de; while ((de = readdir (d)) != NULL) { #endif if (de->d_name[0] == '.') continue; first_file = strdup (de->d_name); break; } closedir (d); return first_file; } else if (errno == ENOTDIR) { return NULL; } return NULL; } char **scan_directory (const char *path) { char *tmp_path = hcstrdup (path); size_t tmp_path_len = strlen (tmp_path); while (tmp_path[tmp_path_len - 1] == '/' || tmp_path[tmp_path_len - 1] == '\\') { tmp_path[tmp_path_len - 1] = 0; tmp_path_len = strlen (tmp_path); } char **files = NULL; size_t num_files = 0; DIR *d = NULL; if ((d = opendir (tmp_path)) != NULL) { #if 0 struct dirent e; for (;;) { memset (&e, 0, sizeof (e)); struct dirent *de = NULL; if (readdir_r (d, &e, &de) != 0) break; if (de == NULL) break; #else struct dirent *de; while ((de = readdir (d)) != NULL) { #endif if (de->d_name[0] == '.') continue; char *path_file; hc_asprintf (&path_file, "%s/%s", tmp_path, de->d_name); DIR *d_test; if ((d_test = opendir (path_file)) != NULL) { closedir (d_test); hcfree (path_file); } else { files = (char **) hcrealloc (files, (num_files + 1) * sizeof (char *), sizeof (char *)); files[num_files] = path_file; num_files++; } } closedir (d); } else if (errno == ENOTDIR) { files = (char **) hcrealloc (files, (num_files + 1) * sizeof (char *), sizeof (char *)); files[num_files] = hcstrdup (path); num_files++; } files = (char **) hcrealloc (files, (num_files + 1) * sizeof (char *), sizeof (char *)); files[num_files] = NULL; hcfree (tmp_path); return (files); } int folder_config_init (hashcat_ctx_t *hashcat_ctx, MAYBE_UNUSED const char *install_folder, MAYBE_UNUSED const char *shared_folder) { folder_config_t *folder_config = hashcat_ctx->folder_config; /** * There's some buggy OpenCL runtime that do not support -I. * A workaround is to chdir() to the OpenCL folder, * then compile the kernels, * then chdir() back to where we came from so we need to save it first */ char *cwd = (char *) hcmalloc (HCBUFSIZ_TINY); if (getcwd (cwd, HCBUFSIZ_TINY - 1) == NULL) { event_log_error (hashcat_ctx, "getcwd(): %s", strerror (errno)); return -1; } /** * folders, as discussed on https://github.com/hashcat/hashcat/issues/20 */ const size_t exec_path_sz = 1024; char *exec_path = (char *) hcmalloc (exec_path_sz); const int rc = get_exec_path (exec_path, exec_path_sz); if (rc == -1) { event_log_error (hashcat_ctx, "get_exec_path() failed"); return -1; } #if defined(__linux__) || defined(__APPLE__) || defined(__FreeBSD__) || defined (__CYGWIN__) static const char SLASH[] = "/"; if (install_folder == NULL) install_folder = SLASH; // makes library use easier char *resolved_install_folder = realpath (install_folder, NULL); char *resolved_exec_path = realpath (exec_path, NULL); if (resolved_install_folder == NULL) resolved_install_folder = hcstrdup (SLASH); /* This causes invalid error out if install_folder (/usr/local/bin) does not exist if (resolved_install_folder == NULL) { event_log_error (hashcat_ctx, "%s: %s", resolved_install_folder, strerror (errno)); return -1; } */ if (resolved_exec_path == NULL) { event_log_error (hashcat_ctx, "%s: %s", resolved_exec_path, strerror (errno)); return -1; } char *install_dir = hcmalloc (HCBUFSIZ_TINY); get_install_dir (install_dir, resolved_exec_path); char *profile_dir = NULL; char *session_dir = NULL; char *shared_dir = NULL; if (strcmp (install_dir, resolved_install_folder) == 0) { struct passwd pw; struct passwd *pwp; char buf[HCBUFSIZ_TINY]; getpwuid_r (getuid (), &pw, buf, HCBUFSIZ_TINY, &pwp); const char *home_dir = pwp->pw_dir; profile_dir = hcmalloc (HCBUFSIZ_TINY); session_dir = hcmalloc (HCBUFSIZ_TINY); get_profile_dir (profile_dir, home_dir); get_session_dir (session_dir, profile_dir); shared_dir = hcstrdup (shared_folder); hc_mkdir (profile_dir, 0700); hc_mkdir (session_dir, 0700); } else { profile_dir = install_dir; session_dir = install_dir; shared_dir = install_dir; } hcfree (resolved_install_folder); hcfree (resolved_exec_path); #else char *install_dir = hcmalloc (HCBUFSIZ_TINY); get_install_dir (install_dir, exec_path); char *profile_dir = install_dir; char *session_dir = install_dir; char *shared_dir = install_dir; #endif hcfree (exec_path); /** * There's alot of problem related to bad support -I parameters when building the kernel. * Each OpenCL runtime handles it slightly different. * The most problematic is with new AMD drivers on Windows, which can not handle quote characters! * The best workaround found so far is to modify the TMP variable (only inside hashcat process) before the runtime is load */ char *cpath; #if defined (_WIN) hc_asprintf (&cpath, "%s\\OpenCL\\", shared_dir); char *cpath_real = (char *) hcmalloc (HCBUFSIZ_TINY); if (GetFullPathName (cpath, HCBUFSIZ_TINY - 1, cpath_real, NULL) == 0) { event_log_error (hashcat_ctx, "%s: %s", cpath, "GetFullPathName()"); return -1; } #else hc_asprintf (&cpath, "%s/OpenCL/", shared_dir); char *cpath_real = (char *) hcmalloc (PATH_MAX); if (realpath (cpath, cpath_real) == NULL) { event_log_error (hashcat_ctx, "%s: %s", cpath, strerror (errno)); return -1; } #endif hcfree (cpath); //if (getenv ("TMP") == NULL) if (1) { char *tmp; hc_asprintf (&tmp, "TMP=%s", cpath_real); putenv (tmp); } #if defined (_WIN) naive_replace (cpath_real, '\\', '/'); // not escaping here, windows using quotes later // naive_escape (cpath_real, PATH_MAX, ' ', '\\'); #else naive_escape (cpath_real, PATH_MAX, ' ', '\\'); #endif /** * kernel cache, we need to make sure folder exist */ char *kernels_folder; hc_asprintf (&kernels_folder, "%s/kernels", profile_dir); hc_mkdir (kernels_folder, 0700); hcfree (kernels_folder); /** * store for later use */ folder_config->cwd = cwd; folder_config->install_dir = install_dir; folder_config->profile_dir = profile_dir; folder_config->session_dir = session_dir; folder_config->shared_dir = shared_dir; folder_config->cpath_real = cpath_real; return 0; } void folder_config_destroy (hashcat_ctx_t *hashcat_ctx) { folder_config_t *folder_config = hashcat_ctx->folder_config; hcfree (folder_config->cpath_real); hcfree (folder_config->cwd); hcfree (folder_config->install_dir); memset (folder_config, 0, sizeof (folder_config_t)); } int hc_mkdir (const char *name, MAYBE_UNUSED const int mode) { #if defined (_WIN) return _mkdir (name); #else return mkdir (name, mode); #endif }