/*** Copyright (C) 2021 J Reece Wilson (a/k/a "Reece"). All rights reserved. File: AuOpen.Win32.cpp Date: 2021-6-12 Author: Reece ***/ #include #include "AuProcesses.hpp" #include "AuOpen.Win32.hpp" #include #define COINIT_APARTMENTTHREADED 0x2 #define COINIT_DISABLE_OLE1DDE 0x4 namespace Aurora::Processes { static AuList> gOpenItems; static AuThreadPrimitives::ConditionMutex gCondMutex; static AuThreadPrimitives::ConditionVariable gCondVariable(AuUnsafeRaiiToShared(gCondMutex.AsPointer())); static AuThreads::ThreadUnique_t gOpenerThread; static void RunTasks() { AU_LOCK_GUARD(gCondMutex); while (AuIsThreadRunning()) { try { for (const auto &[uri, type] : gOpenItems) { bool bDirExists {}; bool bFileExists {}; if (uri.empty()) { continue; } if (!pShellExecuteW) { SysPushErrorUninitialized("Cannot open URIs yet"); continue; } bFileExists = AuIOFS::FileExists(uri); if (!type) { bDirExists = AuIOFS::DirExists(uri); if (!bFileExists && !bDirExists) { SysPushErrorGeneric("Exploit attempt? Attempted to open non-existent file/directory. (request: {})", uri); continue; } if (bFileExists) { if (!AuFS::IsFileBlocked(uri)) { SysPushErrorGeneric("Exploit attempt? Attempted to open untrusted file/directory. (request: {})", uri); continue; } } } else { if (bFileExists) { SysPushErrorGeneric("Exploit attempt? Attempted to open existing file/directory via URI ({})", uri); continue; } } pShellExecuteW(nullptr, bDirExists ? L"explore" : L"open", Locale::ConvertFromUTF8(uri).c_str(), nullptr, nullptr, SW_SHOWNORMAL); } gOpenItems.clear(); gCondVariable->WaitForSignal(); } catch (...) { Debug::PrintError(); AuLogWarn("An error occurred while dispatching a ShellExecute runner frame"); } } } static void OpenerThread() { if (pCoInitializeEx) { (void)pCoInitializeEx(nullptr, COINIT_APARTMENTTHREADED | COINIT_DISABLE_OLE1DDE); } RunTasks(); if (pCoUninitialize) { pCoUninitialize(); } } void InitWin32Opener() { gOpenerThread = AuThreads::ThreadUnique(AuThreads::ThreadInfo( AuMakeShared(AuThreads::IThreadVectorsFunctional::OnEntry_t(std::bind(OpenerThread)), AuThreads::IThreadVectorsFunctional::OnExit_t{}), "COM ShellExecute Runner" )); SysAssert(gOpenerThread); gOpenerThread->Run(); } void DeinitWin32Opener() { if (!gOpenerThread) { return; } gOpenerThread->SendExitSignal(); gCondVariable->Signal(); gOpenerThread.reset(); } AUKN_SYM void OpenUri(const AuString &uri) { AU_LOCK_GUARD(gCondMutex); AuTryInsert(gOpenItems, AuMakePair(uri, true)); gCondVariable->Signal(); } AUKN_SYM void OpenFile(const AuString &file) { auto path = AuIOFS::NormalizePathRet(file); if (path.empty()) { SysPushErrorMemory(); return; } { AU_LOCK_GUARD(gCondMutex); AuTryInsert(gOpenItems, AuMove(AuMakePair(AuMove(path), false))); gCondVariable->Signal(); } } }