296 lines
6.0 KiB
C
296 lines
6.0 KiB
C
/**
|
|
* WinPR: Windows Portable Runtime
|
|
* Handle Management
|
|
*
|
|
* Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <winpr/handle.h>
|
|
|
|
#ifndef _WIN32
|
|
|
|
#include <assert.h>
|
|
#include <pthread.h>
|
|
|
|
#include "../synch/synch.h"
|
|
#include "../thread/thread.h"
|
|
#include "../pipe/pipe.h"
|
|
#include "../comm/comm.h"
|
|
#include "../security/security.h"
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
#include <assert.h>
|
|
|
|
#include "../handle/handle.h"
|
|
|
|
/* _HandleCreators is a NULL-terminated array with a maximun of HANDLE_CREATOR_MAX HANDLE_CREATOR */
|
|
#define HANDLE_CLOSE_CB_MAX 128
|
|
static HANDLE_CLOSE_CB** _HandleCloseCbs = NULL;
|
|
static CRITICAL_SECTION _HandleCloseCbsLock;
|
|
|
|
static pthread_once_t _HandleCloseCbsInitialized = PTHREAD_ONCE_INIT;
|
|
static void _HandleCloseCbsInit()
|
|
{
|
|
/* NB: error management to be done outside of this function */
|
|
assert(_HandleCloseCbs == NULL);
|
|
_HandleCloseCbs = (HANDLE_CLOSE_CB**)calloc(HANDLE_CLOSE_CB_MAX+1, sizeof(HANDLE_CLOSE_CB*));
|
|
InitializeCriticalSection(&_HandleCloseCbsLock);
|
|
assert(_HandleCloseCbs != NULL);
|
|
}
|
|
|
|
/**
|
|
* Returns TRUE on success, FALSE otherwise.
|
|
*/
|
|
BOOL RegisterHandleCloseCb(HANDLE_CLOSE_CB* pHandleCloseCb)
|
|
{
|
|
int i;
|
|
|
|
if (pthread_once(&_HandleCloseCbsInitialized, _HandleCloseCbsInit) != 0)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
if (_HandleCloseCbs == NULL)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
EnterCriticalSection(&_HandleCloseCbsLock);
|
|
|
|
for (i=0; i<HANDLE_CLOSE_CB_MAX; i++)
|
|
{
|
|
if (_HandleCloseCbs[i] == NULL)
|
|
{
|
|
_HandleCloseCbs[i] = pHandleCloseCb;
|
|
LeaveCriticalSection(&_HandleCloseCbsLock);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
LeaveCriticalSection(&_HandleCloseCbsLock);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
|
|
BOOL CloseHandle(HANDLE hObject)
|
|
{
|
|
int i;
|
|
ULONG Type;
|
|
PVOID Object;
|
|
|
|
if (!winpr_Handle_GetInfo(hObject, &Type, &Object))
|
|
return FALSE;
|
|
|
|
if (pthread_once(&_HandleCloseCbsInitialized, _HandleCloseCbsInit) != 0)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
if (_HandleCloseCbs == NULL)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
EnterCriticalSection(&_HandleCloseCbsLock);
|
|
|
|
for (i=0; _HandleCloseCbs[i] != NULL; i++)
|
|
{
|
|
HANDLE_CLOSE_CB* close_cb = (HANDLE_CLOSE_CB*)_HandleCloseCbs[i];
|
|
|
|
if (close_cb && close_cb->IsHandled(hObject))
|
|
{
|
|
BOOL result = close_cb->CloseHandle(hObject);
|
|
LeaveCriticalSection(&_HandleCloseCbsLock);
|
|
return result;
|
|
}
|
|
}
|
|
|
|
LeaveCriticalSection(&_HandleCloseCbsLock);
|
|
|
|
if (Type == HANDLE_TYPE_THREAD)
|
|
{
|
|
WINPR_THREAD* thread;
|
|
thread = (WINPR_THREAD*) Object;
|
|
|
|
if (thread->started)
|
|
{
|
|
pthread_detach(thread->thread);
|
|
}
|
|
|
|
free(thread);
|
|
return TRUE;
|
|
}
|
|
else if (Type == HANDLE_TYPE_PROCESS)
|
|
{
|
|
WINPR_PROCESS* process;
|
|
process = (WINPR_PROCESS*) Object;
|
|
free(process);
|
|
return TRUE;
|
|
}
|
|
else if (Type == HANDLE_TYPE_MUTEX)
|
|
{
|
|
WINPR_MUTEX* mutex;
|
|
mutex = (WINPR_MUTEX*) Object;
|
|
pthread_mutex_destroy(&mutex->mutex);
|
|
free(Object);
|
|
return TRUE;
|
|
}
|
|
else if (Type == HANDLE_TYPE_EVENT)
|
|
{
|
|
WINPR_EVENT* event;
|
|
event = (WINPR_EVENT*) Object;
|
|
|
|
if (!event->bAttached)
|
|
{
|
|
if (event->pipe_fd[0] != -1)
|
|
{
|
|
close(event->pipe_fd[0]);
|
|
event->pipe_fd[0] = -1;
|
|
}
|
|
|
|
if (event->pipe_fd[1] != -1)
|
|
{
|
|
close(event->pipe_fd[1]);
|
|
event->pipe_fd[1] = -1;
|
|
}
|
|
}
|
|
|
|
free(Object);
|
|
return TRUE;
|
|
}
|
|
else if (Type == HANDLE_TYPE_SEMAPHORE)
|
|
{
|
|
WINPR_SEMAPHORE* semaphore;
|
|
semaphore = (WINPR_SEMAPHORE*) Object;
|
|
#ifdef WINPR_PIPE_SEMAPHORE
|
|
|
|
if (semaphore->pipe_fd[0] != -1)
|
|
{
|
|
close(semaphore->pipe_fd[0]);
|
|
semaphore->pipe_fd[0] = -1;
|
|
|
|
if (semaphore->pipe_fd[1] != -1)
|
|
{
|
|
close(semaphore->pipe_fd[1]);
|
|
semaphore->pipe_fd[1] = -1;
|
|
}
|
|
}
|
|
|
|
#else
|
|
#if defined __APPLE__
|
|
semaphore_destroy(mach_task_self(), *((winpr_sem_t*) semaphore->sem));
|
|
#else
|
|
sem_destroy((winpr_sem_t*) semaphore->sem);
|
|
#endif
|
|
#endif
|
|
free(Object);
|
|
return TRUE;
|
|
}
|
|
else if (Type == HANDLE_TYPE_TIMER)
|
|
{
|
|
WINPR_TIMER* timer;
|
|
timer = (WINPR_TIMER*) Object;
|
|
#ifdef __linux__
|
|
|
|
if (timer->fd != -1)
|
|
close(timer->fd);
|
|
|
|
#endif
|
|
free(Object);
|
|
return TRUE;
|
|
}
|
|
else if (Type == HANDLE_TYPE_ANONYMOUS_PIPE)
|
|
{
|
|
WINPR_PIPE* pipe;
|
|
pipe = (WINPR_PIPE*) Object;
|
|
|
|
if (pipe->fd != -1)
|
|
{
|
|
close(pipe->fd);
|
|
}
|
|
|
|
free(Object);
|
|
return TRUE;
|
|
}
|
|
else if (Type == HANDLE_TYPE_NAMED_PIPE)
|
|
{
|
|
WINPR_NAMED_PIPE* pNamedPipe = (WINPR_NAMED_PIPE*) Object;
|
|
|
|
if (pNamedPipe->clientfd != -1)
|
|
{
|
|
//WLOG_DBG(TAG, "%s: closing clientfd %d\n", __FUNCTION__, pNamedPipe->clientfd);
|
|
close(pNamedPipe->clientfd);
|
|
}
|
|
|
|
if (pNamedPipe->serverfd != -1)
|
|
{
|
|
//WLOG_DBG(TAG, "%s: closing serverfd %d\n", __FUNCTION__, pNamedPipe->serverfd);
|
|
close(pNamedPipe->serverfd);
|
|
}
|
|
|
|
if (pNamedPipe->pfnUnrefNamedPipe)
|
|
pNamedPipe->pfnUnrefNamedPipe(pNamedPipe);
|
|
|
|
free((void*)pNamedPipe->lpFileName);
|
|
free((void*)pNamedPipe->lpFilePath);
|
|
free((void*)pNamedPipe->name);
|
|
free(pNamedPipe);
|
|
return TRUE;
|
|
}
|
|
else if (Type == HANDLE_TYPE_ACCESS_TOKEN)
|
|
{
|
|
WINPR_ACCESS_TOKEN* token;
|
|
token = (WINPR_ACCESS_TOKEN*) Object;
|
|
|
|
if (token->Username)
|
|
free(token->Username);
|
|
|
|
if (token->Domain)
|
|
free(token->Domain);
|
|
|
|
free(token);
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL DuplicateHandle(HANDLE hSourceProcessHandle, HANDLE hSourceHandle, HANDLE hTargetProcessHandle,
|
|
LPHANDLE lpTargetHandle, DWORD dwDesiredAccess, BOOL bInheritHandle, DWORD dwOptions)
|
|
{
|
|
*((ULONG_PTR*) lpTargetHandle) = hSourceHandle;
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL GetHandleInformation(HANDLE hObject, LPDWORD lpdwFlags)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL SetHandleInformation(HANDLE hObject, DWORD dwMask, DWORD dwFlags)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
#endif
|