reactos/reactos/subsystems/win32/win32k/ntuser/class.c

2264 lines
63 KiB
C
Raw Normal View History

/*
* ReactOS W32 Subsystem
* Copyright (C) 1998 - 2006 ReactOS Team
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
/* $Id$
*
* COPYRIGHT: See COPYING in the top level directory
* PROJECT: ReactOS kernel
* PURPOSE: Window classes
* FILE: subsys/win32k/ntuser/class.c
* PROGRAMER: Thomas Weidenmueller <w3seek@reactos.com>
* REVISION HISTORY:
* 06-06-2001 CSH Created
*/
/* INCLUDES ******************************************************************/
#include <w32k.h>
#define NDEBUG
#include <debug.h>
#define TRACE DPRINT
#define WARN DPRINT1
#define ERR DPRINT1
/* WINDOWCLASS ***************************************************************/
static VOID
IntFreeClassMenuName(IN OUT PWINDOWCLASS Class)
{
/* free the menu name, if it was changed and allocated */
if (Class->MenuName != NULL && !IS_INTRESOURCE(Class->MenuName) &&
Class->MenuName != (PWSTR)(Class + 1))
{
UserHeapFree(Class->MenuName);
Class->MenuName = NULL;
Class->AnsiMenuName = NULL;
}
}
static VOID
IntDestroyClass(IN OUT PWINDOWCLASS Class)
{
/* there shouldn't be any clones anymore */
- Fix SleepEx. - Put volatile statements in EX_RUNDOWN_REF, IRP, DEVICE_OBJECT, ERESOURCE, FILE_OBJECT, IO_REMOVE_LOCK, WORK_QUEUE_ITEM where required (thanks to Microsoft's changes in the WDK to mark the fields properly). - Update FILE_OBJECT definition. - Add some asserts to some I/O functions. - Add stub support for File Objects created by XP+ Drivers which have File Object Extensions. - Add some fixes to IopDeleteFile, including proper reference counting for the DO and VPB, as well as cleanup when the file is closed without a handle. - Fix a bug in IopSecurityFile. - Queue and unqueue IRPs in all I/O functions. - Fully support IRP cancellation now. - Fix critical bugs in NtDeviceIoControlFile and NtDeviceFsControlFile which were causing double queueing of IRPs and freeing of invalid memory, as well as invalid paramter checking for user-mode buffers. - Add exhaustive validation checks to IoCreateFile, add more failure cases, and validate the EA buffer. Also support IO_ATTACH_DEVICE_API flag. - Implement IoCreateStreamFileObjectEx and IoCreateStreamFileObjectLite and fix several bugs in the original implementation of IoCreateStreamFileObject. - Fix a bug in RtlRaiseException. - Update Io*ShareAccess routines to support XP+ style semantics related to special File Object flags which disable their use. - Add validation to all Query/Set routines so that information clasess, lengths, buffers and alignment are properly checked. - Also add an array for the proper acess rights that each query/set operation requires. - Check backup/restore privileges during I/O File operations. - Check traverse access during I/O File Operations. - Check access privileges to the device during I/O file operations. - Rename IopReferenceDeviceObject and also verify if an exclusive DO is trying to be invalidly opened. - Support various extra security checks during I/O File/Device Parse Routine. - Fix a bug during IopCleanupIrp so that we don't dereference the File OBject if this was a create operation. - Fix some bogus asserts in IofCompleteRequest, and save the IRP Flags before signalling it's event, since the driver might've freed it behind our back. - Fix a large bug in ObInsertObject which affected the insert of unnamed objects with forced security options (Such as process/threads). - Fix the creation of the Process/Thread/Job Obejct Types to that security information is forced. - Remove "Fix PS!!!" messages since the bug is now fixed and these objects now get proper security descriptors. - Fix another bug in ObInsertObjet which wasn't properly validating user-mode objects and always assumed kernel mode. - Silence multiple trace/checkpoint messages that have accumulated throughout time for various debugging purposes. svn path=/trunk/; revision=25118
2006-12-10 18:40:30 +00:00
//ASSERT(Class->Windows == 0);
if (Class->Windows)
{
DPRINT1("FIXME: W3Seek's Class Patch is broken!\n");
Class->Windows = 0;
}
ASSERT(Class->Clone == NULL);
if (Class->Base == Class)
{
/* destruct resources shared with clones */
if (!Class->System && Class->CallProc != NULL)
{
DestroyCallProc(Class->GlobalCallProc ? NULL : Class->Desktop,
Class->CallProc);
}
if (Class->CallProc2 != NULL)
{
DestroyCallProc(Class->GlobalCallProc2 ? NULL : Class->Desktop,
Class->CallProc2);
}
IntFreeClassMenuName(Class);
}
/* free the structure */
if (Class->Desktop != NULL)
{
DesktopHeapFree(Class->Desktop,
Class);
}
else
{
UserHeapFree(Class);
}
}
/* clean all process classes. all process windows must cleaned first!! */
void FASTCALL DestroyProcessClasses(PW32PROCESS Process )
{
PWINDOWCLASS Class;
PW32PROCESSINFO pi = Process->ProcessInfo;
if (pi != NULL)
{
/* free all local classes */
Class = pi->LocalClassList;
while (Class != NULL)
{
pi->LocalClassList = Class->Next;
ASSERT(Class->Base == Class);
IntDestroyClass(Class);
Class = pi->LocalClassList;
}
/* free all global classes */
Class = pi->GlobalClassList;
while (Class != NULL)
{
pi->GlobalClassList = Class->Next;
ASSERT(Class->Base == Class);
IntDestroyClass(Class);
Class = pi->GlobalClassList;
}
/* free all system classes */
Class = pi->SystemClassList;
while (Class != NULL)
{
pi->SystemClassList = Class->Next;
ASSERT(Class->Base == Class);
IntDestroyClass(Class);
Class = pi->SystemClassList;
}
}
}
static BOOL
IntRegisterClassAtom(IN PUNICODE_STRING ClassName,
OUT RTL_ATOM *pAtom)
{
WCHAR szBuf[65];
PWSTR AtomName;
NTSTATUS Status;
if (ClassName->Length != 0)
{
/* FIXME - Don't limit to 64 characters! use SEH when allocating memory! */
if (ClassName->Length / sizeof(WCHAR) >= sizeof(szBuf) / sizeof(szBuf[0]))
{
SetLastWin32Error(ERROR_INVALID_PARAMETER);
return (RTL_ATOM)0;
}
RtlCopyMemory(szBuf,
ClassName->Buffer,
ClassName->Length);
szBuf[ClassName->Length / sizeof(WCHAR)] = UNICODE_NULL;
AtomName = szBuf;
}
else
AtomName = ClassName->Buffer;
Status = RtlAddAtomToAtomTable(gAtomTable,
AtomName,
pAtom);
if (!NT_SUCCESS(Status))
{
SetLastNtError(Status);
return FALSE;
}
return TRUE;
}
static NTSTATUS
IntDeregisterClassAtom(IN RTL_ATOM Atom)
{
return RtlDeleteAtomFromAtomTable(gAtomTable,
Atom);
}
static BOOL
IntSetClassAtom(IN OUT PWINDOWCLASS Class,
IN PUNICODE_STRING ClassName)
{
RTL_ATOM Atom = (RTL_ATOM)0;
/* update the base class first */
Class = Class->Base;
if (!IntRegisterClassAtom(ClassName,
&Atom))
{
return FALSE;
}
IntDeregisterClassAtom(Class->Atom);
Class->Atom = Atom;
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
Class->Atom = Atom;
Class = Class->Next;
}
return TRUE;
}
static WNDPROC
IntGetClassWndProc(IN PWINDOWCLASS Class,
IN PW32PROCESSINFO pi,
IN BOOL Ansi,
IN BOOL UseCallProc2)
{
ASSERT(UserIsEnteredExclusive() == TRUE);
if (Class->System)
{
return (Ansi ? Class->WndProcExtra : Class->WndProc);
}
else
{
if (!Ansi == Class->Unicode)
{
return Class->WndProc;
}
else
{
PCALLPROC *CallProcPtr;
PWINDOWCLASS BaseClass;
/* make sure the call procedures are located on the desktop
of the base class! */
BaseClass = Class->Base;
Class = BaseClass;
CallProcPtr = (UseCallProc2 ? &Class->CallProc2 : &Class->CallProc);
if (*CallProcPtr != NULL)
{
return GetCallProcHandle(*CallProcPtr);
}
else
{
PCALLPROC NewCallProc;
if (pi == NULL)
return NULL;
NewCallProc = CreateCallProc(Class->Desktop,
Class->WndProc,
Class->Unicode,
pi);
if (NewCallProc == NULL)
{
SetLastWin32Error(ERROR_NOT_ENOUGH_MEMORY);
return NULL;
}
*CallProcPtr = NewCallProc;
if (Class->Desktop == NULL)
{
if (UseCallProc2)
Class->GlobalCallProc2 = TRUE;
else
Class->GlobalCallProc = TRUE;
}
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
if (UseCallProc2)
{
Class->CallProc2 = NewCallProc;
Class->GlobalCallProc2 = BaseClass->GlobalCallProc2;
}
else
{
Class->CallProc = NewCallProc;
Class->GlobalCallProc = BaseClass->GlobalCallProc;
}
Class = Class->Next;
}
return GetCallProcHandle(NewCallProc);
}
}
}
}
static WNDPROC
IntSetClassWndProc(IN OUT PWINDOWCLASS Class,
IN WNDPROC WndProc,
IN BOOL Ansi)
{
WNDPROC Ret;
if (Class->System)
{
DPRINT1("Attempted to change window procedure of system window class 0x%p!\n", Class->Atom);
SetLastWin32Error(ERROR_ACCESS_DENIED);
return NULL;
}
/* update the base class first */
Class = Class->Base;
/* resolve any callproc handle if possible */
if (IsCallProcHandle(WndProc))
{
WNDPROC_INFO wpInfo;
if (UserGetCallProcInfo((HANDLE)WndProc,
&wpInfo))
{
WndProc = wpInfo.WindowProc;
/* FIXME - what if wpInfo.IsUnicode doesn't match Ansi? */
}
}
Ret = IntGetClassWndProc(Class,
GetW32ProcessInfo(),
Ansi,
TRUE);
if (Ret == NULL)
{
return NULL;
}
/* update the class info */
Class->Unicode = !Ansi;
Class->WndProc = WndProc;
if (Class->CallProc != NULL)
{
Class->CallProc->WndProc = WndProc;
Class->CallProc->Unicode = !Ansi;
}
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
Class->Unicode = !Ansi;
Class->WndProc = WndProc;
Class = Class->Next;
}
return Ret;
}
static PWINDOWCLASS
IntGetClassForDesktop(IN OUT PWINDOWCLASS BaseClass,
IN OUT PWINDOWCLASS *ClassLink,
IN PDESKTOP Desktop)
{
SIZE_T ClassSize;
PWINDOWCLASS Class;
ASSERT(Desktop != NULL);
ASSERT(BaseClass->Base == BaseClass);
if (BaseClass->Desktop == Desktop)
{
/* it is most likely that a window is created on the same
desktop as the window class. */
return BaseClass;
}
if (BaseClass->Desktop == NULL)
{
ASSERT(BaseClass->Windows == 0);
ASSERT(BaseClass->Clone == NULL);
/* Classes are also located in the shared heap when the class
was created before the thread attached to a desktop. As soon
as a window is created for such a class located on the shared
heap, the class is cloned into the desktop heap on which the
window is created. */
Class = NULL;
}
else
{
/* The user is asking for a class object on a different desktop,
try to find one! */
Class = BaseClass->Clone;
while (Class != NULL)
{
if (Class->Desktop == Desktop)
{
ASSERT(Class->Base == BaseClass);
ASSERT(Class->Clone == NULL);
break;
}
Class = Class->Next;
}
}
if (Class == NULL)
{
/* The window is created on a different desktop, we need to
clone the class object to the desktop heap of the window! */
ClassSize = (SIZE_T)BaseClass->ClassExtraDataOffset +
(SIZE_T)BaseClass->ClsExtra;
Class = DesktopHeapAlloc(Desktop,
ClassSize);
if (Class != NULL)
{
/* simply clone the class */
RtlCopyMemory(Class,
BaseClass,
ClassSize);
/* update some pointers and link the class */
Class->Desktop = Desktop;
Class->Windows = 0;
if (BaseClass->Desktop == NULL)
{
/* we don't really need the base class on the shared
heap anymore, delete it so the only class left is
the clone we just created, which now serves as the
new base class */
ASSERT(BaseClass->Clone == NULL);
ASSERT(Class->Clone == NULL);
Class->Base = Class;
Class->Next = BaseClass->Next;
if (!BaseClass->System && BaseClass->CallProc != NULL)
Class->GlobalCallProc = TRUE;
if (BaseClass->CallProc2 != NULL)
Class->GlobalCallProc2 = TRUE;
/* replace the base class */
(void)InterlockedExchangePointer(ClassLink,
Class);
/* destroy the obsolete copy on the shared heap */
BaseClass->Base = NULL;
BaseClass->Clone = NULL;
IntDestroyClass(BaseClass);
}
else
{
/* link in the clone */
Class->Clone = NULL;
Class->Base = BaseClass;
Class->Next = BaseClass->Clone;
(void)InterlockedExchangePointer(&BaseClass->Clone,
Class);
}
}
else
{
SetLastWin32Error(ERROR_NOT_ENOUGH_MEMORY);
}
}
return Class;
}
PWINDOWCLASS
IntReferenceClass(IN OUT PWINDOWCLASS BaseClass,
IN OUT PWINDOWCLASS *ClassLink,
IN PDESKTOP Desktop)
{
PWINDOWCLASS Class;
ASSERT(BaseClass->Base == BaseClass);
Class = IntGetClassForDesktop(BaseClass,
ClassLink,
Desktop);
if (Class != NULL)
{
Class->Windows++;
}
return Class;
}
static VOID
IntMakeCloneBaseClass(IN OUT PWINDOWCLASS Class,
IN OUT PWINDOWCLASS *BaseClassLink,
IN OUT PWINDOWCLASS *CloneLink)
{
PWINDOWCLASS Clone, BaseClass;
PCALLPROC CallProc;
ASSERT(Class->Base != Class);
ASSERT(Class->Base->Clone != NULL);
ASSERT(Class->Desktop != NULL);
ASSERT(Class->Windows != 0);
ASSERT(Class->Base->Desktop != NULL);
ASSERT(Class->Base->Windows == 0);
/* unlink the clone */
*CloneLink = Class->Next;
Class->Clone = Class->Base->Clone;
BaseClass = Class->Base;
if (!BaseClass->System && BaseClass->CallProc != NULL &&
!BaseClass->GlobalCallProc)
{
/* we need to move the allocated call procedure */
CallProc = BaseClass->CallProc;
Class->CallProc = CloneCallProc(Class->Desktop,
CallProc);
DestroyCallProc(BaseClass->Desktop,
CallProc);
}
if (BaseClass->CallProc2 != NULL &&
!BaseClass->GlobalCallProc2)
{
/* we need to move the allocated call procedure */
CallProc = BaseClass->CallProc2;
Class->CallProc2 = CloneCallProc(Class->Desktop,
CallProc);
DestroyCallProc(BaseClass->Desktop,
CallProc);
}
/* update the class information to make it a base class */
Class->Base = Class;
Class->Next = (*BaseClassLink)->Next;
/* update all clones */
Clone = Class->Clone;
while (Clone != NULL)
{
ASSERT(Clone->Clone == NULL);
Clone->Base = Class;
if (!Class->System)
Clone->CallProc = Class->CallProc;
Clone->CallProc2 = Class->CallProc2;
Clone = Clone->Next;
}
/* link in the new base class */
(void)InterlockedExchangePointer(BaseClassLink,
Class);
}
VOID
IntDereferenceClass(IN OUT PWINDOWCLASS Class,
IN PDESKTOP Desktop,
IN PW32PROCESSINFO pi)
{
PWINDOWCLASS *PrevLink, BaseClass, CurrentClass;
BaseClass = Class->Base;
if (--Class->Windows == 0)
{
if (BaseClass == Class)
{
ASSERT(Class->Base == Class);
/* check if there are clones of the class on other desktops,
link the first clone in if possible. If there are no clones
then leave the class on the desktop heap. It will get moved
to the shared heap when the thread detaches. */
if (BaseClass->Clone != NULL)
{
if (BaseClass->System)
PrevLink = &pi->SystemClassList;
else if (BaseClass->Global)
PrevLink = &pi->GlobalClassList;
else
PrevLink = &pi->LocalClassList;
while (*PrevLink != BaseClass)
{
ASSERT(BaseClass != NULL);
PrevLink = &BaseClass->Next;
}
ASSERT(*PrevLink == BaseClass);
/* make the first clone become the new base class */
IntMakeCloneBaseClass(BaseClass->Clone,
PrevLink,
&BaseClass->Clone);
/* destroy the class, there's still another clone of the class
that now serves as a base class. Make sure we don't destruct
resources shared by all classes (Base = NULL)! */
BaseClass->Base = NULL;
BaseClass->Clone = NULL;
IntDestroyClass(BaseClass);
}
}
else
{
/* locate the cloned class and unlink it */
PrevLink = &BaseClass->Clone;
CurrentClass = BaseClass->Clone;
while (CurrentClass != Class)
{
ASSERT(CurrentClass != NULL);
PrevLink = &CurrentClass->Next;
CurrentClass = CurrentClass->Next;
}
ASSERT(CurrentClass == Class);
(void)InterlockedExchangePointer(PrevLink,
Class->Next);
ASSERT(Class->Base == BaseClass);
ASSERT(Class->Clone == NULL);
/* the class was just a clone, we don't need it anymore */
IntDestroyClass(Class);
}
}
}
static BOOL
IntMoveClassToSharedHeap(IN OUT PWINDOWCLASS Class,
IN OUT PWINDOWCLASS **ClassLinkPtr)
{
PWINDOWCLASS NewClass;
PCALLPROC CallProc;
SIZE_T ClassSize;
ASSERT(Class->Base == Class);
ASSERT(Class->Desktop != NULL);
ASSERT(Class->Windows == 0);
ASSERT(Class->Clone == NULL);
ClassSize = (SIZE_T)Class->ClassExtraDataOffset +
(SIZE_T)Class->ClsExtra;
/* allocate the new base class on the shared heap */
NewClass = UserHeapAlloc(ClassSize);
if (NewClass != NULL)
{
RtlCopyMemory(NewClass,
Class,
ClassSize);
NewClass->Desktop = NULL;
NewClass->Base = NewClass;
if (Class->MenuName == (PWSTR)(Class + 1))
{
ULONG_PTR AnsiDelta = (ULONG_PTR)Class->AnsiMenuName - (ULONG_PTR)Class->MenuName;
/* fixup the self-relative MenuName pointers */
NewClass->MenuName = (PWSTR)(NewClass + 1);
NewClass->AnsiMenuName = (PSTR)((ULONG_PTR)NewClass->MenuName + AnsiDelta);
}
if (!NewClass->System && NewClass->CallProc != NULL &&
!NewClass->GlobalCallProc)
{
/* we need to move the allocated call procedure to the shared heap */
CallProc = NewClass->CallProc;
NewClass->CallProc = CloneCallProc(NULL,
CallProc);
DestroyCallProc(Class->Desktop,
CallProc);
NewClass->GlobalCallProc = TRUE;
}
if (NewClass->CallProc2 != NULL &&
!NewClass->GlobalCallProc2)
{
/* we need to move the allocated call procedure to the shared heap */
CallProc = NewClass->CallProc2;
NewClass->CallProc2 = CloneCallProc(NULL,
CallProc);
DestroyCallProc(Class->Desktop,
CallProc);
NewClass->GlobalCallProc2 = TRUE;
}
/* replace the class in the list */
(void)InterlockedExchangePointer(*ClassLinkPtr,
NewClass);
*ClassLinkPtr = &NewClass->Next;
/* free the obsolete class on the desktop heap */
Class->Base = NULL;
IntDestroyClass(Class);
return TRUE;
}
return FALSE;
}
static VOID
IntCheckDesktopClasses(IN PDESKTOP Desktop,
IN OUT PWINDOWCLASS *ClassList,
IN BOOL FreeOnFailure,
OUT BOOL *Ret)
{
PWINDOWCLASS Class, NextClass, *Link;
/* NOTE: We only need to check base classes! When classes are no longer needed
on a desktop, the clones will be freed automatically as soon as possible.
However, we need to move base classes to the shared heap, as soon as
the last desktop heap where a class is allocated on is about to be destroyed.
If we didn't move the class to the shared heap, the class would become
inaccessible! */
ASSERT(Desktop != NULL);
Link = ClassList;
Class = *Link;
while (Class != NULL)
{
NextClass = Class->Next;
ASSERT(Class->Base == Class);
if (Class->Desktop == Desktop &&
Class->Windows == 0)
{
/* there shouldn't be any clones around anymore! */
ASSERT(Class->Clone == NULL);
/* FIXME - If process is terminating, don't move the class but rather destroy it! */
/* FIXME - We could move the class to another desktop heap if there's still desktops
mapped into the process... */
/* move the class to the shared heap */
if (IntMoveClassToSharedHeap(Class,
&Link))
{
ASSERT(*Link == NextClass);
}
else
{
ASSERT(NextClass == Class->Next);
if (FreeOnFailure)
{
/* unlink the base class */
(void)InterlockedExchangePointer(Link,
Class->Next);
/* we can free the old base class now */
Class->Base = NULL;
IntDestroyClass(Class);
}
else
{
Link = &Class->Next;
*Ret = FALSE;
}
}
}
else
Link = &Class->Next;
Class = NextClass;
}
}
BOOL
IntCheckProcessDesktopClasses(IN PDESKTOP Desktop,
IN BOOL FreeOnFailure)
{
PW32PROCESSINFO pi;
BOOL Ret = TRUE;
pi = GetW32ProcessInfo();
if (pi == NULL)
return TRUE;
/* check all local classes */
IntCheckDesktopClasses(Desktop,
&pi->LocalClassList,
FreeOnFailure,
&Ret);
/* check all global classes */
IntCheckDesktopClasses(Desktop,
&pi->GlobalClassList,
FreeOnFailure,
&Ret);
/* check all system classes */
IntCheckDesktopClasses(Desktop,
&pi->SystemClassList,
FreeOnFailure,
&Ret);
if (!Ret)
{
DPRINT1("Failed to move process classes from desktop 0x%p to the shared heap!\n", Desktop);
SetLastWin32Error(ERROR_NOT_ENOUGH_MEMORY);
}
return Ret;
}
static PWINDOWCLASS
IntCreateClass(IN CONST WNDCLASSEXW* lpwcx,
IN PUNICODE_STRING ClassName,
IN PUNICODE_STRING MenuName,
IN WNDPROC wpExtra,
IN DWORD dwFlags,
IN PDESKTOP Desktop,
IN PW32PROCESSINFO pi)
{
SIZE_T ClassSize;
PWINDOWCLASS Class = NULL;
RTL_ATOM Atom;
NTSTATUS Status = STATUS_SUCCESS;
TRACE("lpwcx=%p ClassName=%wZ MenuName=%wZ wpExtra=%p dwFlags=%08x Desktop=%p pi=%p\n",
lpwcx, ClassName, MenuName, wpExtra, dwFlags, Desktop, pi);
if (!IntRegisterClassAtom(ClassName,
&Atom))
{
DPRINT1("Failed to register class atom!\n");
return NULL;
}
ClassSize = sizeof(WINDOWCLASS) + lpwcx->cbClsExtra;
if (MenuName->Length != 0)
{
ClassSize += MenuName->Length + sizeof(UNICODE_NULL);
ClassSize += RtlUnicodeStringToAnsiSize(MenuName);
}
if (Desktop != NULL)
{
Class = DesktopHeapAlloc(Desktop,
ClassSize);
}
else
{
/* FIXME - the class was created before being connected
to a desktop. It is possible for the desktop window,
but should it be allowed for any other case? */
Class = UserHeapAlloc(ClassSize);
}
if (Class != NULL)
{
RtlZeroMemory(Class,
sizeof(ClassSize));
Class->Desktop = Desktop;
Class->Base = Class;
Class->Atom = Atom;
if (dwFlags & REGISTERCLASS_SYSTEM)
{
dwFlags &= ~REGISTERCLASS_ANSI;
Class->WndProcExtra = wpExtra;
Class->System = TRUE;
}
_SEH_TRY
{
PWSTR strBuf;
PSTR strBufA;
ANSI_STRING AnsiString;
/* need to protect with SEH since accessing the WNDCLASSEX structure
and string buffers might raise an exception! We don't want to
leak memory... */
Class->WndProc = lpwcx->lpfnWndProc;
Class->Style = lpwcx->style;
Class->ClsExtra = lpwcx->cbClsExtra;
Class->WndExtra = lpwcx->cbWndExtra;
Class->hInstance = lpwcx->hInstance;
Class->hIcon = lpwcx->hIcon; /* FIXME */
Class->hIconSm = lpwcx->hIconSm; /* FIXME */
Class->hCursor = lpwcx->hCursor; /* FIXME */
Class->hbrBackground = lpwcx->hbrBackground;
/* make a copy of the string */
strBuf = (PWSTR)(Class + 1);
if (MenuName->Length != 0)
{
Class->MenuName = strBuf;
RtlCopyMemory(Class->MenuName,
MenuName->Buffer,
MenuName->Length);
strBuf += (MenuName->Length / sizeof(WCHAR)) + 1;
}
else
Class->MenuName = MenuName->Buffer;
/* save an ansi copy of the string */
strBufA = (PSTR)strBuf;
if (MenuName->Length != 0)
{
Class->AnsiMenuName = strBufA;
AnsiString.MaximumLength = RtlUnicodeStringToAnsiSize(MenuName);
AnsiString.Buffer = strBufA;
Status = RtlUnicodeStringToAnsiString(&AnsiString,
MenuName,
FALSE);
if (!NT_SUCCESS(Status))
{
DPRINT1("Failed to convert unicode menu name to ansi!\n");
/* life would've been much prettier if ntoskrnl exported RtlRaiseStatus()... */
_SEH_LEAVE;
}
strBufA += AnsiString.Length + 1;
}
else
Class->AnsiMenuName = (PSTR)MenuName->Buffer;
/* calculate the offset of the extra data */
Class->ClassExtraDataOffset = (ULONG_PTR)strBufA - (ULONG_PTR)Class;
if (!(dwFlags & REGISTERCLASS_ANSI))
Class->Unicode = TRUE;
if (Class->Style & CS_GLOBALCLASS)
Class->Global = TRUE;
}
_SEH_HANDLE
{
Status = _SEH_GetExceptionCode();
}
_SEH_END;
if (!NT_SUCCESS(Status))
{
DPRINT1("Failed creating the class: 0x%x\n", Status);
SetLastNtError(Status);
DesktopHeapFree(Desktop,
Class);
Class = NULL;
IntDeregisterClassAtom(Atom);
}
}
else
{
DPRINT1("Failed to allocate class on Desktop 0x%p\n", Desktop);
IntDeregisterClassAtom(Atom);
SetLastWin32Error(ERROR_NOT_ENOUGH_MEMORY);
}
return Class;
}
static PWINDOWCLASS
IntFindClass(IN RTL_ATOM Atom,
IN HINSTANCE hInstance,
IN PWINDOWCLASS *ClassList,
OUT PWINDOWCLASS **Link OPTIONAL)
{
PWINDOWCLASS Class, *PrevLink = ClassList;
Class = *PrevLink;
while (Class != NULL)
{
if (Class->Atom == Atom &&
(hInstance == NULL || Class->hInstance == hInstance) &&
!Class->Destroying)
{
ASSERT(Class->Base == Class);
if (Link != NULL)
*Link = PrevLink;
break;
}
PrevLink = &Class->Next;
Class = Class->Next;
}
return Class;
}
BOOL
IntGetAtomFromStringOrAtom(IN PUNICODE_STRING ClassName,
OUT RTL_ATOM *Atom)
{
BOOL Ret = FALSE;
if (ClassName->Length != 0)
{
WCHAR szBuf[65];
PWSTR AtomName;
NTSTATUS Status;
/* NOTE: Caller has to protect the call with SEH! */
if (ClassName->Length != 0)
{
/* FIXME - Don't limit to 64 characters! use SEH when allocating memory! */
if (ClassName->Length / sizeof(WCHAR) >= sizeof(szBuf) / sizeof(szBuf[0]))
{
SetLastWin32Error(ERROR_INVALID_PARAMETER);
return (RTL_ATOM)0;
}
/* We need to make a local copy of the class name! The caller could
modify the buffer and we could overflow in RtlLookupAtomInAtomTable.
We're protected by SEH, but the ranges that might be accessed were
not probed... */
RtlCopyMemory(szBuf,
ClassName->Buffer,
ClassName->Length);
szBuf[ClassName->Length / sizeof(WCHAR)] = UNICODE_NULL;
AtomName = szBuf;
}
else
AtomName = ClassName->Buffer;
/* lookup the atom */
Status = RtlLookupAtomInAtomTable(gAtomTable,
AtomName,
Atom);
if (NT_SUCCESS(Status))
{
Ret = TRUE;
}
else
{
if (Status == STATUS_OBJECT_NAME_NOT_FOUND)
{
SetLastWin32Error(ERROR_CANNOT_FIND_WND_CLASS);
}
else
{
SetLastNtError(Status);
}
}
}
else
{
ASSERT(IS_ATOM(ClassName->Buffer));
*Atom = (RTL_ATOM)((ULONG_PTR)ClassName->Buffer);
Ret = TRUE;
}
return Ret;
}
RTL_ATOM
IntGetClassAtom(IN PUNICODE_STRING ClassName,
IN HINSTANCE hInstance OPTIONAL,
IN PW32PROCESSINFO pi OPTIONAL,
OUT PWINDOWCLASS *BaseClass OPTIONAL,
OUT PWINDOWCLASS **Link OPTIONAL)
{
RTL_ATOM Atom = (RTL_ATOM)0;
if (IntGetAtomFromStringOrAtom(ClassName,
&Atom) &&
BaseClass != NULL && Atom != (RTL_ATOM)0)
{
PWINDOWCLASS Class;
/* attempt to locate the class object */
ASSERT(pi != NULL);
/* Step 1: try to find an exact match of locally registered classes */
Class = IntFindClass(Atom,
hInstance,
&pi->LocalClassList,
Link);
if (Class != NULL)
{
goto FoundClass;
}
/* Step 2: try to find any globally registered class. The hInstance
is not relevant for global classes */
Class = IntFindClass(Atom,
NULL,
&pi->GlobalClassList,
Link);
if (Class != NULL)
{
goto FoundClass;
}
/* Step 3: try to find a system class */
Class = IntFindClass(Atom,
NULL,
&pi->SystemClassList,
Link);
if (Class == NULL)
{
SetLastWin32Error(ERROR_CLASS_DOES_NOT_EXIST);
return (RTL_ATOM)0;
}
FoundClass:
*BaseClass = Class;
}
return Atom;
}
RTL_ATOM
UserRegisterClass(IN CONST WNDCLASSEXW* lpwcx,
IN PUNICODE_STRING ClassName,
IN PUNICODE_STRING MenuName,
IN HANDLE hMenu, /* FIXME */
IN WNDPROC wpExtra,
IN DWORD dwFlags)
{
PW32THREADINFO ti;
PW32PROCESSINFO pi;
PWINDOWCLASS Class;
RTL_ATOM ClassAtom;
RTL_ATOM Ret = (RTL_ATOM)0;
/* NOTE: Accessing the buffers in ClassName and MenuName may raise exceptions! */
ti = GetW32ThreadInfo();
if (ti == NULL)
{
SetLastWin32Error(ERROR_NOT_ENOUGH_MEMORY);
return (RTL_ATOM)0;
}
pi = ti->kpi;
/* try to find a previously registered class */
ClassAtom = IntGetClassAtom(ClassName,
lpwcx->hInstance,
pi,
&Class,
NULL);
if (ClassAtom != (RTL_ATOM)0)
{
if (lpwcx->style & CS_GLOBALCLASS)
{
// global classes shall not have same names as system classes
if (Class->Global || Class->System)
{
DPRINT("Class 0x%p does already exist!\n", ClassAtom);
SetLastWin32Error(ERROR_CLASS_ALREADY_EXISTS);
return (RTL_ATOM)0;
}
}
else if ( !Class->Global && !Class->System)
{
// local class already exists
DPRINT("Class 0x%p does already exist!\n", ClassAtom);
SetLastWin32Error(ERROR_CLASS_ALREADY_EXISTS);
return (RTL_ATOM)0;
}
}
Class = IntCreateClass(lpwcx,
ClassName,
MenuName,
wpExtra,
dwFlags,
ti->Desktop,
pi);
if (Class != NULL)
{
PWINDOWCLASS *List;
/* FIXME - pass the PMENU pointer to IntCreateClass instead! */
Class->hMenu = hMenu;
/* Register the class */
if (Class->System)
List = &pi->SystemClassList;
else if (Class->Global)
List = &pi->GlobalClassList;
else
List = &pi->LocalClassList;
Class->Next = *List;
(void)InterlockedExchangePointer(List,
Class);
Ret = Class->Atom;
}
return Ret;
}
BOOL
UserUnregisterClass(IN PUNICODE_STRING ClassName,
IN HINSTANCE hInstance)
{
PWINDOWCLASS *Link;
PW32PROCESSINFO pi;
RTL_ATOM ClassAtom;
PWINDOWCLASS Class;
pi = GetW32ProcessInfo();
if (pi == NULL)
{
SetLastWin32Error(ERROR_NOT_ENOUGH_MEMORY);
return FALSE;
}
/* NOTE: Accessing the buffer in ClassName may raise an exception! */
ClassAtom = IntGetClassAtom(ClassName,
hInstance,
pi,
&Class,
&Link);
if (ClassAtom == (RTL_ATOM)0)
{
return FALSE;
}
ASSERT(Class != NULL);
if (Class->Windows != 0 ||
Class->Clone != NULL)
{
SetLastWin32Error(ERROR_CLASS_HAS_WINDOWS);
return FALSE;
}
/* must be a base class! */
ASSERT(Class->Base == Class);
/* unlink the class */
*Link = Class->Next;
if (NT_SUCCESS(IntDeregisterClassAtom(Class->Atom)))
{
/* finally free the resources */
IntDestroyClass(Class);
return TRUE;
}
return FALSE;
}
INT
UserGetClassName(IN PWINDOWCLASS Class,
IN OUT PUNICODE_STRING ClassName,
IN BOOL Ansi)
{
NTSTATUS Status = STATUS_SUCCESS;
WCHAR szStaticTemp[32];
PWSTR szTemp = NULL;
ULONG BufLen = sizeof(szStaticTemp);
INT Ret = 0;
/* Note: Accessing the buffer in ClassName may raise an exception! */
_SEH_TRY
{
if (Ansi)
{
PANSI_STRING AnsiClassName = (PANSI_STRING)ClassName;
UNICODE_STRING UnicodeClassName;
/* limit the size of the static buffer on the stack to the
size of the buffer provided by the caller */
if (BufLen / sizeof(WCHAR) > AnsiClassName->MaximumLength)
{
BufLen = AnsiClassName->MaximumLength * sizeof(WCHAR);
}
/* find out how big the buffer needs to be */
Status = RtlQueryAtomInAtomTable(gAtomTable,
Class->Atom,
NULL,
NULL,
szStaticTemp,
&BufLen);
if (Status == STATUS_BUFFER_TOO_SMALL)
{
if (BufLen / sizeof(WCHAR) > AnsiClassName->MaximumLength)
{
/* the buffer required exceeds the ansi buffer provided,
pretend like we're using the ansi buffer and limit the
size to the buffer size provided */
BufLen = AnsiClassName->MaximumLength * sizeof(WCHAR);
}
/* allocate a temporary buffer that can hold the unicode class name */
szTemp = ExAllocatePool(PagedPool,
BufLen);
if (szTemp == NULL)
{
SetLastWin32Error(ERROR_NOT_ENOUGH_MEMORY);
_SEH_LEAVE;
}
/* query the class name */
Status = RtlQueryAtomInAtomTable(gAtomTable,
Class->Atom,
NULL,
NULL,
szTemp,
&BufLen);
}
else
szTemp = szStaticTemp;
if (NT_SUCCESS(Status))
{
/* convert the atom name to ansi */
RtlInitUnicodeString(&UnicodeClassName,
szTemp);
Status = RtlUnicodeStringToAnsiString(AnsiClassName,
&UnicodeClassName,
FALSE);
if (!NT_SUCCESS(Status))
{
SetLastNtError(Status);
_SEH_LEAVE;
}
}
Ret = BufLen / sizeof(WCHAR);
}
else /* !Ansi */
{
BufLen = ClassName->MaximumLength;
/* query the atom name */
Status = RtlQueryAtomInAtomTable(gAtomTable,
Class->Atom,
NULL,
NULL,
ClassName->Buffer,
&BufLen);
if (!NT_SUCCESS(Status))
{
SetLastNtError(Status);
_SEH_LEAVE;
}
Ret = BufLen / sizeof(WCHAR);
}
}
_SEH_HANDLE
{
SetLastNtError(_SEH_GetExceptionCode());
}
_SEH_END;
if (Ansi && szTemp != NULL && szTemp != szStaticTemp)
{
ExFreePool(szTemp);
}
return Ret;
}
ULONG_PTR
UserGetClassLongPtr(IN PWINDOWCLASS Class,
IN INT Index,
IN BOOL Ansi)
{
ULONG_PTR Ret = 0;
if (Index >= 0)
{
PULONG_PTR Data;
if (Index + sizeof(ULONG_PTR) < Index ||
Index + sizeof(ULONG_PTR) > Class->ClsExtra)
{
SetLastWin32Error(ERROR_INVALID_PARAMETER);
return 0;
}
Data = (PULONG_PTR)((ULONG_PTR)Class + Class->ClassExtraDataOffset + Index);
/* FIXME - Data might be a unaligned pointer! Might be a problem on
certain architectures, maybe using RtlCopyMemory is a
better choice for those architectures! */
return *Data;
}
switch (Index)
{
case GCL_CBWNDEXTRA:
Ret = (ULONG_PTR)Class->WndExtra;
break;
case GCL_CBCLSEXTRA:
Ret = (ULONG_PTR)Class->ClsExtra;
break;
case GCLP_HBRBACKGROUND:
Ret = (ULONG_PTR)Class->hbrBackground;
break;
case GCLP_HCURSOR:
/* FIXME - get handle from pointer to CURSOR object */
Ret = (ULONG_PTR)Class->hCursor;
break;
case GCLP_HICON:
/* FIXME - get handle from pointer to ICON object */
Ret = (ULONG_PTR)Class->hIcon;
break;
case GCLP_HICONSM:
/* FIXME - get handle from pointer to ICON object */
Ret = (ULONG_PTR)Class->hIconSm;
break;
case GCLP_HMODULE:
Ret = (ULONG_PTR)Class->hInstance;
break;
case GCLP_MENUNAME:
/* NOTE: Returns pointer in kernel heap! */
if (Ansi)
Ret = (ULONG_PTR)Class->AnsiMenuName;
else
Ret = (ULONG_PTR)Class->MenuName;
break;
case GCL_STYLE:
Ret = (ULONG_PTR)Class->Style;
break;
case GCLP_WNDPROC:
Ret = (ULONG_PTR)IntGetClassWndProc(Class,
GetW32ProcessInfo(),
Ansi,
FALSE);
break;
case GCW_ATOM:
Ret = (ULONG_PTR)Class->Atom;
break;
default:
SetLastWin32Error(ERROR_INVALID_INDEX);
break;
}
return Ret;
}
static BOOL
IntSetClassMenuName(IN PWINDOWCLASS Class,
IN PUNICODE_STRING MenuName)
{
BOOL Ret = FALSE;
/* change the base class first */
Class = Class->Base;
if (MenuName->Length != 0)
{
ANSI_STRING AnsiString;
PWSTR strBufW;
AnsiString.MaximumLength = RtlUnicodeStringToAnsiSize(MenuName);
strBufW = UserHeapAlloc(MenuName->Length + sizeof(UNICODE_NULL) +
AnsiString.MaximumLength);
if (strBufW != NULL)
{
_SEH_TRY
{
NTSTATUS Status;
/* copy the unicode string */
RtlCopyMemory(strBufW,
MenuName->Buffer,
MenuName->Length);
strBufW[MenuName->Length / sizeof(WCHAR)] = UNICODE_NULL;
/* create an ansi copy of the string */
AnsiString.Buffer = (PSTR)(strBufW + (MenuName->Length / sizeof(WCHAR)) + 1);
Status = RtlUnicodeStringToAnsiString(&AnsiString,
MenuName,
FALSE);
if (!NT_SUCCESS(Status))
{
SetLastNtError(Status);
_SEH_LEAVE;
}
Ret = TRUE;
}
_SEH_HANDLE
{
SetLastNtError(_SEH_GetExceptionCode());
}
_SEH_END;
if (Ret)
{
/* update the base class */
IntFreeClassMenuName(Class);
Class->MenuName = strBufW;
Class->AnsiMenuName = AnsiString.Buffer;
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
Class->MenuName = strBufW;
Class->AnsiMenuName = AnsiString.Buffer;
Class = Class->Next;
}
}
else
{
DPRINT1("Failed to copy class menu name!\n");
UserHeapFree(strBufW);
}
}
else
SetLastWin32Error(ERROR_NOT_ENOUGH_MEMORY);
}
else
{
ASSERT(IS_INTRESOURCE(MenuName->Buffer));
/* update the base class */
IntFreeClassMenuName(Class);
Class->MenuName = MenuName->Buffer;
Class->AnsiMenuName = (PSTR)MenuName->Buffer;
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
Class->MenuName = MenuName->Buffer;
Class->AnsiMenuName = (PSTR)MenuName->Buffer;
Class = Class->Next;
}
Ret = TRUE;
}
return Ret;
}
ULONG_PTR
UserSetClassLongPtr(IN PWINDOWCLASS Class,
IN INT Index,
IN ULONG_PTR NewLong,
IN BOOL Ansi)
{
ULONG_PTR Ret = 0;
/* NOTE: For GCLP_MENUNAME and GCW_ATOM this function may raise an exception! */
/* change the information in the base class first, then update the clones */
Class = Class->Base;
if (Index >= 0)
{
PULONG_PTR Data;
if (Index + sizeof(ULONG_PTR) < Index ||
Index + sizeof(ULONG_PTR) > Class->ClsExtra)
{
SetLastWin32Error(ERROR_INVALID_PARAMETER);
return 0;
}
Data = (PULONG_PTR)((ULONG_PTR)Class + Class->ClassExtraDataOffset + Index);
/* FIXME - Data might be a unaligned pointer! Might be a problem on
certain architectures, maybe using RtlCopyMemory is a
better choice for those architectures! */
Ret = *Data;
*Data = NewLong;
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
*(PULONG_PTR)((ULONG_PTR)Class + Class->ClassExtraDataOffset + Index) = NewLong;
Class = Class->Next;
}
return Ret;
}
switch (Index)
{
case GCL_CBWNDEXTRA:
Ret = (ULONG_PTR)Class->WndExtra;
Class->WndExtra = (INT)NewLong;
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
Class->WndExtra = (INT)NewLong;
Class = Class->Next;
}
break;
case GCL_CBCLSEXTRA:
SetLastWin32Error(ERROR_INVALID_PARAMETER);
break;
case GCLP_HBRBACKGROUND:
Ret = (ULONG_PTR)Class->hbrBackground;
Class->hbrBackground = (HBRUSH)NewLong;
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
Class->hbrBackground = (HBRUSH)NewLong;
Class = Class->Next;
}
break;
case GCLP_HCURSOR:
/* FIXME - get handle from pointer to CURSOR object */
Ret = (ULONG_PTR)Class->hCursor;
Class->hCursor = (HANDLE)NewLong;
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
Class->hCursor = (HANDLE)NewLong;
Class = Class->Next;
}
break;
case GCLP_HICON:
/* FIXME - get handle from pointer to ICON object */
Ret = (ULONG_PTR)Class->hIcon;
Class->hIcon = (HANDLE)NewLong;
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
Class->hIcon = (HANDLE)NewLong;
Class = Class->Next;
}
break;
case GCLP_HICONSM:
/* FIXME - get handle from pointer to ICON object */
Ret = (ULONG_PTR)Class->hIconSm;
Class->hIconSm = (HANDLE)NewLong;
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
Class->hIconSm = (HANDLE)NewLong;
Class = Class->Next;
}
break;
case GCLP_HMODULE:
Ret = (ULONG_PTR)Class->hInstance;
Class->hInstance = (HINSTANCE)NewLong;
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
Class->hInstance = (HINSTANCE)NewLong;
Class = Class->Next;
}
break;
case GCLP_MENUNAME:
{
PUNICODE_STRING Value = (PUNICODE_STRING)NewLong;
if (!IntSetClassMenuName(Class,
Value))
{
DPRINT("Setting the class menu name failed!\n");
}
/* FIXME - really return NULL? Wine does so... */
break;
}
case GCL_STYLE:
Ret = (ULONG_PTR)Class->Style;
Class->Style = (UINT)NewLong;
/* FIXME - what if the CS_GLOBALCLASS style is changed? should we
move the class to the appropriate list? For now, we save
the original value in Class->Global, so we can always
locate the appropriate list */
/* update the clones */
Class = Class->Clone;
while (Class != NULL)
{
Class->Style = (UINT)NewLong;
Class = Class->Next;
}
break;
case GCLP_WNDPROC:
Ret = (ULONG_PTR)IntSetClassWndProc(Class,
(WNDPROC)NewLong,
Ansi);
break;
case GCW_ATOM:
{
PUNICODE_STRING Value = (PUNICODE_STRING)NewLong;
Ret = (ULONG_PTR)Class->Atom;
if (!IntSetClassAtom(Class,
Value))
{
Ret = 0;
}
break;
}
default:
SetLastWin32Error(ERROR_INVALID_INDEX);
break;
}
return Ret;
}
static BOOL
UserGetClassInfo(IN PWINDOWCLASS Class,
OUT PWNDCLASSEXW lpwcx,
IN BOOL Ansi,
HINSTANCE hInstance)
{
lpwcx->style = Class->Style;
lpwcx->lpfnWndProc = IntGetClassWndProc(Class,
GetW32ProcessInfo(),
Ansi,
FALSE);
lpwcx->cbClsExtra = Class->ClsExtra;
lpwcx->cbWndExtra = Class->WndExtra;
lpwcx->hIcon = Class->hIcon; /* FIXME - get handle from pointer */
lpwcx->hCursor = Class->hCursor; /* FIXME - get handle from pointer */
lpwcx->hbrBackground = Class->hbrBackground;
if (Ansi)
((PWNDCLASSEXA)lpwcx)->lpszMenuName = Class->AnsiMenuName;
else
lpwcx->lpszMenuName = Class->MenuName;
if (hInstance)
lpwcx->hInstance = hInstance;
else if (Class->Global)
lpwcx->hInstance = NULL;
else
lpwcx->hInstance = Class->hInstance;
lpwcx->lpszClassName = (LPCWSTR)((ULONG_PTR)Class->Atom); /* FIXME - return the string? */
lpwcx->hIconSm = Class->hIconSm; /* FIXME - get handle from pointer */
return TRUE;
}
/* SYSCALLS *****************************************************************/
RTL_ATOM NTAPI
NtUserRegisterClassEx(IN CONST WNDCLASSEXW* lpwcx,
IN PUNICODE_STRING ClassName,
IN PUNICODE_STRING MenuName,
IN WNDPROC wpExtra,
IN DWORD Flags,
IN HMENU hMenu)
/*
* FUNCTION:
* Registers a new class with the window manager
* ARGUMENTS:
* lpwcx = Win32 extended window class structure
* bUnicodeClass = Whether to send ANSI or unicode strings
* to window procedures
* wpExtra = Extra window procedure, if this is not null, its used for the second window procedure for standard controls.
* RETURNS:
* Atom identifying the new class
*/
{
WNDCLASSEXW CapturedClassInfo = {0};
UNICODE_STRING CapturedName = {0}, CapturedMenuName = {0};
RTL_ATOM Ret = (RTL_ATOM)0;
if (Flags & ~REGISTERCLASS_ALL)
{
SetLastWin32Error(ERROR_INVALID_FLAGS);
return Ret;
}
UserEnterExclusive();
_SEH_TRY
{
/* Probe the parameters and basic parameter checks */
if (ProbeForReadUint(&lpwcx->cbSize) != sizeof(WNDCLASSEXW))
{
goto InvalidParameter;
}
ProbeForRead(lpwcx,
sizeof(WNDCLASSEXW),
sizeof(ULONG));
RtlCopyMemory(&CapturedClassInfo,
lpwcx,
sizeof(WNDCLASSEXW));
CapturedName = ProbeForReadUnicodeString(ClassName);
CapturedMenuName = ProbeForReadUnicodeString(MenuName);
if (CapturedName.Length & 1 || CapturedMenuName.Length & 1 ||
CapturedClassInfo.cbClsExtra < 0 ||
CapturedClassInfo.cbClsExtra + CapturedName.Length +
CapturedMenuName.Length + sizeof(WINDOWCLASS) < CapturedClassInfo.cbClsExtra ||
CapturedClassInfo.cbWndExtra < 0 ||
CapturedClassInfo.hInstance == NULL)
{
goto InvalidParameter;
}
if (CapturedName.Length != 0)
{
ProbeForRead(CapturedName.Buffer,
CapturedName.Length,
sizeof(WCHAR));
}
else
{
if (!IS_ATOM(CapturedName.Buffer))
{
goto InvalidParameter;
}
}
if (CapturedMenuName.Length != 0)
{
ProbeForRead(CapturedMenuName.Buffer,
CapturedMenuName.Length,
sizeof(WCHAR));
}
else if (CapturedMenuName.Buffer != NULL &&
!IS_INTRESOURCE(CapturedMenuName.Buffer))
{
InvalidParameter:
SetLastWin32Error(ERROR_INVALID_PARAMETER);
_SEH_LEAVE;
}
/* Register the class */
Ret = UserRegisterClass(&CapturedClassInfo,
&CapturedName,
&CapturedMenuName,
hMenu, /* FIXME - pass pointer */
wpExtra,
Flags);
}
_SEH_HANDLE
{
SetLastNtError(_SEH_GetExceptionCode());
}
_SEH_END;
UserLeave();
return Ret;
}
ULONG_PTR NTAPI
NtUserGetClassLong(IN HWND hWnd,
IN INT Offset,
IN BOOL Ansi)
{
PWINDOW_OBJECT Window;
ULONG_PTR Ret = 0;
if (Offset != GCLP_WNDPROC)
{
UserEnterShared();
}
else
{
UserEnterExclusive();
}
Window = UserGetWindowObject(hWnd);
if (Window != NULL)
{
Ret = UserGetClassLongPtr(Window->Class,
Offset,
Ansi);
if (Ret != 0 && Offset == GCLP_MENUNAME && !IS_INTRESOURCE(Ret))
{
Ret = (ULONG_PTR)DesktopHeapAddressToUser(Window->Class->Desktop,
(PVOID)Ret);
}
}
UserLeave();
return Ret;
}
ULONG_PTR STDCALL
NtUserSetClassLong(HWND hWnd,
INT Offset,
ULONG_PTR dwNewLong,
BOOL Ansi)
{
PW32PROCESSINFO pi;
PWINDOW_OBJECT Window;
ULONG_PTR Ret = 0;
UserEnterExclusive();
pi = GetW32ProcessInfo();
if (pi == NULL)
goto Cleanup;
Window = UserGetWindowObject(hWnd);
if (Window != NULL)
{
if (Window->ti->kpi != pi)
{
SetLastWin32Error(ERROR_ACCESS_DENIED);
goto Cleanup;
}
_SEH_TRY
{
UNICODE_STRING Value;
/* probe the parameters */
if (Offset == GCW_ATOM || Offset == GCLP_MENUNAME)
{
Value = ProbeForReadUnicodeString((PUNICODE_STRING)dwNewLong);
if (Value.Length & 1)
{
goto InvalidParameter;
}
if (Value.Length != 0)
{
ProbeForRead(Value.Buffer,
Value.Length,
sizeof(WCHAR));
}
else
{
if (Offset == GCW_ATOM && !IS_ATOM(Value.Buffer))
{
goto InvalidParameter;
}
else if (Offset == GCLP_MENUNAME && !IS_INTRESOURCE(Value.Buffer))
{
InvalidParameter:
SetLastWin32Error(ERROR_INVALID_PARAMETER);
_SEH_LEAVE;
}
}
dwNewLong = (ULONG_PTR)&Value;
}
Ret = UserSetClassLongPtr(Window->Class,
Offset,
dwNewLong,
Ansi);
}
_SEH_HANDLE
{
SetLastNtError(_SEH_GetExceptionCode());
}
_SEH_END;
}
Cleanup:
UserLeave();
return Ret;
}
DWORD STDCALL
NtUserSetClassWord(DWORD Unknown0,
DWORD Unknown1,
DWORD Unknown2)
{
return(0);
}
BOOL NTAPI
NtUserUnregisterClass(IN PUNICODE_STRING ClassNameOrAtom,
IN HINSTANCE hInstance)
{
UNICODE_STRING CapturedClassName;
BOOL Ret = FALSE;
UserEnterExclusive();
_SEH_TRY
{
/* probe the paramters */
CapturedClassName = ProbeForReadUnicodeString(ClassNameOrAtom);
if (CapturedClassName.Length & 1)
{
goto InvalidParameter;
}
if (CapturedClassName.Length != 0)
{
ProbeForRead(CapturedClassName.Buffer,
CapturedClassName.Length,
sizeof(WCHAR));
}
else
{
if (!IS_ATOM(CapturedClassName.Buffer))
{
InvalidParameter:
SetLastWin32Error(ERROR_INVALID_PARAMETER);
_SEH_LEAVE;
}
}
/* unregister the class */
Ret = UserUnregisterClass(&CapturedClassName,
hInstance);
}
_SEH_HANDLE
{
SetLastNtError(_SEH_GetExceptionCode());
}
_SEH_END;
UserLeave();
return Ret;
}
/* NOTE: for system classes hInstance is not NULL here, but User32Instance */
BOOL STDCALL
NtUserGetClassInfo(
HINSTANCE hInstance,
PUNICODE_STRING ClassName,
LPWNDCLASSEXW lpWndClassEx,
BOOL Ansi)
{
UNICODE_STRING CapturedClassName;
PWINDOWCLASS Class;
RTL_ATOM ClassAtom;
PW32PROCESSINFO pi;
BOOL Ret = FALSE;
/* NOTE: need exclusive lock because getting the wndproc might require the
creation of a call procedure handle */
UserEnterExclusive();
pi = GetW32ProcessInfo();
if (pi == NULL)
{
ERR("GetW32ProcessInfo() returned NULL!\n");
goto Cleanup;
}
_SEH_TRY
{
/* probe the paramters */
CapturedClassName = ProbeForReadUnicodeString(ClassName);
if (IS_ATOM(CapturedClassName.Buffer))
TRACE("hInst %p atom %04X lpWndClassEx %p Ansi %d\n", hInstance, CapturedClassName.Buffer, lpWndClassEx, Ansi);
else
TRACE("hInst %p class %wZ lpWndClassEx %p Ansi %d\n", hInstance, &CapturedClassName, lpWndClassEx, Ansi);
if (CapturedClassName.Length & 1)
{
goto InvalidParameter;
}
if (CapturedClassName.Length != 0)
{
ProbeForRead(CapturedClassName.Buffer,
CapturedClassName.Length,
sizeof(WCHAR));
}
else
{
if (!IS_ATOM(CapturedClassName.Buffer))
{
ERR("NtUserGetClassInfo() got ClassName instead of Atom!\n");
goto InvalidParameter;
}
}
if (ProbeForReadUint(&lpWndClassEx->cbSize) != sizeof(WNDCLASSEXW))
{
InvalidParameter:
SetLastWin32Error(ERROR_INVALID_PARAMETER);
_SEH_LEAVE;
}
ProbeForWrite(lpWndClassEx,
sizeof(WNDCLASSEXW),
sizeof(ULONG));
ClassAtom = IntGetClassAtom(&CapturedClassName,
hInstance,
pi,
&Class,
NULL);
if (ClassAtom != (RTL_ATOM)0)
{
Ret = UserGetClassInfo(Class,
lpWndClassEx,
Ansi,
hInstance);
if (Ret)
{
lpWndClassEx->lpszClassName = CapturedClassName.Buffer;
/* FIXME - handle Class->Desktop == NULL!!!!! */
if (Class->MenuName != NULL &&
!IS_INTRESOURCE(Class->MenuName))
{
lpWndClassEx->lpszMenuName = DesktopHeapAddressToUser(Class->Desktop,
(Ansi ?
(PVOID)Class->AnsiMenuName :
(PVOID)Class->MenuName));
}
/* Undocumented behavior! Return the class atom as a BOOL! */
Ret = (BOOL)ClassAtom;
if (!(Class->Global || Class->System) && hInstance == NULL)
{
WARN("Tried to get information of a non-existing class\n");
SetLastWin32Error(ERROR_CLASS_DOES_NOT_EXIST);
Ret = FALSE;
}
}
}
else
{
WARN("Tried to get information of a non-existing class\n");
SetLastWin32Error(ERROR_CLASS_DOES_NOT_EXIST);
}
}
_SEH_HANDLE
{
SetLastWin32Error(ERROR_CLASS_DOES_NOT_EXIST);
}
_SEH_END;
Cleanup:
UserLeave();
return Ret;
}
INT NTAPI
NtUserGetClassName (IN HWND hWnd,
OUT PUNICODE_STRING ClassName,
IN BOOL Ansi)
{
PWINDOW_OBJECT Window;
UNICODE_STRING CapturedClassName;
INT Ret = 0;
UserEnterShared();
Window = UserGetWindowObject(hWnd);
if (Window != NULL)
{
_SEH_TRY
{
ProbeForWriteUnicodeString(ClassName);
CapturedClassName = *ClassName;
/* get the class name */
Ret = UserGetClassName(Window->Class,
&CapturedClassName,
Ansi);
if (Ret != 0)
{
/* update the Length field */
ClassName->Length = CapturedClassName.Length;
}
}
_SEH_HANDLE
{
SetLastNtError(_SEH_GetExceptionCode());
}
_SEH_END;
}
UserLeave();
return Ret;
}
DWORD STDCALL
NtUserGetWOWClass(DWORD Unknown0,
DWORD Unknown1)
{
return(0);
}
/* EOF */