#include "fileutils.h" #include #include #include #include "logging.h" bool fileutils::file_exists(LPCSTR szPath) { DWORD dwAttrib = GetFileAttributesA(szPath); return (dwAttrib != INVALID_FILE_ATTRIBUTES && !(dwAttrib & FILE_ATTRIBUTE_DIRECTORY)); } bool fileutils::file_exists(LPCWSTR szPath) { DWORD dwAttrib = GetFileAttributesW(szPath); return (dwAttrib != INVALID_FILE_ATTRIBUTES && !(dwAttrib & FILE_ATTRIBUTE_DIRECTORY)); } bool fileutils::file_exists(const std::string &file_path) { return file_exists(file_path.c_str()); } bool fileutils::file_exists(const std::filesystem::path &file_path) { return file_exists(file_path.c_str()); } bool fileutils::verify_header_pe(const std::filesystem::path &file_path) { if (!file_exists(file_path)) { return false; } // open file HANDLE dll_file; dll_file = CreateFileW( file_path.c_str(), GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); if (!dll_file) { return false; } // get size LARGE_INTEGER dll_file_size; if (!GetFileSizeEx(dll_file, &dll_file_size) || (size_t) dll_file_size.QuadPart < sizeof(PIMAGE_DOS_HEADER)) { CloseHandle(dll_file); return false; } // create file mapping HANDLE dll_mapping = CreateFileMappingW(dll_file, NULL, PAGE_READONLY, 0, 0, NULL); if (!dll_mapping) { CloseHandle(dll_file); return false; } // map view of file LPVOID dll_file_base = MapViewOfFile(dll_mapping, FILE_MAP_READ, 0, 0, 0); if (!dll_file_base) { CloseHandle(dll_file); CloseHandle(dll_mapping); return false; } // verify header bool valid = false; auto dll_dos = reinterpret_cast(dll_file_base); if (dll_dos->e_magic == IMAGE_DOS_SIGNATURE) { // verify architecture auto dll_nt = (PIMAGE_NT_HEADERS) ((uint8_t*) dll_dos + dll_dos->e_lfanew); if ((size_t) dll_nt - (size_t) dll_file_base < (size_t) dll_file_size.QuadPart) { auto dll_file_header = (PIMAGE_FILE_HEADER) &dll_nt->FileHeader; if ((size_t) dll_file_header - (size_t) dll_file_base < (size_t) dll_file_size.QuadPart) { #if SPICE64 valid = dll_file_header->Machine == IMAGE_FILE_MACHINE_AMD64; if (!valid) { log_fatal("fileutils", "{} (32 bit) can't be loaded using spice64.exe - please use spice.exe for this game.", file_path.string()); } #else valid = dll_file_header->Machine == IMAGE_FILE_MACHINE_I386; if (!valid) { log_fatal("fileutils", "{} (64 bit) can't be loaded using spice.exe - please use spice64.exe for this game.", file_path.string()); } #endif } } } // clean up and return UnmapViewOfFile(dll_file_base); CloseHandle(dll_file); CloseHandle(dll_mapping); return valid; } bool fileutils::version_pe(const std::filesystem::path &file_path, char *ver) { DWORD dwHandle = 0; DWORD dwLen = GetFileVersionInfoSizeW(file_path.c_str(), &dwHandle); if (!dwLen) { return false; } auto buf = std::make_unique(dwLen); if (!GetFileVersionInfoW(file_path.c_str(), dwHandle, dwLen, buf.get())) { return false; } VS_FIXEDFILEINFO *pvi = nullptr; UINT uLen = 0; if (!VerQueryValueW(buf.get(), L"\\", reinterpret_cast(&pvi), &uLen)) { return false; } sprintf(ver, "%d.%d.%d.%d", (int) (pvi->dwProductVersionMS >> 16), (int) (pvi->dwFileVersionMS & 0xFFFF), (int) (pvi->dwFileVersionLS >> 16), (int) (pvi->dwFileVersionLS & 0xFFFF)); return true; } bool fileutils::dir_exists(const std::filesystem::path &dir_path) { std::error_code err; auto status = std::filesystem::status(dir_path, err); if (err) { return false; } return std::filesystem::is_directory(status); } bool fileutils::dir_create(const std::filesystem::path &dir_path) { std::error_code err; auto ret = std::filesystem::create_directory(dir_path, err); return ret && !err; } bool fileutils::dir_create_log(const std::string_view &module, const std::filesystem::path &dir_path) { std::error_code err; auto ret = std::filesystem::create_directory(dir_path, err); if (err) { log_warning(module, "failed to create directory '{}': {}", dir_path.string(), err.message()); } else if (ret) { log_misc(module, "created directory '{}'", dir_path.string()); } return ret && !err; } bool fileutils::dir_create_recursive(const std::filesystem::path &dir_path) { std::error_code err; auto ret = std::filesystem::create_directories(dir_path, err); return ret && !err; } bool fileutils::dir_create_recursive_log(const std::string_view &module, const std::filesystem::path &dir_path) { std::error_code err; auto ret = std::filesystem::create_directories(dir_path, err); if (err) { log_warning(module, "failed to create directory (recursive) '{}': {}", dir_path.string(), err.message()); } else if (ret) { log_misc(module, "created directory (recursive) '{}'", dir_path.string()); } return ret && !err; } void fileutils::dir_scan(const std::string &path, std::vector &vec, bool recursive) { // check directory if (std::filesystem::exists(path) && std::filesystem::is_directory(path)) { if (recursive) { for (const auto &entry : std::filesystem::recursive_directory_iterator(path)) { if (!std::filesystem::is_directory(entry)) { auto path = entry.path().string(); vec.emplace_back(std::move(path)); } } } else { for (const auto &entry : std::filesystem::directory_iterator(path)) { if (!std::filesystem::is_directory(entry)) { auto path = entry.path().string(); vec.emplace_back(std::move(path)); } } } } // determinism std::sort(vec.begin(), vec.end()); } bool fileutils::text_write(const std::filesystem::path &file_path, std::string text) { std::ofstream out(file_path, std::ios::out | std::ios::binary); if (out) { out << text; out.close(); return true; } return false; } std::string fileutils::text_read(const std::filesystem::path &file_path) { std::ifstream in(file_path, std::ios::in | std::ios::binary); if (in) { std::string contents; in.seekg(0, std::ios::end); contents.reserve(in.tellg()); in.seekg(0, std::ios::beg); std::copy(std::istreambuf_iterator(in), std::istreambuf_iterator(), std::back_inserter(contents)); in.close(); return contents; } return std::string(); } bool fileutils::bin_write(const std::filesystem::path &path, uint8_t *data, size_t len) { // write to disk std::ofstream out(path, std::ios::out | std::ios::binary); if (out) { out.write((const char*) data, len); out.close(); return true; } return false; } std::vector *fileutils::bin_read(const std::filesystem::path &path) { // read from disk std::ifstream in(path, std::ios::in | std::ios::binary | std::ios::ate); auto contents = new std::vector(); if (in) { contents->resize((unsigned) in.tellg()); in.seekg(0, std::ios::beg); if (!in.read((char*) contents->data(), contents->size())) { contents->clear(); } in.close(); } return contents; }