mirror of
https://gitlab.winehq.org/wine/wine.git
synced 2025-08-29 02:33:58 +02:00
1032 lines
34 KiB
C
1032 lines
34 KiB
C
/*
|
|
* Window stations and desktops
|
|
*
|
|
* Copyright 2002 Alexandre Julliard
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#if 0
|
|
#pragma makedep unix
|
|
#endif
|
|
|
|
#include <assert.h>
|
|
#include <stdarg.h>
|
|
#include <stddef.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
#include "ntstatus.h"
|
|
#define WIN32_NO_STATUS
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "ntuser.h"
|
|
#include "ddk/wdm.h"
|
|
#include "ntgdi_private.h"
|
|
#include "ntuser_private.h"
|
|
#include "wine/server.h"
|
|
#include "wine/debug.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(winstation);
|
|
WINE_DECLARE_DEBUG_CHANNEL(win);
|
|
|
|
|
|
#define DESKTOP_ALL_ACCESS 0x01ff
|
|
|
|
struct shared_input_cache
|
|
{
|
|
const shared_object_t *object;
|
|
UINT64 id;
|
|
DWORD tid;
|
|
};
|
|
|
|
struct session_thread_data
|
|
{
|
|
const shared_object_t *shared_desktop; /* thread desktop shared session cached object */
|
|
const shared_object_t *shared_queue; /* thread message queue shared session cached object */
|
|
struct shared_input_cache shared_input; /* current thread input shared session cached object */
|
|
struct shared_input_cache shared_foreground; /* foreground thread input shared session cached object */
|
|
struct shared_input_cache other_thread_input; /* other thread input shared session cached object */
|
|
};
|
|
|
|
struct session_block
|
|
{
|
|
struct list entry; /* entry in the session block list */
|
|
const char *data; /* base pointer for the mmaped data */
|
|
SIZE_T offset; /* offset of data in the session shared mapping */
|
|
SIZE_T size; /* size of the mmaped data */
|
|
};
|
|
|
|
static pthread_mutex_t session_lock = PTHREAD_MUTEX_INITIALIZER;
|
|
static struct list session_blocks = LIST_INIT(session_blocks);
|
|
const session_shm_t *shared_session;
|
|
|
|
static struct session_thread_data *get_session_thread_data(void)
|
|
{
|
|
struct user_thread_info *thread_info = get_user_thread_info();
|
|
if (!thread_info->session_data) thread_info->session_data = calloc(1, sizeof(*thread_info->session_data));
|
|
return thread_info->session_data;
|
|
}
|
|
|
|
void shared_object_acquire_seqlock( const shared_object_t *object, UINT64 *seq )
|
|
{
|
|
while ((*seq = ReadNoFence64( &object->seq )) & 1) YieldProcessor();
|
|
__SHARED_READ_FENCE;
|
|
}
|
|
|
|
BOOL shared_object_release_seqlock( const shared_object_t *object, UINT64 seq )
|
|
{
|
|
__SHARED_READ_FENCE;
|
|
return ReadNoFence64( &object->seq ) == seq;
|
|
}
|
|
|
|
static object_id_t shared_object_get_id( const shared_object_t *object )
|
|
{
|
|
struct object_lock lock = OBJECT_LOCK_INIT;
|
|
do
|
|
{
|
|
shared_object_acquire_seqlock( object, &lock.seq );
|
|
lock.id = object->id;
|
|
} while (!shared_object_release_seqlock( object, lock.seq ));
|
|
return lock.id;
|
|
}
|
|
|
|
static NTSTATUS map_shared_session_block( SIZE_T offset, SIZE_T size, struct session_block **ret )
|
|
{
|
|
static const WCHAR nameW[] =
|
|
{
|
|
'\\','K','e','r','n','e','l','O','b','j','e','c','t','s','\\',
|
|
'_','_','w','i','n','e','_','s','e','s','s','i','o','n',0
|
|
};
|
|
UNICODE_STRING name = RTL_CONSTANT_STRING( nameW );
|
|
LARGE_INTEGER off = {.QuadPart = offset - (offset % system_info.AllocationGranularity)};
|
|
struct session_block *block;
|
|
OBJECT_ATTRIBUTES attr;
|
|
unsigned int status;
|
|
HANDLE handle;
|
|
|
|
assert( offset + size > offset );
|
|
|
|
if (!(block = calloc( 1, sizeof(*block) ))) return STATUS_NO_MEMORY;
|
|
|
|
InitializeObjectAttributes( &attr, &name, 0, NULL, NULL );
|
|
if ((status = NtOpenSection( &handle, SECTION_MAP_READ, &attr )))
|
|
WARN( "Failed to open shared session section, status %#x\n", status );
|
|
else
|
|
{
|
|
if ((status = NtMapViewOfSection( handle, GetCurrentProcess(), (void **)&block->data, 0, 0,
|
|
&off, &block->size, ViewUnmap, 0, PAGE_READONLY )))
|
|
WARN( "Failed to map shared session block, status %#x\n", status );
|
|
else
|
|
{
|
|
list_add_tail( &session_blocks, &block->entry );
|
|
block->offset = off.QuadPart;
|
|
assert( block->offset + block->size > block->offset );
|
|
}
|
|
NtClose( handle );
|
|
}
|
|
|
|
if (status) free( block );
|
|
else *ret = block;
|
|
return status;
|
|
}
|
|
|
|
static NTSTATUS find_shared_session_block( SIZE_T offset, SIZE_T size, struct session_block **ret )
|
|
{
|
|
struct session_block *block;
|
|
UINT status;
|
|
|
|
assert( offset + size > offset );
|
|
|
|
pthread_mutex_lock( &session_lock );
|
|
|
|
LIST_FOR_EACH_ENTRY( block, &session_blocks, struct session_block, entry )
|
|
{
|
|
if (block->offset < offset && offset + size <= block->offset + block->size)
|
|
{
|
|
*ret = block;
|
|
pthread_mutex_unlock( &session_lock );
|
|
return STATUS_SUCCESS;
|
|
}
|
|
}
|
|
|
|
if ((status = map_shared_session_block( offset, size, ret )))
|
|
{
|
|
WARN( "Failed to map session block for offset %s, size %s, status %#x\n",
|
|
wine_dbgstr_longlong(offset), wine_dbgstr_longlong(size), status );
|
|
}
|
|
|
|
pthread_mutex_unlock( &session_lock );
|
|
|
|
return status;
|
|
}
|
|
|
|
const shared_object_t *find_shared_session_object( object_id_t id, mem_size_t offset )
|
|
{
|
|
struct session_block *block = NULL;
|
|
const shared_object_t *object;
|
|
NTSTATUS status;
|
|
|
|
if (id && !(status = find_shared_session_block( offset, sizeof(*object), &block )))
|
|
{
|
|
object = (const shared_object_t *)(block->data + offset - block->offset);
|
|
if (id == shared_object_get_id( object )) return object;
|
|
WARN( "Session object id doesn't match expected id %s\n", wine_dbgstr_longlong(id) );
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void shared_session_init(void)
|
|
{
|
|
struct session_block *block;
|
|
UINT status;
|
|
|
|
if ((status = find_shared_session_block( 0, sizeof(*shared_session), &block )))
|
|
ERR( "Failed to map initial shared session block, status %#x\n", status );
|
|
else
|
|
shared_session = (const session_shm_t *)block->data;
|
|
}
|
|
|
|
NTSTATUS get_shared_desktop( struct object_lock *lock, const desktop_shm_t **desktop_shm )
|
|
{
|
|
struct session_thread_data *data = get_session_thread_data();
|
|
const shared_object_t *object;
|
|
|
|
TRACE( "lock %p, desktop_shm %p\n", lock, desktop_shm );
|
|
|
|
if (!(object = data->shared_desktop))
|
|
{
|
|
struct obj_locator locator;
|
|
|
|
SERVER_START_REQ( get_thread_desktop )
|
|
{
|
|
req->tid = GetCurrentThreadId();
|
|
wine_server_call( req );
|
|
locator = reply->locator;
|
|
}
|
|
SERVER_END_REQ;
|
|
|
|
data->shared_desktop = find_shared_session_object( locator.id, locator.offset );
|
|
if (!(object = data->shared_desktop)) return STATUS_INVALID_HANDLE;
|
|
}
|
|
|
|
if (!lock->id || !shared_object_release_seqlock( object, lock->seq ))
|
|
{
|
|
shared_object_acquire_seqlock( object, &lock->seq );
|
|
*desktop_shm = &object->shm.desktop;
|
|
lock->id = object->id;
|
|
return STATUS_PENDING;
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
NTSTATUS get_shared_queue( struct object_lock *lock, const queue_shm_t **queue_shm )
|
|
{
|
|
struct session_thread_data *data = get_session_thread_data();
|
|
const shared_object_t *object;
|
|
|
|
TRACE( "lock %p, queue_shm %p\n", lock, queue_shm );
|
|
|
|
if (!(object = data->shared_queue))
|
|
{
|
|
struct obj_locator locator;
|
|
|
|
SERVER_START_REQ( get_msg_queue )
|
|
{
|
|
wine_server_call( req );
|
|
locator = reply->locator;
|
|
}
|
|
SERVER_END_REQ;
|
|
|
|
data->shared_queue = find_shared_session_object( locator.id, locator.offset );
|
|
if (!(object = data->shared_queue)) return STATUS_INVALID_HANDLE;
|
|
}
|
|
|
|
if (!lock->id || !shared_object_release_seqlock( object, lock->seq ))
|
|
{
|
|
shared_object_acquire_seqlock( object, &lock->seq );
|
|
*queue_shm = &object->shm.queue;
|
|
lock->id = object->id;
|
|
return STATUS_PENDING;
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
static NTSTATUS try_get_shared_input( UINT tid, struct object_lock *lock, const input_shm_t **input_shm,
|
|
struct shared_input_cache *cache )
|
|
{
|
|
const shared_object_t *object;
|
|
BOOL valid = TRUE;
|
|
|
|
if (!(object = cache->object))
|
|
{
|
|
struct obj_locator locator;
|
|
|
|
SERVER_START_REQ( get_thread_input )
|
|
{
|
|
req->tid = tid;
|
|
wine_server_call( req );
|
|
locator = reply->locator;
|
|
}
|
|
SERVER_END_REQ;
|
|
|
|
cache->id = locator.id;
|
|
cache->object = find_shared_session_object( locator.id, locator.offset );
|
|
if (!(object = cache->object)) return STATUS_INVALID_HANDLE;
|
|
}
|
|
|
|
/* check object validity by comparing ids, within the object seqlock */
|
|
if ((valid = cache->id == object->id) && !tid)
|
|
{
|
|
/* check that a previously locked foreground thread input is still foreground */
|
|
valid = !!object->shm.input.foreground;
|
|
}
|
|
|
|
if (!lock->id || !shared_object_release_seqlock( object, lock->seq ))
|
|
{
|
|
shared_object_acquire_seqlock( object, &lock->seq );
|
|
if (!(lock->id = object->id)) lock->id = -1;
|
|
*input_shm = &object->shm.input;
|
|
return STATUS_PENDING;
|
|
}
|
|
|
|
if (!valid)
|
|
{
|
|
/* object has been invalidated, start over */
|
|
memset( cache, 0, sizeof(*cache) );
|
|
memset( lock, 0, sizeof(*lock) );
|
|
}
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
NTSTATUS get_shared_input( UINT tid, struct object_lock *lock, const input_shm_t **input_shm )
|
|
{
|
|
struct session_thread_data *data = get_session_thread_data();
|
|
struct shared_input_cache *cache;
|
|
UINT status;
|
|
|
|
TRACE( "tid %u, lock %p, input_shm %p\n", tid, lock, input_shm );
|
|
|
|
if (tid == GetCurrentThreadId()) cache = &data->shared_input;
|
|
else if (!tid) cache = &data->shared_foreground;
|
|
else cache = &data->other_thread_input;
|
|
|
|
if (tid != cache->tid) memset( cache, 0, sizeof(*cache) );
|
|
cache->tid = tid;
|
|
|
|
do { status = try_get_shared_input( tid, lock, input_shm, cache ); }
|
|
while (!status && !cache->id);
|
|
|
|
return status;
|
|
}
|
|
|
|
BOOL is_virtual_desktop(void)
|
|
{
|
|
struct object_lock lock = OBJECT_LOCK_INIT;
|
|
const desktop_shm_t *desktop_shm;
|
|
BOOL ret = FALSE;
|
|
UINT status;
|
|
|
|
while ((status = get_shared_desktop( &lock, &desktop_shm )) == STATUS_PENDING)
|
|
ret = !!(desktop_shm->flags & DF_WINE_VIRTUAL_DESKTOP);
|
|
if (status) ret = FALSE;
|
|
|
|
return ret;
|
|
}
|
|
|
|
BOOL is_service_process(void)
|
|
{
|
|
static const WCHAR wine_service_station_name[] = {'_','_','w','i','n','e','s','e','r','v','i','c','e','_','w','i','n','s','t','a','t','i','o','n',0};
|
|
WCHAR name[MAX_PATH];
|
|
|
|
return NtUserGetObjectInformation( NtUserGetProcessWindowStation(), UOI_NAME, name, sizeof(name), NULL ) &&
|
|
!wcscmp( name, wine_service_station_name );
|
|
}
|
|
|
|
/***********************************************************************
|
|
* NtUserCreateWindowStation (win32u.@)
|
|
*/
|
|
HWINSTA WINAPI NtUserCreateWindowStation( OBJECT_ATTRIBUTES *attr, ACCESS_MASK access, ULONG arg3,
|
|
ULONG arg4, ULONG arg5, ULONG arg6, ULONG arg7 )
|
|
{
|
|
HANDLE ret;
|
|
|
|
if (attr->ObjectName->Length >= MAX_PATH * sizeof(WCHAR))
|
|
{
|
|
RtlSetLastWin32Error( ERROR_FILENAME_EXCED_RANGE );
|
|
return 0;
|
|
}
|
|
|
|
SERVER_START_REQ( create_winstation )
|
|
{
|
|
req->flags = 0;
|
|
req->access = access;
|
|
req->attributes = attr->Attributes;
|
|
req->rootdir = wine_server_obj_handle( attr->RootDirectory );
|
|
wine_server_add_data( req, attr->ObjectName->Buffer, attr->ObjectName->Length );
|
|
wine_server_call_err( req );
|
|
ret = wine_server_ptr_handle( reply->handle );
|
|
}
|
|
SERVER_END_REQ;
|
|
return ret;
|
|
}
|
|
|
|
/******************************************************************************
|
|
* NtUserOpenWindowStation (win32u.@)
|
|
*/
|
|
HWINSTA WINAPI NtUserOpenWindowStation( OBJECT_ATTRIBUTES *attr, ACCESS_MASK access )
|
|
{
|
|
HANDLE ret = 0;
|
|
|
|
SERVER_START_REQ( open_winstation )
|
|
{
|
|
req->access = access;
|
|
req->attributes = attr->Attributes;
|
|
req->rootdir = wine_server_obj_handle( attr->RootDirectory );
|
|
wine_server_add_data( req, attr->ObjectName->Buffer, attr->ObjectName->Length );
|
|
if (!wine_server_call_err( req )) ret = wine_server_ptr_handle( reply->handle );
|
|
}
|
|
SERVER_END_REQ;
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* NtUserCloseWindowStation (win32u.@)
|
|
*/
|
|
BOOL WINAPI NtUserCloseWindowStation( HWINSTA handle )
|
|
{
|
|
BOOL ret;
|
|
SERVER_START_REQ( close_winstation )
|
|
{
|
|
req->handle = wine_server_obj_handle( handle );
|
|
ret = !wine_server_call_err( req );
|
|
}
|
|
SERVER_END_REQ;
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* NtUSerGetProcessWindowStation (win32u.@)
|
|
*/
|
|
HWINSTA WINAPI NtUserGetProcessWindowStation(void)
|
|
{
|
|
HWINSTA ret = 0;
|
|
|
|
SERVER_START_REQ( get_process_winstation )
|
|
{
|
|
if (!wine_server_call_err( req ))
|
|
ret = wine_server_ptr_handle( reply->handle );
|
|
}
|
|
SERVER_END_REQ;
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* NtUserSetProcessWindowStation (win32u.@)
|
|
*/
|
|
BOOL WINAPI NtUserSetProcessWindowStation( HWINSTA handle )
|
|
{
|
|
BOOL ret;
|
|
|
|
SERVER_START_REQ( set_process_winstation )
|
|
{
|
|
req->handle = wine_server_obj_handle( handle );
|
|
ret = !wine_server_call_err( req );
|
|
}
|
|
SERVER_END_REQ;
|
|
|
|
reset_monitor_update_serial();
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* NtUserCreateDesktopEx (win32u.@)
|
|
*/
|
|
HDESK WINAPI NtUserCreateDesktopEx( OBJECT_ATTRIBUTES *attr, UNICODE_STRING *device,
|
|
DEVMODEW *devmode, DWORD flags, ACCESS_MASK access,
|
|
ULONG heap_size )
|
|
{
|
|
WCHAR buffer[MAX_PATH];
|
|
HANDLE ret;
|
|
|
|
if ((device && device->Length) || (devmode && !(flags & DF_WINE_VIRTUAL_DESKTOP)))
|
|
{
|
|
RtlSetLastWin32Error( ERROR_INVALID_PARAMETER );
|
|
return 0;
|
|
}
|
|
if (attr->ObjectName->Length >= MAX_PATH * sizeof(WCHAR))
|
|
{
|
|
RtlSetLastWin32Error( ERROR_FILENAME_EXCED_RANGE );
|
|
return 0;
|
|
}
|
|
SERVER_START_REQ( create_desktop )
|
|
{
|
|
req->flags = flags;
|
|
req->access = access;
|
|
req->attributes = attr->Attributes;
|
|
wine_server_add_data( req, attr->ObjectName->Buffer, attr->ObjectName->Length );
|
|
wine_server_call_err( req );
|
|
ret = wine_server_ptr_handle( reply->handle );
|
|
}
|
|
SERVER_END_REQ;
|
|
if (!devmode) return ret;
|
|
|
|
lstrcpynW( buffer, attr->ObjectName->Buffer, attr->ObjectName->Length / sizeof(WCHAR) + 1 );
|
|
if (!user_driver->pCreateDesktop( buffer, devmode->dmPelsWidth, devmode->dmPelsHeight ))
|
|
{
|
|
NtUserCloseDesktop( ret );
|
|
return 0;
|
|
}
|
|
|
|
/* force update display cache to use virtual desktop display settings */
|
|
if (flags & DF_WINE_VIRTUAL_DESKTOP) update_display_cache( TRUE );
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* NtUserOpenDesktop (win32u.@)
|
|
*/
|
|
HDESK WINAPI NtUserOpenDesktop( OBJECT_ATTRIBUTES *attr, DWORD flags, ACCESS_MASK access )
|
|
{
|
|
HANDLE ret = 0;
|
|
|
|
access |= DESKTOP_READOBJECTS | DESKTOP_WRITEOBJECTS;
|
|
|
|
if (attr->ObjectName->Length >= MAX_PATH * sizeof(WCHAR))
|
|
{
|
|
RtlSetLastWin32Error( ERROR_FILENAME_EXCED_RANGE );
|
|
return 0;
|
|
}
|
|
|
|
SERVER_START_REQ( open_desktop )
|
|
{
|
|
req->winsta = wine_server_obj_handle( attr->RootDirectory );
|
|
req->flags = flags;
|
|
req->access = access;
|
|
req->attributes = attr->Attributes;
|
|
wine_server_add_data( req, attr->ObjectName->Buffer, attr->ObjectName->Length );
|
|
if (!wine_server_call_err( req )) ret = wine_server_ptr_handle( reply->handle );
|
|
}
|
|
SERVER_END_REQ;
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* NtUserCloseDesktop (win32u.@)
|
|
*/
|
|
BOOL WINAPI NtUserCloseDesktop( HDESK handle )
|
|
{
|
|
BOOL ret;
|
|
SERVER_START_REQ( close_desktop )
|
|
{
|
|
req->handle = wine_server_obj_handle( handle );
|
|
ret = !wine_server_call_err( req );
|
|
}
|
|
SERVER_END_REQ;
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* NtUserGetThreadDesktop (win32u.@)
|
|
*/
|
|
HDESK WINAPI NtUserGetThreadDesktop( DWORD thread )
|
|
{
|
|
HDESK ret = 0;
|
|
|
|
SERVER_START_REQ( get_thread_desktop )
|
|
{
|
|
req->tid = thread;
|
|
if (!wine_server_call_err( req )) ret = wine_server_ptr_handle( reply->handle );
|
|
}
|
|
SERVER_END_REQ;
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* NtUserSetThreadDesktop (win32u.@)
|
|
*/
|
|
BOOL WINAPI NtUserSetThreadDesktop( HDESK handle )
|
|
{
|
|
BOOL ret, was_virtual_desktop = is_virtual_desktop();
|
|
struct obj_locator locator;
|
|
|
|
SERVER_START_REQ( set_thread_desktop )
|
|
{
|
|
req->handle = wine_server_obj_handle( handle );
|
|
ret = !wine_server_call_err( req );
|
|
locator = reply->locator;
|
|
}
|
|
SERVER_END_REQ;
|
|
|
|
if (ret) /* reset the desktop windows */
|
|
{
|
|
struct user_thread_info *thread_info = get_user_thread_info();
|
|
struct session_thread_data *data = get_session_thread_data();
|
|
data->shared_desktop = find_shared_session_object( locator.id, locator.offset );
|
|
memset( &data->shared_foreground, 0, sizeof(data->shared_foreground) );
|
|
thread_info->client_info.top_window = 0;
|
|
thread_info->client_info.msg_window = 0;
|
|
if (was_virtual_desktop != is_virtual_desktop()) update_display_cache( TRUE );
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* NtUserOpenInputDesktop (win32u.@)
|
|
*/
|
|
HDESK WINAPI NtUserOpenInputDesktop( DWORD flags, BOOL inherit, ACCESS_MASK access )
|
|
{
|
|
HANDLE ret = 0;
|
|
|
|
TRACE( "(%x,%i,%x)\n", flags, inherit, access );
|
|
|
|
access |= DESKTOP_READOBJECTS | DESKTOP_WRITEOBJECTS;
|
|
|
|
if (flags)
|
|
FIXME( "partial stub flags %08x\n", flags );
|
|
|
|
SERVER_START_REQ( open_input_desktop )
|
|
{
|
|
req->flags = flags;
|
|
req->access = access;
|
|
req->attributes = inherit ? OBJ_INHERIT : 0;
|
|
if (!wine_server_call_err( req )) ret = wine_server_ptr_handle( reply->handle );
|
|
}
|
|
SERVER_END_REQ;
|
|
|
|
return ret;
|
|
}
|
|
|
|
BOOL WINAPI NtUserSwitchDesktop( HDESK desktop )
|
|
{
|
|
TRACE( "desktop %p\n", desktop );
|
|
|
|
SERVER_START_REQ( set_input_desktop )
|
|
{
|
|
req->handle = wine_server_obj_handle( desktop );
|
|
if (wine_server_call_err( req )) return FALSE;
|
|
}
|
|
SERVER_END_REQ;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************
|
|
* NtUserBuildNameList (win32u.@)
|
|
*/
|
|
NTSTATUS WINAPI NtUserBuildNameList( HWINSTA handle, ULONG size, struct ntuser_name_list *list,
|
|
ULONG *ret_size )
|
|
{
|
|
const ULONG header_size = offsetof( struct ntuser_name_list, strings[1] ); /* header + final null */
|
|
WCHAR *buffer;
|
|
NTSTATUS status;
|
|
ULONG count, result, total;
|
|
|
|
if (size <= header_size) return STATUS_INVALID_HANDLE;
|
|
if (!(buffer = malloc( size - header_size ))) return STATUS_NO_MEMORY;
|
|
|
|
SERVER_START_REQ( enum_winstation )
|
|
{
|
|
req->handle = wine_server_obj_handle( handle );
|
|
wine_server_set_reply( req, buffer, size - header_size );
|
|
status = wine_server_call( req );
|
|
result = wine_server_reply_size( reply );
|
|
total = reply->total;
|
|
count = reply->count;
|
|
}
|
|
SERVER_END_REQ;
|
|
|
|
if (!status || status == STATUS_BUFFER_TOO_SMALL)
|
|
{
|
|
list->size = header_size + result;
|
|
list->count = count;
|
|
memcpy( list->strings, buffer, result );
|
|
list->strings[result / sizeof(WCHAR)] = 0;
|
|
*ret_size = header_size + total;
|
|
}
|
|
return status;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* NtUserGetObjectInformation (win32u.@)
|
|
*/
|
|
BOOL WINAPI NtUserGetObjectInformation( HANDLE handle, INT index, void *info,
|
|
DWORD len, DWORD *needed )
|
|
{
|
|
BOOL ret;
|
|
|
|
static const WCHAR desktopW[] = {'D','e','s','k','t','o','p',0};
|
|
static const WCHAR window_stationW[] = {'W','i','n','d','o','w','S','t','a','t','i','o','n',0};
|
|
|
|
switch(index)
|
|
{
|
|
case UOI_FLAGS:
|
|
{
|
|
USEROBJECTFLAGS *obj_flags = info;
|
|
if (needed) *needed = sizeof(*obj_flags);
|
|
if (len < sizeof(*obj_flags))
|
|
{
|
|
RtlSetLastWin32Error( ERROR_BUFFER_OVERFLOW );
|
|
return FALSE;
|
|
}
|
|
SERVER_START_REQ( set_user_object_info )
|
|
{
|
|
req->handle = wine_server_obj_handle( handle );
|
|
req->flags = 0;
|
|
ret = !wine_server_call_err( req );
|
|
if (ret)
|
|
{
|
|
/* FIXME: inherit flag */
|
|
obj_flags->dwFlags = reply->old_obj_flags;
|
|
}
|
|
}
|
|
SERVER_END_REQ;
|
|
}
|
|
return ret;
|
|
|
|
case UOI_TYPE:
|
|
SERVER_START_REQ( set_user_object_info )
|
|
{
|
|
req->handle = wine_server_obj_handle( handle );
|
|
req->flags = 0;
|
|
ret = !wine_server_call_err( req );
|
|
if (ret)
|
|
{
|
|
size_t size = reply->is_desktop ? sizeof(desktopW) : sizeof(window_stationW);
|
|
if (needed) *needed = size;
|
|
if (len < size)
|
|
{
|
|
RtlSetLastWin32Error( ERROR_INSUFFICIENT_BUFFER );
|
|
ret = FALSE;
|
|
}
|
|
else memcpy( info, reply->is_desktop ? desktopW : window_stationW, size );
|
|
}
|
|
}
|
|
SERVER_END_REQ;
|
|
return ret;
|
|
|
|
case UOI_NAME:
|
|
{
|
|
WCHAR buffer[MAX_PATH];
|
|
SERVER_START_REQ( set_user_object_info )
|
|
{
|
|
req->handle = wine_server_obj_handle( handle );
|
|
req->flags = 0;
|
|
wine_server_set_reply( req, buffer, sizeof(buffer) - sizeof(WCHAR) );
|
|
ret = !wine_server_call_err( req );
|
|
if (ret)
|
|
{
|
|
size_t size = wine_server_reply_size( reply );
|
|
buffer[size / sizeof(WCHAR)] = 0;
|
|
size += sizeof(WCHAR);
|
|
if (needed) *needed = size;
|
|
if (len < size)
|
|
{
|
|
RtlSetLastWin32Error( ERROR_INSUFFICIENT_BUFFER );
|
|
ret = FALSE;
|
|
}
|
|
else memcpy( info, buffer, size );
|
|
}
|
|
}
|
|
SERVER_END_REQ;
|
|
}
|
|
return ret;
|
|
|
|
case UOI_USER_SID:
|
|
FIXME( "not supported index %d\n", index );
|
|
/* fall through */
|
|
default:
|
|
RtlSetLastWin32Error( ERROR_INVALID_PARAMETER );
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
* NtUserSetObjectInformation (win32u.@)
|
|
*/
|
|
BOOL WINAPI NtUserSetObjectInformation( HANDLE handle, INT index, void *info, DWORD len )
|
|
{
|
|
BOOL ret;
|
|
const USEROBJECTFLAGS *obj_flags = info;
|
|
|
|
if (index != UOI_FLAGS || !info || len < sizeof(*obj_flags))
|
|
{
|
|
RtlSetLastWin32Error( ERROR_INVALID_PARAMETER );
|
|
return FALSE;
|
|
}
|
|
/* FIXME: inherit flag */
|
|
SERVER_START_REQ( set_user_object_info )
|
|
{
|
|
req->handle = wine_server_obj_handle( handle );
|
|
req->flags = SET_USER_OBJECT_SET_FLAGS;
|
|
req->obj_flags = obj_flags->dwFlags;
|
|
ret = !wine_server_call_err( req );
|
|
}
|
|
SERVER_END_REQ;
|
|
return ret;
|
|
}
|
|
|
|
#ifdef _WIN64
|
|
static inline TEB64 *NtCurrentTeb64(void) { return NULL; }
|
|
#else
|
|
static inline TEB64 *NtCurrentTeb64(void) { return (TEB64 *)NtCurrentTeb()->GdiBatchCount; }
|
|
#endif
|
|
|
|
HWND get_desktop_window(void)
|
|
{
|
|
struct ntuser_thread_info *thread_info = NtUserGetThreadInfo();
|
|
BOOL is_service;
|
|
|
|
if (thread_info->top_window) return UlongToHandle( thread_info->top_window );
|
|
|
|
/* don't create an actual explorer desktop window for services */
|
|
is_service = is_service_process();
|
|
|
|
SERVER_START_REQ( get_desktop_window )
|
|
{
|
|
req->force = is_service;
|
|
if (!wine_server_call( req ))
|
|
{
|
|
thread_info->top_window = reply->top_window;
|
|
thread_info->msg_window = reply->msg_window;
|
|
}
|
|
}
|
|
SERVER_END_REQ;
|
|
|
|
if (!thread_info->top_window)
|
|
{
|
|
static const WCHAR appnameW[] = {'\\','?','?','\\','C',':','\\','w','i','n','d','o','w','s',
|
|
'\\','s','y','s','t','e','m','3','2','\\','e','x','p','l','o','r','e','r','.','e','x','e',0};
|
|
static const WCHAR cmdlineW[] = {'"','C',':','\\','w','i','n','d','o','w','s','\\',
|
|
's','y','s','t','e','m','3','2','\\','e','x','p','l','o','r','e','r','.','e','x','e','"',
|
|
' ','/','d','e','s','k','t','o','p',0};
|
|
static const WCHAR system_dir[] = {'C',':','\\','w','i','n','d','o','w','s','\\',
|
|
's','y','s','t','e','m','3','2','\\',0};
|
|
RTL_USER_PROCESS_PARAMETERS params = { sizeof(params), sizeof(params) };
|
|
PS_ATTRIBUTE_LIST ps_attr;
|
|
PS_CREATE_INFO create_info;
|
|
WCHAR desktop[MAX_PATH];
|
|
PEB *peb = NtCurrentTeb()->Peb;
|
|
HANDLE process, thread;
|
|
unsigned int status;
|
|
|
|
SERVER_START_REQ( set_user_object_info )
|
|
{
|
|
req->handle = wine_server_obj_handle( NtUserGetThreadDesktop(GetCurrentThreadId()) );
|
|
req->flags = SET_USER_OBJECT_GET_FULL_NAME;
|
|
wine_server_set_reply( req, desktop, sizeof(desktop) - sizeof(WCHAR) );
|
|
if (!wine_server_call( req ))
|
|
{
|
|
size_t size = wine_server_reply_size( reply );
|
|
desktop[size / sizeof(WCHAR)] = 0;
|
|
TRACE( "starting explorer for desktop %s\n", debugstr_w(desktop) );
|
|
}
|
|
else
|
|
desktop[0] = 0;
|
|
}
|
|
SERVER_END_REQ;
|
|
|
|
params.Flags = PROCESS_PARAMS_FLAG_NORMALIZED;
|
|
params.Environment = peb->ProcessParameters->Environment;
|
|
params.EnvironmentSize = peb->ProcessParameters->EnvironmentSize;
|
|
params.hStdError = peb->ProcessParameters->hStdError;
|
|
RtlInitUnicodeString( ¶ms.CurrentDirectory.DosPath, system_dir );
|
|
RtlInitUnicodeString( ¶ms.ImagePathName, appnameW + 4 );
|
|
RtlInitUnicodeString( ¶ms.CommandLine, cmdlineW );
|
|
RtlInitUnicodeString( ¶ms.WindowTitle, appnameW + 4 );
|
|
RtlInitUnicodeString( ¶ms.Desktop, desktop );
|
|
|
|
ps_attr.TotalLength = sizeof(ps_attr);
|
|
ps_attr.Attributes[0].Attribute = PS_ATTRIBUTE_IMAGE_NAME;
|
|
ps_attr.Attributes[0].Size = sizeof(appnameW) - sizeof(WCHAR);
|
|
ps_attr.Attributes[0].ValuePtr = (WCHAR *)appnameW;
|
|
ps_attr.Attributes[0].ReturnLength = NULL;
|
|
|
|
if (NtCurrentTeb64() && !NtCurrentTeb64()->TlsSlots[WOW64_TLS_FILESYSREDIR])
|
|
{
|
|
NtCurrentTeb64()->TlsSlots[WOW64_TLS_FILESYSREDIR] = TRUE;
|
|
status = NtCreateUserProcess( &process, &thread, PROCESS_ALL_ACCESS, THREAD_ALL_ACCESS,
|
|
NULL, NULL, 0, THREAD_CREATE_FLAGS_CREATE_SUSPENDED, ¶ms,
|
|
&create_info, &ps_attr );
|
|
NtCurrentTeb64()->TlsSlots[WOW64_TLS_FILESYSREDIR] = FALSE;
|
|
}
|
|
else
|
|
status = NtCreateUserProcess( &process, &thread, PROCESS_ALL_ACCESS, THREAD_ALL_ACCESS,
|
|
NULL, NULL, 0, THREAD_CREATE_FLAGS_CREATE_SUSPENDED, ¶ms,
|
|
&create_info, &ps_attr );
|
|
if (!status)
|
|
{
|
|
NtResumeThread( thread, NULL );
|
|
TRACE_(win)( "started explorer\n" );
|
|
NtUserWaitForInputIdle( process, 10000, FALSE );
|
|
NtClose( thread );
|
|
NtClose( process );
|
|
}
|
|
else ERR_(win)( "failed to start explorer %x\n", status );
|
|
|
|
SERVER_START_REQ( get_desktop_window )
|
|
{
|
|
req->force = 1;
|
|
if (!wine_server_call( req ))
|
|
{
|
|
thread_info->top_window = reply->top_window;
|
|
thread_info->msg_window = reply->msg_window;
|
|
}
|
|
}
|
|
SERVER_END_REQ;
|
|
}
|
|
|
|
if (!thread_info->top_window) ERR_(win)( "failed to create desktop window\n" );
|
|
else user_driver->pSetDesktopWindow( UlongToHandle( thread_info->top_window ));
|
|
|
|
register_builtin_classes();
|
|
return UlongToHandle( thread_info->top_window );
|
|
}
|
|
|
|
static HANDLE get_winstations_dir_handle(void)
|
|
{
|
|
char bufferA[64];
|
|
WCHAR buffer[64];
|
|
UNICODE_STRING str;
|
|
OBJECT_ATTRIBUTES attr;
|
|
NTSTATUS status;
|
|
HANDLE dir;
|
|
|
|
snprintf( bufferA, sizeof(bufferA), "\\Sessions\\%u\\Windows\\WindowStations", NtCurrentTeb()->Peb->SessionId );
|
|
str.Buffer = buffer;
|
|
str.MaximumLength = asciiz_to_unicode( buffer, bufferA );
|
|
str.Length = str.MaximumLength - sizeof(WCHAR);
|
|
InitializeObjectAttributes( &attr, &str, 0, 0, NULL );
|
|
status = NtOpenDirectoryObject( &dir, DIRECTORY_CREATE_OBJECT | DIRECTORY_TRAVERSE, &attr );
|
|
return status ? 0 : dir;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* get_default_desktop
|
|
*
|
|
* Get the name of the desktop to use for this app if not specified explicitly.
|
|
*/
|
|
static const WCHAR *get_default_desktop( void *buf, size_t buf_size )
|
|
{
|
|
const WCHAR *p, *appname = NtCurrentTeb()->Peb->ProcessParameters->ImagePathName.Buffer;
|
|
KEY_VALUE_PARTIAL_INFORMATION *info = buf;
|
|
WCHAR *buffer = buf;
|
|
HKEY tmpkey, appkey;
|
|
DWORD len;
|
|
|
|
static const WCHAR defaultW[] = {'D','e','f','a','u','l','t',0};
|
|
|
|
if ((p = wcsrchr( appname, '/' ))) appname = p + 1;
|
|
if ((p = wcsrchr( appname, '\\' ))) appname = p + 1;
|
|
len = lstrlenW(appname);
|
|
if (len > MAX_PATH) return defaultW;
|
|
memcpy( buffer, appname, len * sizeof(WCHAR) );
|
|
asciiz_to_unicode( buffer + len, "\\Explorer" );
|
|
|
|
/* @@ Wine registry key: HKCU\Software\Wine\AppDefaults\app.exe\Explorer */
|
|
if ((tmpkey = reg_open_hkcu_key( "Software\\Wine\\AppDefaults" )))
|
|
{
|
|
appkey = reg_open_key( tmpkey, buffer, lstrlenW(buffer) * sizeof(WCHAR) );
|
|
NtClose( tmpkey );
|
|
if (appkey)
|
|
{
|
|
len = query_reg_ascii_value( appkey, "Desktop", info, buf_size );
|
|
NtClose( appkey );
|
|
if (len) return (const WCHAR *)info->Data;
|
|
}
|
|
}
|
|
|
|
/* @@ Wine registry key: HKCU\Software\Wine\Explorer */
|
|
if ((appkey = reg_open_hkcu_key( "Software\\Wine\\Explorer" )))
|
|
{
|
|
len = query_reg_ascii_value( appkey, "Desktop", info, buf_size );
|
|
NtClose( appkey );
|
|
if (len) return (const WCHAR *)info->Data;
|
|
}
|
|
|
|
return defaultW;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* winstation_init
|
|
*
|
|
* Connect to the process window station and desktop.
|
|
*/
|
|
void winstation_init(void)
|
|
{
|
|
RTL_USER_PROCESS_PARAMETERS *params = NtCurrentTeb()->Peb->ProcessParameters;
|
|
WCHAR *winstation = NULL, *desktop = NULL, *buffer = NULL;
|
|
HANDLE handle, dir = NULL;
|
|
OBJECT_ATTRIBUTES attr;
|
|
UNICODE_STRING str;
|
|
|
|
static const WCHAR winsta0[] = {'W','i','n','S','t','a','0',0};
|
|
|
|
if (params->Desktop.Length)
|
|
{
|
|
buffer = malloc( params->Desktop.Length + sizeof(WCHAR) );
|
|
memcpy( buffer, params->Desktop.Buffer, params->Desktop.Length );
|
|
buffer[params->Desktop.Length / sizeof(WCHAR)] = 0;
|
|
if ((desktop = wcschr( buffer, '\\' )))
|
|
{
|
|
*desktop++ = 0;
|
|
winstation = buffer;
|
|
}
|
|
else desktop = buffer;
|
|
}
|
|
|
|
/* set winstation if explicitly specified, or if we don't have one yet */
|
|
if (buffer || !NtUserGetProcessWindowStation())
|
|
{
|
|
str.Buffer = (WCHAR *)(winstation ? winstation : winsta0);
|
|
str.Length = str.MaximumLength = lstrlenW( str.Buffer ) * sizeof(WCHAR);
|
|
dir = get_winstations_dir_handle();
|
|
InitializeObjectAttributes( &attr, &str, OBJ_CASE_INSENSITIVE | OBJ_OPENIF,
|
|
dir, NULL );
|
|
|
|
handle = NtUserCreateWindowStation( &attr, STANDARD_RIGHTS_REQUIRED | WINSTA_ALL_ACCESS, 0, 0, 0, 0, 0 );
|
|
if (handle)
|
|
{
|
|
NtUserSetProcessWindowStation( handle );
|
|
/* only WinSta0 is visible */
|
|
if (!winstation || !wcsicmp( winstation, winsta0 ))
|
|
{
|
|
USEROBJECTFLAGS flags;
|
|
flags.fInherit = FALSE;
|
|
flags.fReserved = FALSE;
|
|
flags.dwFlags = WSF_VISIBLE;
|
|
NtUserSetObjectInformation( handle, UOI_FLAGS, &flags, sizeof(flags) );
|
|
}
|
|
}
|
|
}
|
|
if (buffer || !NtUserGetThreadDesktop( GetCurrentThreadId() ))
|
|
{
|
|
char buffer[4096];
|
|
str.Buffer = (WCHAR *)(desktop ? desktop : get_default_desktop( buffer, sizeof(buffer) ));
|
|
str.Length = str.MaximumLength = lstrlenW( str.Buffer ) * sizeof(WCHAR);
|
|
if (!dir) dir = get_winstations_dir_handle();
|
|
InitializeObjectAttributes( &attr, &str, OBJ_CASE_INSENSITIVE | OBJ_OPENIF,
|
|
dir, NULL );
|
|
|
|
handle = NtUserCreateDesktopEx( &attr, NULL, NULL, 0, STANDARD_RIGHTS_REQUIRED | DESKTOP_ALL_ACCESS, 0 );
|
|
if (handle) NtUserSetThreadDesktop( handle );
|
|
}
|
|
NtClose( dir );
|
|
free( buffer );
|
|
}
|