/* * PROJECT: ReactOS Compatibility Layer Shell Extension * LICENSE: GPL-2.0-or-later (https://spdx.org/licenses/GPL-2.0-or-later) * PURPOSE: CLayerUIPropPage implementation * COPYRIGHT: Copyright 2015-2019 Mark Jansen (mark.jansen@reactos.org) */ #include "precomp.h" #include #include #include #include #include #include #include const GUID CLSID_CLayerUIPropPage = { 0x513D916F, 0x2A8E, 0x4F51, { 0xAE, 0xAB, 0x0C, 0xBC, 0x76, 0xFB, 0x1A, 0xF8 } }; #define GPLK_USER 1 #define GPLK_MACHINE 2 #define MAX_LAYER_LENGTH 256 static struct { const PCWSTR Display; const PCWSTR Name; } g_CompatModes[] = { { L"Windows 95", L"WIN95" }, { L"Windows 98/ME", L"WIN98" }, { L"Windows NT 4.0 (SP5)", L"NT4SP5" }, { L"Windows 2000", L"WIN2000" }, { L"Windows XP (SP2)", L"WINXPSP2" }, { L"Windows XP (SP3)", L"WINXPSP3" }, { L"Windows Server 2003 (SP1)", L"WINSRV03SP1" }, { L"Windows Server 2008 (SP1)", L"WINSRV08SP1" }, { L"Windows Vista", L"VISTARTM" }, { L"Windows Vista (SP1)", L"VISTASP1" }, { L"Windows Vista (SP2)", L"VISTASP2" }, { L"Windows 7", L"WIN7RTM" }, { L"Windows 7 (SP1)", L"WIN7SP1" }, { L"Windows 8", L"WIN8RTM" }, { L"Windows 8.1", L"WIN81RTM" }, { NULL, NULL } }; static struct { const PCWSTR Name; DWORD Id; } g_Layers[] = { { L"256COLOR", IDC_CHKRUNIN256COLORS }, { L"640X480", IDC_CHKRUNIN640480RES }, { L"DISABLETHEMES", IDC_CHKDISABLEVISUALTHEMES }, #if 0 { L"DISABLEDWM", IDC_??, TRUE }, { L"HIGHDPIAWARE", IDC_??, TRUE }, { L"RUNASADMIN", IDC_??, TRUE }, #endif { NULL, 0 } }; static const WCHAR* g_AllowedExtensions[] = { L".exe", L".msi", L".pif", L".bat", L".cmd", 0 }; BOOL IsBuiltinLayer(PCWSTR Name) { size_t n; for (n = 0; g_Layers[n].Name; ++n) { if (!wcsicmp(g_Layers[n].Name, Name)) { return TRUE; } } for (n = 0; g_CompatModes[n].Name; ++n) { if (!wcsicmp(g_CompatModes[n].Name, Name)) { return TRUE; } } return FALSE; } void ACDBG_FN(PCSTR FunctionName, PCWSTR Format, ...) { WCHAR Buffer[512]; WCHAR* Current = Buffer; size_t Length = _countof(Buffer); StringCchPrintfExW(Current, Length, &Current, &Length, STRSAFE_NULL_ON_FAILURE, L"[%-20S] ", FunctionName); va_list ArgList; va_start(ArgList, Format); StringCchVPrintfExW(Current, Length, &Current, &Length, STRSAFE_NULL_ON_FAILURE, Format, ArgList); va_end(ArgList); OutputDebugStringW(Buffer); } #define ACDBG(fmt, ...) ACDBG_FN(__FUNCTION__, fmt, ##__VA_ARGS__ ) CLayerUIPropPage::CLayerUIPropPage() : m_IsSfcProtected(FALSE) , m_AllowPermLayer(FALSE) , m_LayerQueryFlags(GPLK_USER) /* TODO: When do we read from HKLM? */ , m_RegistryOSMode(0) , m_OSMode(0) , m_RegistryEnabledLayers(0) , m_EnabledLayers(0) { CComBSTR title; title.LoadString(g_hModule, IDS_COMPAT_TITLE); m_psp.pszTitle = title.Detach(); m_psp.dwFlags |= PSP_USETITLE; } CLayerUIPropPage::~CLayerUIPropPage() { CComBSTR title; title.Attach((BSTR)m_psp.pszTitle); } HRESULT CLayerUIPropPage::InitFile(PCWSTR Filename) { CString ExpandedFilename; DWORD dwRequired = ExpandEnvironmentStringsW(Filename, NULL, 0); if (dwRequired > 0) { LPWSTR Buffer = ExpandedFilename.GetBuffer(dwRequired); DWORD dwReturned = ExpandEnvironmentStringsW(Filename, Buffer, dwRequired); if (dwRequired == dwReturned) { ExpandedFilename.ReleaseBufferSetLength(dwReturned - 1); ACDBG(L"Expanded '%s' => '%s'\r\n", Filename, (PCWSTR)ExpandedFilename); } else { ExpandedFilename.ReleaseBufferSetLength(0); ExpandedFilename = Filename; ACDBG(L"Failed during expansion '%s'\r\n", Filename); } } else { ACDBG(L"Failed to expand '%s'\r\n", Filename); ExpandedFilename = Filename; } PCWSTR pwszExt = PathFindExtensionW(ExpandedFilename); if (!pwszExt) { ACDBG(L"Failed to find an extension: '%s'\r\n", (PCWSTR)ExpandedFilename); return E_FAIL; } if (!wcsicmp(pwszExt, L".lnk")) { WCHAR Buffer[MAX_PATH]; if (!GetExeFromLnk(ExpandedFilename, Buffer, _countof(Buffer))) { ACDBG(L"Failed to read link target from: '%s'\r\n", (PCWSTR)ExpandedFilename); return E_FAIL; } if (!wcsicmp(Buffer, ExpandedFilename)) { ACDBG(L"Link redirects to itself: '%s'\r\n", (PCWSTR)ExpandedFilename); return E_FAIL; } return InitFile(Buffer); } CString tmp; if (tmp.GetEnvironmentVariable(L"SystemRoot")) { tmp += L"\\System32"; if (ExpandedFilename.GetLength() >= tmp.GetLength() && ExpandedFilename.Left(tmp.GetLength()).MakeLower() == tmp.MakeLower()) { ACDBG(L"Ignoring System32: %s\r\n", (PCWSTR)ExpandedFilename); return E_FAIL; } tmp.GetEnvironmentVariable(L"SystemRoot"); tmp += L"\\WinSxs"; if (ExpandedFilename.GetLength() >= tmp.GetLength() && ExpandedFilename.Left(tmp.GetLength()).MakeLower() == tmp.MakeLower()) { ACDBG(L"Ignoring WinSxs: %s\r\n", (PCWSTR)ExpandedFilename); return E_FAIL; } } for (size_t n = 0; g_AllowedExtensions[n]; ++n) { if (!wcsicmp(g_AllowedExtensions[n], pwszExt)) { m_Filename = ExpandedFilename; ACDBG(L"Got: %s\r\n", (PCWSTR)ExpandedFilename); m_IsSfcProtected = SfcIsFileProtected(NULL, m_Filename); m_AllowPermLayer = AllowPermLayer(ExpandedFilename); return S_OK; } } ACDBG(L"Extension not included: '%s'\r\n", pwszExt); return E_FAIL; } static BOOL GetLayerInfo(PCWSTR Filename, DWORD QueryFlags, PDWORD OSMode, PDWORD Enabledlayers, CSimpleArray& customLayers) { WCHAR wszLayers[MAX_LAYER_LENGTH] = { 0 }; DWORD dwBytes = sizeof(wszLayers); *OSMode = *Enabledlayers = 0; customLayers.RemoveAll(); if (!SdbGetPermLayerKeys(Filename, wszLayers, &dwBytes, QueryFlags)) return FALSE; for (PWCHAR Layer = wcstok(wszLayers, L" "); Layer; Layer = wcstok(NULL, L" ")) { size_t n; for (n = 0; g_Layers[n].Name; ++n) { if (!wcsicmp(g_Layers[n].Name, Layer)) { *Enabledlayers |= (1<& lhs, const CSimpleArray& rhs) { if (lhs.GetSize() != rhs.GetSize()) return FALSE; for (int n = 0; n < lhs.GetSize(); ++n) { if (lhs[n] != rhs[n]) return FALSE; } return TRUE; } BOOL CLayerUIPropPage::HasChanges() const { if (m_RegistryEnabledLayers != m_EnabledLayers) return TRUE; if (m_RegistryOSMode != m_OSMode) return TRUE; if (!ArrayEquals(m_RegistryCustomLayers, m_CustomLayers)) return TRUE; return FALSE; } int CLayerUIPropPage::OnApply() { if (HasChanges()) { BOOL bMachine = m_LayerQueryFlags == GPLK_MACHINE; for (size_t n = 0; g_CompatModes[n].Name; ++n) SetPermLayerState(m_Filename, g_CompatModes[n].Name, 0, bMachine, (n+1) == m_OSMode); for (size_t n = 0; g_Layers[n].Name; ++n) { SetPermLayerState(m_Filename, g_Layers[n].Name, 0, bMachine, ((1< 0) customLayers += L", "; customLayers += m_CustomLayers[j]; } SetDlgItemTextW(IDC_ENABLED_LAYERS, customLayers); SetModified(HasChanges()); } LRESULT CLayerUIPropPage::OnCtrlCommand(WORD wNotifyCode, WORD wID, HWND hWndCtl, BOOL &bHandled) { UpdateControls(); return 0; } LRESULT CLayerUIPropPage::OnEditModes(WORD wNotifyCode, WORD wID, HWND hWndCtl, BOOL &bHandled) { if (ShowEditCompatModes(m_hWnd, this)) UpdateControls(); return 0; } LRESULT CLayerUIPropPage::OnClickNotify(INT uCode, LPNMHDR hdr, BOOL& bHandled) { if (hdr->idFrom == IDC_INFOLINK) ShellExecute(NULL, L"open", L"https://www.reactos.org/forum/viewforum.php?f=4", NULL, NULL, SW_SHOW); return 0; } static BOOL DisableShellext() { HKEY hkey; LSTATUS ret = RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"SOFTWARE\\Policies\\Microsoft\\Windows\\AppCompat", 0, KEY_QUERY_VALUE, &hkey); BOOL Disable = FALSE; if (ret == ERROR_SUCCESS) { DWORD dwValue = 0; DWORD type, size = sizeof(dwValue); ret = RegQueryValueExW(hkey, L"DisableEngine", NULL, &type, (PBYTE)&dwValue, &size); if (ret == ERROR_SUCCESS && type == REG_DWORD) { Disable = !!dwValue; } if (!Disable) { size = sizeof(dwValue); ret = RegQueryValueExW(hkey, L"DisablePropPage", NULL, &type, (PBYTE)&dwValue, &size); if (ret == ERROR_SUCCESS && type == REG_DWORD) { Disable = !!dwValue; } } RegCloseKey(hkey); } return Disable; } STDMETHODIMP CLayerUIPropPage::Initialize(PCIDLIST_ABSOLUTE pidlFolder, LPDATAOBJECT pDataObj, HKEY hkeyProgID) { FORMATETC etc = { CF_HDROP, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }; STGMEDIUM stg; if (DisableShellext()) return E_ACCESSDENIED; HRESULT hr = pDataObj->GetData(&etc, &stg); if (FAILED(hr)) { ACDBG(L"Failed to retrieve Data from pDataObj.\r\n"); return E_INVALIDARG; } hr = E_FAIL; HDROP hdrop = (HDROP)GlobalLock(stg.hGlobal); if (hdrop) { UINT uNumFiles = DragQueryFileW(hdrop, 0xFFFFFFFF, NULL, 0); if (uNumFiles == 1) { WCHAR szFile[MAX_PATH * 2]; if (DragQueryFileW(hdrop, 0, szFile, _countof(szFile))) { this->AddRef(); hr = InitFile(szFile); } else { ACDBG(L"Failed to query the file.\r\n"); } } else { ACDBG(L"Invalid number of files: %d\r\n", uNumFiles); } GlobalUnlock(stg.hGlobal); } else { ACDBG(L"Could not lock stg.hGlobal\r\n"); } ReleaseStgMedium(&stg); return hr; }