From 3824ef638c47e4b26ddfaaecf50c25d32d284232 Mon Sep 17 00:00:00 2001 From: Bryan Roe Date: Thu, 28 May 2020 00:54:30 -0700 Subject: [PATCH] Updated project files, and fixed compiler warnings --- makefile | 2 +- meshconsole/MeshConsole.vcxproj | 2 - meshservice/MeshService.vcxproj | 2 - microscript/ILibDuktape_ChildProcess.c | 3 +- microscript/ILibDuktape_HECI.c | 1313 --------------------- microscript/ILibDuktape_HECI.h | 25 - microscript/ILibDuktape_Polyfills.c | 11 + microscript/ILibDuktape_ScriptContainer.c | 4 - microscript/ILibDuktape_fs.c | 2 - modules/heci.js | 1 + 10 files changed, 15 insertions(+), 1350 deletions(-) delete mode 100644 microscript/ILibDuktape_HECI.c delete mode 100644 microscript/ILibDuktape_HECI.h diff --git a/makefile b/makefile index eb004da..a217685 100644 --- a/makefile +++ b/makefile @@ -118,7 +118,7 @@ SOURCES += microscript/ILibDuktape_SimpleDataStore.c microscript/ILibDuktape_Gen SOURCES += microscript/ILibDuktape_fs.c microscript/ILibDuktape_SHA256.c microscript/ILibduktape_EventEmitter.c SOURCES += microscript/ILibDuktape_EncryptionStream.c microscript/ILibDuktape_Polyfills.c microscript/ILibDuktape_Dgram.c SOURCES += microscript/ILibDuktape_ScriptContainer.c microscript/ILibDuktape_MemoryStream.c microscript/ILibDuktape_NetworkMonitor.c -SOURCES += microscript/ILibDuktape_ChildProcess.c microscript/ILibDuktape_HECI.c microscript/ILibDuktape_HttpStream.c microscript/ILibDuktape_Debugger.c +SOURCES += microscript/ILibDuktape_ChildProcess.c microscript/ILibDuktape_HttpStream.c microscript/ILibDuktape_Debugger.c SOURCES += $(ADDITIONALSOURCES) # Mesh Agent core diff --git a/meshconsole/MeshConsole.vcxproj b/meshconsole/MeshConsole.vcxproj index 29eac37..b5b7376 100644 --- a/meshconsole/MeshConsole.vcxproj +++ b/meshconsole/MeshConsole.vcxproj @@ -68,7 +68,6 @@ - @@ -118,7 +117,6 @@ - diff --git a/meshservice/MeshService.vcxproj b/meshservice/MeshService.vcxproj index 096bb1a..7c72d1c 100644 --- a/meshservice/MeshService.vcxproj +++ b/meshservice/MeshService.vcxproj @@ -573,7 +573,6 @@ - @@ -625,7 +624,6 @@ - diff --git a/microscript/ILibDuktape_ChildProcess.c b/microscript/ILibDuktape_ChildProcess.c index 538fa58..5c0f5cc 100644 --- a/microscript/ILibDuktape_ChildProcess.c +++ b/microscript/ILibDuktape_ChildProcess.c @@ -170,7 +170,8 @@ duk_ret_t ILibDuktape_ChildProcess_waitExit(duk_context *ctx) } duk_push_this(ctx); // [spawnedProcess] - char *_target = Duktape_GetStringPropertyValue(ctx, -1, "_target", NULL); + //char *_target = Duktape_GetStringPropertyValue(ctx, -1, "_target", NULL); + if (!ILibChain_IsLinkAlive(Duktape_GetPointerProperty(ctx, -1, ILibDuktape_ChildProcess_Manager))) { return(ILibDuktape_Error(ctx, "Cannot waitExit() because JS Engine is exiting")); diff --git a/microscript/ILibDuktape_HECI.c b/microscript/ILibDuktape_HECI.c deleted file mode 100644 index c205bc3..0000000 --- a/microscript/ILibDuktape_HECI.c +++ /dev/null @@ -1,1313 +0,0 @@ -/* -Copyright 2006 - 2018 Intel Corporation - -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. -*/ - -#include "ILibDuktape_HECI.h" -#include "ILibDuktapeModSearch.h" -#include "ILibDuktape_Helpers.h" -#include "ILibDuktape_DuplexStream.h" -#include "ILibDuktape_EventEmitter.h" -#include "ILibDuktape_ChildProcess.h" -#include "../microstack/ILibParsers.h" -#include "../microstack/ILibProcessPipe.h" -#include "../microstack/ILibRemoteLogging.h" - -#ifndef _NOHECI - -#ifdef WIN32 -#include -#include -#include -#include -DEFINE_GUID(GUID_DEVINTERFACE_HECI, 0xE2D1FF34, 0x3458, 0x49A9, 0x88, 0xDA, 0x8E, 0x69, 0x15, 0xCE, 0x9B, 0xE5); -#elif defined(_POSIX) -typedef struct HECI_client -{ - unsigned int max_msg_length; - unsigned char protocol_version; - unsigned char reserved[3]; -}HECI_client; -typedef struct uuid_le -{ - unsigned char b[16]; -}uuid_le; -struct HECI_CONNECT_client_data -{ - union - { - uuid_le uuid; - HECI_client properties; - }; -}; -#endif - -#define ILibDuktape_HECI_ChainLink "\xFF_HECI_ChainLink" -#define ILibDuktape_HECI_Descriptor "\xFF_HECI_Descriptor" -#define ILibDuktape_HECI_ChildProcess "\xFF_HECI_ChildProcess" -#define ILibDuktape_HECI_Q "\xFF_HECI_Q" -#define ILibDuktape_HECI_CTX_PTRS "\xFF_HECI_CTX_PTRS" -//#define ILibDuktape_HECI_IoctlWaitHandle "\xFF_HECI_IoctlWaitHandle" -#define ILibDuktape_HECI_OVERLAPPED "\xFF_HECI_OVERLAPPED" -#define ILibDuktape_HECI_Child "\xFF_HECI_Child" -#define ILibDuktape_HECI_Parent "\xFF_HECI_Parent" -#define ILibDuktape_HECI_Root "\xFF_HECI_Root" -#define ILibDuktape_HECI_MaxBufferSize "\xFF_HECI_MaxBufSize" -#define ILibDuktape_HECI_SessionMemPtr "\xFF_HECI_SessionMemPtr" -#define ILibDuktape_HECI_Session_NoPipeline "\xFF_HECI_Session_NoPipeline" -#define ILibDuktape_HECI_Session_Metadata "\xFF_HECI_Session_Metadata" - -#ifdef __DOXY__ -/*! -\implements EventEmitter -\brief JavaScript object interface for HECI calls. require('heci') to use; -*/ -class Heci -{ -public: - /*! - \brief Performs an Ioctl on the HECI device - \param code Ioctl Code to invoke - \param inBuffer \ Input data for Ioctl. Can be null - \param outBuffer \ Optional. Output data from Ioctl. Must be specified if Ioctl code returns data - \param callback Dispatched when a response is received from the HECI device\n - status Success Code. 0 = Success, Error code on failure\n - buffer \ Output Buffer\n - args Optional parameters that were passed in\n - \param args Optional arguments to pass when the callback is called - */ - void doIoctl(code, inBuffer[, outBuffer], callback[, ...args]); -}; -#endif - -typedef struct ILibDuktape_HECI_ioctl_data -{ - duk_context *ctx; - uintptr_t ctxnonce; - void *heciObject; - void *data; - void *Q; - void *chain; - ILibProcessPipe_Manager pipeManager; -#ifdef WIN32 - OVERLAPPED v; - HANDLE device; - DWORD bytesReceived; - DWORD abort; -#elif defined(_POSIX) - int device; -#endif - void *reserved; - - int code; - char *outBuffer; - void *outBuffer_obj; - duk_size_t outBufferLen; - duk_size_t bufferLen; - char buffer[]; -}ILibDuktape_HECI_ioctl_data; - -typedef struct ILibDuktape_HECI_Session -{ - void *chain; - int noPipelining; - ILibDuktape_DuplexStream *stream; -#ifdef WIN32 - OVERLAPPED v; - OVERLAPPED wv; - ILibProcessPipe_Manager mgr; - HANDLE descriptor; - DWORD bytesRead; -#else - int descriptor; -#endif - ILibQueue PendingWrites; - duk_size_t bufferSize; - char buffer[]; -}ILibDuktape_HECI_Session; -typedef struct ILibDuktape_HECI_WriteState -{ - ILibDuktape_HECI_Session *session; - int returnIgnored; -#ifndef WIN32 - int bufferOffset; -#endif - int bufferLen; - char buffer[]; -}ILibDuktape_HECI_WriteState; - -typedef struct HECI_chainLink -{ - ILibChain_Link link; - duk_context *ctx; - ILibDuktape_HECI_Session *session; - void *heciObject; - int descriptor; - int paused; -}HECI_chainLink; - -void ILibDuktape_HECI_Push(duk_context *ctx, void *chain); -ILibTransport_DoneState ILibDuktape_HECI_Session_WriteHandler_Process(ILibDuktape_HECI_Session *session); -extern int ILibDuktape_HECI_Debug; - -#ifdef WIN32 -BOOL ILibDuktape_HECI_Session_ReceiveSink(void *chain, HANDLE event, ILibWaitHandle_ErrorStatus errors, void* user); - -HANDLE ILibDuktape_HECI_windowsInit() -{ - PSP_DEVICE_INTERFACE_DETAIL_DATA deviceDetail = NULL; - HDEVINFO hDeviceInfo; - DWORD bufferSize; - SP_DEVICE_INTERFACE_DATA interfaceData; - LONG ii = 0; - HANDLE retVal = NULL; - - if (ILibDuktape_HECI_Debug) { printf("ILibDuktape_HECI_windowsInit()\n"); } - - // Find all devices that have our interface - hDeviceInfo = SetupDiGetClassDevs((LPGUID)&GUID_DEVINTERFACE_HECI, NULL, NULL, DIGCF_PRESENT | DIGCF_DEVICEINTERFACE); - if (hDeviceInfo == INVALID_HANDLE_VALUE) - { - if (ILibDuktape_HECI_Debug) { printf("...[FAILED]\n"); } - return(NULL); - } - if (ILibDuktape_HECI_Debug) { printf("...[Acquired hDeviceInfo]\n"); } - - - // Setup the interface data struct - interfaceData.cbSize = sizeof(interfaceData); - for (ii = 0; - SetupDiEnumDeviceInterfaces(hDeviceInfo, NULL, (LPGUID)&GUID_DEVINTERFACE_HECI, ii, &interfaceData); - ++ii) - { - // Found our device instance - if (!SetupDiGetDeviceInterfaceDetail(hDeviceInfo, &interfaceData, NULL, 0, &bufferSize, NULL)) - { - DWORD err = GetLastError(); - if (err != ERROR_INSUFFICIENT_BUFFER) - { - continue; - } - } - - // Allocate a big enough buffer to get detail data - deviceDetail = (PSP_DEVICE_INTERFACE_DETAIL_DATA)ILibMemory_AllocateA(bufferSize); - if (deviceDetail == NULL) { continue; } - - // Setup the device interface struct - deviceDetail->cbSize = sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA); - - // Try again to get the device interface detail info - if (!SetupDiGetDeviceInterfaceDetail(hDeviceInfo, &interfaceData, deviceDetail, bufferSize, NULL, NULL)) - { - deviceDetail = NULL; - continue; - } - - break; - } - SetupDiDestroyDeviceInfoList(hDeviceInfo); - - if (deviceDetail == NULL) - { - if (ILibDuktape_HECI_Debug) { printf("...[deviceDetail FAILED]\n"); } - return(NULL); - } - - retVal = CreateFile(deviceDetail->DevicePath, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_FLAG_OVERLAPPED, NULL); - if (retVal == INVALID_HANDLE_VALUE) - { - if (ILibDuktape_HECI_Debug) { printf("...[FAILED to acquire descriptor]\n"); } - return(NULL); - } - if (ILibDuktape_HECI_Debug) { printf("...[Acquired Descriptor]\n"); } - - return(retVal); -} -#else -int ILibDuktape_HECI_linuxInit() -{ - int fd, flags; - - if ((fd = open("/dev/mei", O_RDWR)) == -1 && (fd = open("/dev/mei0", O_RDWR)) == -1) - { - return(-1); - } - else - { - flags = fcntl(fd, F_GETFL, 0); - if (fcntl(fd, F_SETFL, O_NONBLOCK | flags) == -1) { printf("Failed to set O_NONBLOCK\n"); close(fd); fd = -1; } - return(fd); - } -} -#endif - -duk_ret_t ILibDuktape_HECI_SessionFinalizer(duk_context *ctx) -{ - if (ILibDuktape_HECI_Debug) { printf("ILibDuktape_HECI_SessionFinalizer()\n"); } - if (duk_has_prop_string(ctx, 0, ILibDuktape_HECI_SessionMemPtr)) - { - duk_get_prop_string(ctx, 0, ILibDuktape_HECI_SessionMemPtr); - ILibDuktape_HECI_Session *s = (ILibDuktape_HECI_Session*)Duktape_GetBuffer(ctx, -1, NULL); - if (s != NULL && s->PendingWrites != NULL) { ILibQueue_Destroy(s->PendingWrites); } // ToDo: If there is anything pending, we need to clear that too - if (s != NULL) { s->stream = NULL; } - } - return(0); -} - - -void ILibDuktape_HECI_Session_EmitErrorEvent(void *chain, void *session) -{ - ILibDuktape_HECI_Session *s = (ILibDuktape_HECI_Session*)session; - duk_context *ctx = s->stream->readableStream->ctx; - if (ILibIsRunningOnChainThread(chain) == 0) { Duktape_RunOnEventLoop(chain, duk_ctx_nonce(ctx), ctx, ILibDuktape_HECI_Session_EmitErrorEvent, NULL, session); return; } - - duk_push_heapptr(ctx, s->stream->ParentObject); // [session] - duk_get_prop_string(ctx, -1, "emit"); // [session][emit] - duk_swap_top(ctx, -2); // [emit][this] - duk_push_string(ctx, "error"); // [emit][this][error] - duk_push_error_object(ctx, DUK_ERR_ERROR, "HECI Connection Error"); // [emit][this][error][err] - if (duk_pcall_method(ctx, 2) != 0) { ILibDuktape_Process_UncaughtExceptionEx(ctx, "HECI.session.onError(): "); } - duk_pop(ctx); // ... -} -void ILibDuktape_HECI_Session_EmitStreamReady(void *chain, void *session) -{ - if (ILibIsRunningOnChainThread(chain) == 0) { Duktape_RunOnEventLoop(chain, duk_ctx_nonce(((ILibDuktape_HECI_Session*)session)->stream->writableStream->ctx), ((ILibDuktape_HECI_Session*)session)->stream->writableStream->ctx, ILibDuktape_HECI_Session_EmitStreamReady, NULL, session); return; } - ILibDuktape_DuplexStream_Ready(((ILibDuktape_HECI_Session*)session)->stream); -} - -#ifdef WIN32 -BOOL ILibDuktape_HECI_Session_WriteHandler_Ready(void *chain, HANDLE event, ILibWaitHandle_ErrorStatus errors, void* user) -{ - if (errors != ILibWaitHandle_ErrorStatus_NONE) { return(FALSE); } - - ILibDuktape_HECI_Session *session = (ILibDuktape_HECI_Session*)user; - DWORD bytesWritten; - - if (!ILibMemory_CanaryOK(session)) { return(FALSE); } - - ILibChain_RemoveWaitHandle(session->chain, session->wv.hEvent); - - if (session->noPipelining == 0) - { - ILibDuktape_HECI_WriteState *state = (ILibDuktape_HECI_WriteState*)ILibQueue_DeQueue(session->PendingWrites); - free(state); - } - - if (GetOverlappedResult(session->descriptor, &(session->wv), &bytesWritten, FALSE) == 0) - { - // Broken Connection - ILibDuktape_HECI_Session_EmitErrorEvent(session->chain, (void*)session); - } - else - { - if (session->noPipelining == 0) - { - // Write Completed - ILibDuktape_HECI_Session_WriteHandler_Process(session); - } - } - return(TRUE); -} -#endif - -ILibTransport_DoneState ILibDuktape_HECI_Session_WriteHandler_Process(ILibDuktape_HECI_Session *session) -{ - ILibTransport_DoneState retVal = ILibTransport_DoneState_ERROR; - int returnIgnored = 0; - -#ifdef WIN32 - DWORD bytesWritten; - BOOL result = TRUE; -#else - ssize_t bytesWritten; -#endif - - while (session->noPipelining || ILibQueue_GetCount(session->PendingWrites) > 0) - { - ILibDuktape_HECI_WriteState *state = (ILibDuktape_HECI_WriteState*)ILibQueue_PeekQueue(session->PendingWrites); - returnIgnored = state->returnIgnored; - -#ifdef WIN32 - if ((result = WriteFile(state->session->descriptor, state->buffer, state->bufferLen, &bytesWritten, &(state->session->wv))) == TRUE) - { - if (session->noPipelining == 0) { ILibQueue_DeQueue(state->session->PendingWrites); free(state); } - } - else - { - break; - } -#elif defined(_POSIX) - - bytesWritten = write(state->session->descriptor, state->buffer + state->bufferOffset, state->bufferLen - state->bufferOffset); - if (bytesWritten > 0) - { - state->bufferOffset += bytesWritten; - if (state->bufferOffset == state->bufferLen) { ILibQueue_DeQueue(state->session->PendingWrites); free(state); retVal = session->noPipelining == 0 ? ILibTransport_DoneState_COMPLETE:ILibTransport_DoneState_INCOMPLETE; } - } - else - { - if (errno != EAGAIN) - { - // Error Occured - retVal = ILibTransport_DoneState_ERROR; - ILibDuktape_HECI_Session_EmitErrorEvent(session->chain, (void*)session); - } - else - { - retVal = ILibTransport_DoneState_INCOMPLETE; - } - break; - } -#endif - - if (session->noPipelining != 0) { break; } - } - -#ifdef WIN32 - if (result == FALSE) - { - if (GetLastError() == ERROR_IO_PENDING) - { - // Not done writing - retVal = ILibTransport_DoneState_INCOMPLETE; - ILibChain_AddWaitHandle(session->chain, session->wv.hEvent, -1, ILibDuktape_HECI_Session_WriteHandler_Ready, session); - } - else - { - // Error Occured - retVal = ILibTransport_DoneState_ERROR; - ILibDuktape_HECI_Session_EmitErrorEvent(session->chain, (void*)session); - } - } - else - { - if (session->noPipelining == 0) - { - // No more Pending Writes - retVal = ILibTransport_DoneState_COMPLETE; - if (returnIgnored != 0) { ILibDuktape_HECI_Session_EmitStreamReady(session->chain, (void*)session); } - } - else - { - retVal = ILibTransport_DoneState_INCOMPLETE; - } - } -#else - if (ILibQueue_GetCount(session->PendingWrites) == 0 && session->noPipelining == 0) - { - // No more Pending Writes - retVal = ILibTransport_DoneState_COMPLETE; - if (returnIgnored != 0) { ILibDuktape_HECI_Session_EmitStreamReady(session->chain, (void*)session); } - } -#endif - return(retVal); -} - -#ifdef WIN32 -void __stdcall ILibDuktape_HECI_Session_WriteHandler(ULONG_PTR obj) -{ - // This Method is always dispatched from the WindowsRunLoop APC Thread - - ILibDuktape_HECI_WriteState *state = (ILibDuktape_HECI_WriteState*)obj; - ILibQueue_EnQueue(state->session->PendingWrites, state); - - if (ILibQueue_GetCount(state->session->PendingWrites) == 1) - { - // No Pending Writes, so we can go ahead and send out the first block - ILibDuktape_HECI_Session_WriteHandler_Process(state->session); - } -} -#elif defined(_POSIX) -ILibTransport_DoneState ILibDuktape_HECI_Session_WriteHandler(void *chain, ILibDuktape_HECI_WriteState* state) -{ - // This Method is always dispatched from the Microstack Thread - ILibQueue_EnQueue(state->session->PendingWrites, state); - - if (ILibQueue_GetCount(state->session->PendingWrites) == 1) - { - return(ILibDuktape_HECI_Session_WriteHandler_Process(state->session)); - } - else - { - return(ILibTransport_DoneState_INCOMPLETE); - } -} -#endif - -ILibTransport_DoneState ILibDuktape_HECI_Session_WriteSink_NoPipeline(void *chain, void *user) -{ - // This is always called from the Microstack Thread - - ILibDuktape_HECI_WriteState *state = (ILibDuktape_HECI_WriteState*)user; - ILibQueue_EnQueue(state->session->PendingWrites, state); - if (ILibQueue_GetCount(state->session->PendingWrites) == 1) - { - return(ILibDuktape_HECI_Session_WriteHandler_Process(state->session)); - } - else - { - return(ILibTransport_DoneState_INCOMPLETE); - } -} -ILibTransport_DoneState ILibDuktape_HECI_Session_WriteSink(ILibDuktape_DuplexStream *stream, char *buffer, int bufferLen, void *user) -{ - if ((duk_size_t)bufferLen > ((ILibDuktape_HECI_Session*)user)->bufferSize) { return(ILibTransport_DoneState_ERROR); } - - ILibDuktape_HECI_Session *session = (ILibDuktape_HECI_Session*)user; - ILibDuktape_HECI_WriteState *state = (ILibDuktape_HECI_WriteState*)ILibMemory_Allocate(bufferLen + sizeof(ILibDuktape_HECI_WriteState), 0, NULL, NULL); - state->session = session; - state->bufferLen = bufferLen; - memcpy_s(state->buffer, bufferLen, buffer, bufferLen); - - if (session->noPipelining == 0) - { -#if defined(WIN32) - state->returnIgnored = 1; - ILibDuktape_HECI_Session_WriteHandler((ULONG_PTR)state); -#elif defined(_POSIX) - if (ILibIsRunningOnChainThread(stream->readableStream->chain) != 0) - { - return(ILibDuktape_HECI_Session_WriteHandler(NULL, state)); - } - else - { - state->returnIgnored = 1; - ILibChain_RunOnMicrostackThreadEx(stream->readableStream->chain, (ILibChain_StartEvent)ILibDuktape_HECI_Session_WriteHandler, state); - } -#endif - } - else - { - // We can't pipeline write requests - if (ILibIsRunningOnChainThread(stream->readableStream->chain) != 0) - { - return(ILibDuktape_HECI_Session_WriteSink_NoPipeline(stream->readableStream->chain, state)); - } - else - { - state->returnIgnored = 1; - Duktape_RunOnEventLoop(stream->readableStream->chain, duk_ctx_nonce(stream->readableStream->ctx), stream->readableStream->ctx, (Duktape_EventLoopDispatch)ILibDuktape_HECI_Session_WriteSink_NoPipeline, NULL, state); - } - } - - return(ILibTransport_DoneState_INCOMPLETE); -} -void ILibDuktape_HECI_Session_EndSink(ILibDuktape_DuplexStream *stream, void *user) -{ - duk_context *ctx = stream->readableStream->ctx; - duk_push_this(ctx); - duk_get_prop_string(ctx, -1, "disconnect"); - duk_swap_top(ctx, -2); - if (duk_pcall_method(ctx, 0) != 0) { ILibDuktape_Process_UncaughtException(ctx); } - duk_pop(ctx); -} -void ILibDuktape_HECI_Session_PauseSink(ILibDuktape_DuplexStream *sender, void *user) -{ -#ifdef WIN32 - // NO-OP Because we are already PAUSED, since we context switched - UNREFERENCED_PARAMETER(sender); - UNREFERENCED_PARAMETER(user); -#else - UNREFERENCED_PARAMETER(sender); - UNREFERENCED_PARAMETER(user); -#endif -} -void ILibDuktape_HECI_Session_ResumeSink_NoPipeline(void *chain, void *user) -{ - // This is always called from the Microstack Thread - ILibDuktape_HECI_Session *session = (ILibDuktape_HECI_Session*)user; - ILibDuktape_HECI_WriteState *state = (ILibDuktape_HECI_WriteState*)ILibQueue_DeQueue(session->PendingWrites); - free(state); - - if (ILibQueue_GetCount(session->PendingWrites) == 0) - { - ILibDuktape_HECI_Session_EmitStreamReady(session->chain, session); - } - else - { - ILibDuktape_HECI_Session_WriteHandler_Process(session); - } -} -void ILibDuktape_HECI_Session_ResumeSink(ILibDuktape_DuplexStream *sender, void *user) -{ - ILibDuktape_HECI_Session *session = (ILibDuktape_HECI_Session*)user; - if (session->noPipelining != 0) - { - ILibDuktape_HECI_Session_ResumeSink_NoPipeline(sender->readableStream->chain, session); - } - -#ifdef WIN32 - BOOL result = ReadFile(session->descriptor, session->buffer, (DWORD)session->bufferSize, &(session->bytesRead), &(session->v)); - if (result == TRUE || GetLastError() == ERROR_IO_PENDING) - { - //if (ILibDuktape_HECI_Debug) { printf("...[Wait Handle Added]\n"); } - ILibChain_AddWaitHandle(session->chain, session->v.hEvent, -1, ILibDuktape_HECI_Session_ReceiveSink, session); - } -#endif -} -#ifdef WIN32 -void ILibDuktape_HECI_Session_ReceiveSink2(void *chain, void *user) -{ - ILibDuktape_HECI_Session *session = (ILibDuktape_HECI_Session*)user; - if (!ILibMemory_CanaryOK(session)) { return; } - - ILibDuktape_DuplexStream_WriteData(session->stream, session->buffer, session->bytesRead); - if (session->stream != NULL && !session->stream->readableStream->paused) - { - ILibDuktape_HECI_Session_ResumeSink(session->stream, session->stream->user); - } -} -BOOL ILibDuktape_HECI_Session_ReceiveSink(void *chain, HANDLE event, ILibWaitHandle_ErrorStatus errors, void* user) -{ - //if (ILibDuktape_HECI_Debug) { printf("ILibDuktape_HECI_Session_ReceiveSink\n"); } - if (errors != ILibWaitHandle_ErrorStatus_NONE) - { - if (ILibDuktape_HECI_Debug) { printf("ILibDuktape_HECI_Session_ReceiveSink\n"); } - if (ILibDuktape_HECI_Debug) { printf("...[ERROR: %d]\n", errors); } - return(FALSE); - } - //if (ILibDuktape_HECI_Debug) { printf("...[SIGNALED]\n"); } - ILibDuktape_HECI_Session *session = (ILibDuktape_HECI_Session*)user; - if (ILibMemory_CanaryOK(session)) - { - if (GetOverlappedResult(session->descriptor, &(session->v), &(session->bytesRead), FALSE) == TRUE) - { - ILibDuktape_DuplexStream_WriteData(session->stream, session->buffer, session->bytesRead); - if (session->stream != NULL && !session->stream->readableStream->paused) - { - ILibDuktape_HECI_Session_ResumeSink(session->stream, session->stream->user); - } - } - } - return(FALSE); -} -void __stdcall ILibDuktape_HECI_Session_Start(ULONG_PTR obj) -{ - ILibDuktape_HECI_Session *session = (ILibDuktape_HECI_Session*)obj; - DWORD bytesRead; - if (ILibDuktape_HECI_Debug) { printf("ILibDuktape_HECI_Session_Start()\n"); } - BOOL result = ReadFile(session->descriptor, session->buffer, (DWORD)session->bufferSize, &bytesRead, &(session->v)); - - duk_push_heapptr(session->stream->readableStream->ctx, session->stream->ParentObject); // [session] - ILibChain_AddWaitHandleEx(session->chain, session->v.hEvent, -1, ILibDuktape_HECI_Session_ReceiveSink, session, (char*)Duktape_GetStringPropertyValue(session->stream->readableStream->ctx, -1, ILibDuktape_HECI_Session_Metadata, "heci.session")); - duk_pop(session->stream->readableStream->ctx); // ... -} -#endif - -duk_ret_t ILibDuktape_HECI_create_OnClientConnect(duk_context *ctx) -{ - int statusCode = duk_require_int(ctx, 0); - ILibDuktape_HECI_Session *session = NULL; - duk_dup(ctx, 2); // [Session] - if (statusCode != 0) - { - duk_get_prop_string(ctx, -1, "emit"); // [session][emit] - duk_swap_top(ctx, -2); // [emit][this] - duk_push_string(ctx, "error"); // [emit][this][error] - duk_push_error_object(ctx, DUK_ERR_ERROR, "HECI Connection Error [%d]", statusCode); // [emit][this][error][err] - duk_push_int(ctx, statusCode); duk_put_prop_string(ctx, -2, "errno"); - if (duk_pcall_method(ctx, 2) != 0) { ILibDuktape_Process_UncaughtExceptionEx(ctx, "HECI.session.onError(): "); } - duk_pop(ctx); // ... - } - else - { - duk_size_t bufferLen; - char *buffer = (char*)Duktape_GetBuffer(ctx, 1, &bufferLen); - if (bufferLen > 4) - { - duk_push_int(ctx, ((int*)buffer)[0]); - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_MaxBufferSize); // [session] - - session = (ILibDuktape_HECI_Session*)Duktape_PushBuffer(ctx, sizeof(ILibDuktape_HECI_Session) + ((int*)buffer)[0]); // [session][buffer] - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_SessionMemPtr); // [session] -#ifdef WIN32 - session->v.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL); - session->wv.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL); -#endif - session->chain = Duktape_GetChain(ctx); - session->bufferSize = (duk_size_t)((int*)buffer)[0]; - session->stream = ILibDuktape_DuplexStream_Init(ctx, - ILibDuktape_HECI_Session_WriteSink, ILibDuktape_HECI_Session_EndSink, - ILibDuktape_HECI_Session_PauseSink, ILibDuktape_HECI_Session_ResumeSink, session); - ILibDuktape_CreateReadonlyProperty_int(ctx, "maxBufferSize", (int)session->bufferSize); - session->PendingWrites = ILibQueue_Create(); - duk_push_current_function(ctx); - session->noPipelining = Duktape_GetIntPropertyValue(ctx, -1, ILibDuktape_HECI_Session_NoPipeline, 0); - duk_pop(ctx); -#ifdef _POSIX - //printf("Session: %p\n", session); - duk_get_prop_string(ctx, -1, ILibDuktape_HECI_Child); // [session][heci] - duk_get_prop_string(ctx, -1, ILibDuktape_HECI_ChainLink); // [session][heci][link] - HECI_chainLink *link = (HECI_chainLink*)duk_get_pointer(ctx, -1); - link->session = session; - duk_pop_2(ctx); // [session] -#endif - - //printf("NoPipeline: %d\n", session->noPipelining); - } - else - { - // Even tho it was a success, the result buffer is invalid - duk_get_prop_string(ctx, -1, "emit"); // [session][emit] - duk_swap_top(ctx, -2); // [emit][this] - duk_push_string(ctx, "error"); // [emit][this][error] - duk_push_error_object(ctx, DUK_ERR_ERROR, "HECI Connection Error"); // [emit][this][error][err] - if (duk_pcall_method(ctx, 2) != 0) { ILibDuktape_Process_UncaughtExceptionEx(ctx, "HECI.session.onError(): "); } - duk_pop(ctx); // ... - } - } - - if (session != NULL) - { - // Hookup the Send/Receive logic -#ifdef WIN32 - duk_push_this(ctx); // [HECI] - session->descriptor = (HANDLE)Duktape_GetPointerProperty(ctx, -1, ILibDuktape_HECI_Descriptor); - duk_del_prop_string(ctx, -1, ILibDuktape_HECI_Descriptor); - duk_get_prop_string(ctx, -1, ILibDuktape_HECI_ChildProcess); // [HECI][childProcess] - duk_get_prop_string(ctx, -1, ILibDuktape_ChildProcess_Manager); // [HECI][childProcess][manager] - session->mgr = (ILibProcessPipe_Manager)duk_get_pointer(ctx, -1); - ILibDuktape_HECI_Session_Start((ULONG_PTR)session); -#else - duk_push_this(ctx); // [HECI] - session->descriptor = Duktape_GetIntPropertyValue(ctx, -1, ILibDuktape_HECI_Descriptor, -1); - ILibForceUnBlockChain(session->chain); -#endif - - duk_dup(ctx, 2); - duk_get_prop_string(ctx, -1, "emit"); // [session][emit] - duk_swap_top(ctx, -2); // [emit][this] - duk_push_string(ctx, "connect"); // [emit][this][connect] - if (duk_pcall_method(ctx, 1) != 0) { ILibDuktape_Process_UncaughtExceptionEx(ctx, "HECI.session.onConnect(): "); } - duk_pop(ctx); // ... - - - } - return(0); -} -duk_ret_t ILibDuktape_HECI_Session_connect(duk_context *ctx) -{ - if (ILibDuktape_HECI_Debug) { printf("connect()\n"); } - - int i; - int nargs = duk_get_top(ctx); - duk_push_this(ctx); // [Session] - duk_get_prop_string(ctx, -1, ILibDuktape_HECI_Child); // [Session][HECI] - duk_remove(ctx, -2); // [HECI] - - duk_get_prop_string(ctx, -1, "doIoctl"); // [HECI][func] - duk_swap_top(ctx, -2); // [doIoctl][this] - duk_get_prop_string(ctx, -1, "IOCTL"); // [doIoctl][this][IOCTL] - duk_get_prop_string(ctx, -1, "CLIENT_CONNECT"); // [doIoctl][this][IOCTL][CLIENT_CONNECT] - duk_remove(ctx, -2); // [doIoctl][this][CLIENT_CONNECT] - duk_dup(ctx, 0); // [doIoctl][this][CLIENT_CONNECT][guid] - duk_push_fixed_buffer(ctx, 16); // [doIoctl][this][CLIENT_CONNECT][guid][outBuffer] - duk_push_buffer_object(ctx, -1, 0, 16, DUK_BUFOBJ_NODEJS_BUFFER); // [doIoctl][this][CLIENT_CONNECT][guid][outBuffer][buf] - duk_remove(ctx, -2); - duk_push_c_function(ctx, ILibDuktape_HECI_create_OnClientConnect, DUK_VARARGS); // [doIoctl][this][CLIENT_CONNECT][guid][outBuffer][callback] - duk_push_int(ctx, 0); - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_Session_NoPipeline); - - for (i = 1; i < nargs; ++i) - { - if (duk_is_function(ctx, i)) { ILibDuktape_EventEmitter_AddOnce(ILibDuktape_EventEmitter_GetEmitter_fromThis(ctx), "connect", duk_require_heapptr(ctx, i)); } - else if (duk_is_object(ctx, i)) - { - int noPipeline = Duktape_GetIntPropertyValue(ctx, i, "noPipeline", 0); - duk_push_int(ctx, noPipeline); - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_Session_NoPipeline); - } - } - duk_push_this(ctx); // [doIoctl][this][CLIENT_CONNECT][guid][outBuffer][callback][Session] - duk_call_method(ctx, 5); // [retVal] - duk_pop(ctx); // ... - return(0); -} -#ifdef WIN32 -void __stdcall ILibDuktape_HECI_Session_CloseSink2(ULONG_PTR obj) -{ - HANDLE h = (HANDLE)obj; - CloseHandle(h); -} -#endif -duk_ret_t ILibDuktape_HECI_Session_close(duk_context *ctx) -{ - if (ILibDuktape_HECI_Debug) { printf("heci.disconnect()\n\n"); } - duk_push_this(ctx); // [session] - - if (duk_has_prop_string(ctx, -1, ILibDuktape_HECI_Child)) - { - duk_get_prop_string(ctx, -1, ILibDuktape_HECI_Child); // [session][heci] - duk_get_prop_string(ctx, -1, "disconnect"); // [session][heci][close] - duk_swap_top(ctx, -2); // [session][close][this] - duk_call_method(ctx, 0); - } - - duk_push_this(ctx); -#ifdef WIN32 - ILibDuktape_HECI_Session *session = NULL; - if (duk_has_prop_string(ctx, -1, ILibDuktape_HECI_SessionMemPtr)) - { - duk_get_prop_string(ctx, -1, ILibDuktape_HECI_SessionMemPtr); // [HECI][SESSION] - session = (ILibDuktape_HECI_Session*)Duktape_GetBuffer(ctx, -1, NULL); - - ILibChain_RemoveWaitHandle(session->chain, session->v.hEvent); - ILibChain_RemoveWaitHandle(session->chain, session->wv.hEvent); - session->stream = NULL; - CloseHandle(session->descriptor); - CloseHandle(session->v.hEvent); session->v.hEvent = NULL; - CloseHandle(session->wv.hEvent); session->wv.hEvent = NULL; - } -#else - int d = Duktape_GetIntPropertyValue(ctx, -1, ILibDuktape_HECI_Descriptor, -1); - HECI_chainLink *hcl = (HECI_chainLink*)Duktape_GetPointerProperty(ctx, -1, ILibDuktape_HECI_ChainLink); - if (hcl != NULL) - { - hcl->descriptor = -1; - if (d != -1) { close(d); }; - duk_del_prop_string(ctx, -1, ILibDuktape_HECI_Descriptor); - } -#endif - - duk_push_this(ctx); // [Session] - duk_get_prop_string(ctx, -1, ILibDuktape_HECI_Root); // [Session][root] - ILibDuktape_Push_ObjectStash(ctx); // [Session][root][stash] - duk_del_prop_string(ctx, -1, Duktape_GetStashKey(duk_get_heapptr(ctx, -3))); - - return(0); -} -duk_ret_t ILibDuktape_HECI_session_descriptorMetadata(duk_context *ctx) -{ - duk_push_this(ctx); // [session] - duk_get_prop_string(ctx, -1, ILibDuktape_HECI_Session_Metadata); // [session][oldVal] - duk_push_string(ctx, ", "); // [session][oldVal][newVal] - duk_string_concat(ctx, -2); duk_remove(ctx, -2); // [session][val] - duk_dup(ctx, 0); // [session][val][newVal] - duk_string_concat(ctx, -2); duk_remove(ctx, -2); // [session][newVal] - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_Session_Metadata); // [session] - - ILibDuktape_HECI_Session *session = (ILibDuktape_HECI_Session*)Duktape_GetBufferProperty(ctx, -1, ILibDuktape_HECI_SessionMemPtr); - if (session != NULL) - { - ILibChain_WaitHandle_UpdateMetadata(duk_ctx_chain(ctx), session->v.hEvent, (char*)Duktape_GetStringPropertyValue(ctx, -1, ILibDuktape_HECI_Session_Metadata, "heci.session")); - } - return(0); -} -duk_ret_t ILibDuktape_HECI_create(duk_context *ctx) -{ - if (ILibDuktape_HECI_Debug) { printf("\n\nILibDuktape_HECI_create()\n"); } - - duk_push_object(ctx); // [Session] - ILibDuktape_WriteID(ctx, "heci.session"); - ILibDuktape_HECI_Push(ctx, NULL); // [Session][HECI] - duk_dup(ctx, -2); // [Session][HECI][Session] - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_Parent); // [Session][HECI] - duk_push_this(ctx); // [session][HECI][root] - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_Root); // [session][HECI] - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_Child); // [Session] - duk_push_string(ctx, "heci.session"); duk_put_prop_string(ctx, -2, ILibDuktape_HECI_Session_Metadata); - ILibDuktape_CreateEventWithSetterEx(ctx, "descriptorMetadata", ILibDuktape_HECI_session_descriptorMetadata); - - duk_push_this(ctx); // [Session][root] - ILibDuktape_Push_ObjectStash(ctx); // [Session][root][stash] - duk_dup(ctx, -3); // [Session][root][stash][Session] - duk_put_prop_string(ctx, -2, Duktape_GetStashKey(duk_get_heapptr(ctx, -1))); // [Session][root][stash] - duk_pop(ctx); // [Session][root] - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_Root); // [Session] - - ILibDuktape_EventEmitter *emitter = ILibDuktape_EventEmitter_Create(ctx); - ILibDuktape_EventEmitter_CreateEventEx(emitter, "connect"); - ILibDuktape_EventEmitter_CreateEventEx(emitter, "error"); - ILibDuktape_CreateProperty_InstanceMethod(ctx, "connect", ILibDuktape_HECI_Session_connect, DUK_VARARGS); - ILibDuktape_CreateFinalizer(ctx, ILibDuktape_HECI_SessionFinalizer); - ILibDuktape_CreateInstanceMethod(ctx, "disconnect", ILibDuktape_HECI_Session_close, 0); - return(1); -} - - -void ILibDuktape_HECI_IoctlHandler_Dispatch(void *chain, void *user) -{ - ILibDuktape_HECI_ioctl_data *data = (ILibDuktape_HECI_ioctl_data*)user; - duk_size_t count; - int i; - duk_context *ctx; - - if (!ILibMemory_CanaryOK(data) || ! ILibMemory_CanaryOK(data->reserved)) - { - // Abort Dispatch, becuase the HECI object was GC'ed. - if (ILibDuktape_HECI_Debug) { printf("ILibDuktape_HECI_IoctlHandler_Dispatch() [ABORTED]\n"); } - if (ILibMemory_CanaryOK(data)) - { - if (ILibDuktape_HECI_Debug) { printf("...Release HECI Ioctl Data [%p]\n", (void*)data); } - ILibMemory_Free(data); - } - return; - } - - ctx = data->ctx; - duk_push_heapptr(data->ctx, data->data); // [array] - duk_push_heapptr(data->ctx, data->heciObject); // [array][heci] - duk_get_prop_index(data->ctx, -2, 2); // [array][heci][callback] - duk_swap_top(data->ctx, -2); // [array][callback][this] - count = duk_get_length(data->ctx, -3); - duk_push_int(data->ctx, data->code); // [array][callback][this][status] - duk_get_prop_index(data->ctx, -4, 1); // [array][callback][this][status][buffer] - - for (i = 3; i < (int)count; ++i) - { - duk_get_prop_index(data->ctx, -i - 2, i); // [array][callback][this][status][buffer][...args...] - } - if (duk_pcall_method(data->ctx, (duk_idx_t)count - 1) != 0) { ILibDuktape_Process_UncaughtExceptionEx(data->ctx, "heci.ioctlHandler_Dispatch.callback(): "); } - duk_pop_2(data->ctx); // ... - - duk_push_heapptr(data->ctx, data->heciObject); // [heci] - ILibDuktape_Push_ObjectStash(data->ctx); // [heci][stash] - duk_del_prop_string(data->ctx, -1, Duktape_GetStashKey(data->data)); // (This will dereference args passed to doIoctl) - duk_pop_2(ctx); // ... - if (ILibDuktape_HECI_Debug) { printf("doIoctl() [COMPLETE] -> ** Release HECI Ioctl Data [%p]\n", (void*)data); } - ILibMemory_Free(data); -} -#ifdef WIN32 -//void ILibDuktape_HECI_NextIoctl(ILibQueue q); -void ILibDuktape_HECI_NextIoctl(duk_context *ctx, void *heci); -BOOL ILibDuktape_HECI_IoctlHandler(void * chain, HANDLE h, ILibWaitHandle_ErrorStatus errors, void *user) -{ - if (!ILibMemory_CanaryOK(user)) { return(FALSE); } - - duk_context *ctx = (duk_context*)((void**)user)[0]; - int top = duk_get_top(ctx); - - duk_push_heapptr(ctx, ((void**)user)[1]); // [HECI] - duk_get_prop_string(ctx, -1, ILibDuktape_HECI_Q); // [HECI][QUEUE] - HANDLE descriptor = Duktape_GetPointerProperty(ctx, -2, ILibDuktape_HECI_Descriptor); - OVERLAPPED *p = (OVERLAPPED*)Duktape_GetBufferProperty(ctx, -2, ILibDuktape_HECI_OVERLAPPED); - DWORD bytesReceived = 0; - - if (errors == ILibWaitHandle_ErrorStatus_NONE) - { - if (GetOverlappedResult(descriptor, p, &bytesReceived, FALSE) == FALSE) { return(TRUE); } - duk_queue_deQueue(ctx, -1); // [HECI][QUEUE][OBJ] - duk_get_prop_string(ctx, -1, "callback"); // [HECI][QUEUE][OBJ][callback] - duk_get_prop_string(ctx, -1, "apply"); // [HECI][QUEUE][OBJ][callback][apply] - duk_swap_top(ctx, -2); // [HECI][QUEUE][OBJ][apply][func] - duk_dup(ctx, -5); // [HECI][QUEUE][OBJ][apply][func][this] - duk_get_prop_string(ctx, -4, "args"); // [HECI][QUEUE][OBJ][apply][func][this][args] - duk_get_prop_string(ctx, -5, "outbuffer"); // [HECI][QUEUE][OBJ][apply][func][this][args][outbuffer] - if (!duk_is_null(ctx, -1)) { duk_buffer_slice(ctx, -1, 0, bytesReceived); } // [HECI][QUEUE][OBJ][apply][func][this][args][outbuffer] - duk_array_unshift(ctx, -2); // [HECI][QUEUE][OBJ][apply][func][this][args] - duk_push_int(ctx, 0); duk_array_unshift(ctx, -2); // [HECI][QUEUE][OBJ][apply][func][this][args] - duk_pcall_method(ctx, 2); duk_pop_2(ctx); // [HECI][QUEUE] - if (!duk_queue_isEmpty(ctx, -1)) - { - ILibDuktape_HECI_NextIoctl(ctx, ((void**)user)[1]); - duk_set_top(ctx, top); - return(TRUE); - } - } - else - { - DebugBreak(); - } - duk_set_top(ctx, top); - return(FALSE); -} -void ILibDuktape_HECI_NextIoctl(duk_context *ctx, void *heci) -{ - duk_push_heapptr(ctx, heci); // [HECI] - HANDLE descriptor = (HANDLE)Duktape_GetPointerProperty(ctx, -1, ILibDuktape_HECI_Descriptor); - OVERLAPPED *p = (OVERLAPPED*)Duktape_GetBufferProperty(ctx, -1, ILibDuktape_HECI_OVERLAPPED); - void **ptrs = (void**)Duktape_GetBufferProperty(ctx, -1, ILibDuktape_HECI_CTX_PTRS); - - duk_get_prop_string(ctx, -1, ILibDuktape_HECI_Q); // [HECI][QUEUE] - - duk_queue_peek(ctx, -1); // [HECI][QUEUE][OBJ] - duk_size_t bufferLen = 0; - duk_size_t outBufferLen = 0; - char *buffer = Duktape_GetBufferPropertyEx(ctx, -1, "buffer", &bufferLen); - char *outBuffer = Duktape_GetBufferPropertyEx(ctx, -1, "outbuffer", &outBufferLen); - DWORD dwIoControlCode = (DWORD)Duktape_GetIntPropertyValue(ctx, -1, "code", 0); - - DeviceIoControl(descriptor, dwIoControlCode, buffer, (DWORD)bufferLen, outBuffer, (DWORD)outBufferLen, NULL, p); - ILibChain_AddWaitHandleEx(duk_ctx_chain(ctx), p->hEvent, 5000, ILibDuktape_HECI_IoctlHandler, ptrs, "heci.doIoctl()"); -} -#endif - -duk_ret_t ILibDuktape_HECI_doIoctl(duk_context *ctx) -{ - int code = duk_require_int(ctx, 0); - duk_size_t bufferLen = 0; - char *buffer = duk_is_null(ctx, 1) ? NULL : (char*)Duktape_GetBuffer(ctx, 1, &bufferLen); - int nargs = duk_get_top(ctx); - int i; - duk_size_t outBufferLen; - char *outBuffer; - int cbx; - - if (duk_is_buffer(ctx, 2) || duk_is_buffer_data(ctx, 2)) - { - outBuffer = (char*)Duktape_GetBuffer(ctx, 2, &outBufferLen); - cbx = 3; - } - else - { - outBuffer = NULL; - outBufferLen = 0; - cbx = 2; - } - -#ifdef _POSIX - if (outBuffer == NULL) - { - outBuffer = buffer; - outBufferLen = bufferLen; - } - else - { - if (bufferLen < outBufferLen) { return(ILibDuktape_Error(ctx, "HECI.doIoctl(): Output Buffer too small")); } - memcpy_s(outBuffer, outBufferLen, buffer, bufferLen); - } -#endif - - duk_require_function(ctx, cbx); // Make sure a callback function was specified - - duk_push_this(ctx); // [HECI] -#ifdef WIN32 - if (!duk_has_prop_string(ctx, -1, ILibDuktape_HECI_Descriptor)) { return(ILibDuktape_Error(ctx, "Invalid Operation. Call Reset()")); } -#endif - duk_get_prop_string(ctx, -1, ILibDuktape_HECI_Q); // [HECI][QUEUE] - duk_push_object(ctx); // [HECI][QUEUE][OBJ] - - duk_dup(ctx, cbx); duk_put_prop_string(ctx, -2, "callback"); - duk_dup(ctx, 1); duk_put_prop_string(ctx, -2, "buffer"); - if (outBufferLen > 0) { duk_dup(ctx, 2); } else { duk_push_null(ctx); } - duk_put_prop_string(ctx, -2, "outbuffer"); - duk_push_int(ctx, code); duk_put_prop_string(ctx, -2, "code"); - void **ptrs = Duktape_PushBuffer(ctx, 2 * sizeof(void*)); duk_put_prop_string(ctx, -2, ILibDuktape_HECI_CTX_PTRS); - ptrs[0] = ctx; - ptrs[1] = duk_get_heapptr(ctx, -3); - - duk_push_array(ctx); // [HECI][QUEUE][OBJ][ARGS] - for (i = cbx + 1; i < nargs; ++i) - { - duk_dup(ctx, i); // [HECI][QUEUE][OBJ][ARGS][arg] - duk_array_push(ctx, -2); // [HECI][QUEUE][OBJ][ARGS] - } - duk_put_prop_string(ctx, -2, "args"); // [HECI][QUEUE][OBJ] - duk_queue_enQueue(ctx, -2); // [HECI][QUEUE] - if (duk_get_length(ctx, -1) == 1) - { -#ifdef WIN32 - ILibDuktape_HECI_NextIoctl(ctx, duk_get_heapptr(ctx, -2)); -#else - ILibForceUnBlockChain(duk_ctx_chain(ctx)); -#endif - } - return(0); -// -// duk_require_function(ctx, cbx); -// duk_push_this(ctx); // [heci] -// duk_get_prop_string(ctx, -1, ILibDuktape_HECI_Q); // [heci][q] -// qptr = duk_get_heapptr(ctx, -1); -// duk_pop(ctx); // [heci] -// -// ILibDuktape_Push_ObjectStash(ctx); // [heci][stash] -// duk_push_array(ctx); // [heci][stash][array] -// ILibDuktape_HECI_ioctl_data *data; -// data = (ILibDuktape_HECI_ioctl_data*)ILibMemory_SmartAllocate(bufferLen + sizeof(ILibDuktape_HECI_ioctl_data)); -// data->ctxnonce = duk_ctx_nonce(ctx); -// if (ILibDuktape_HECI_Debug) { printf("doIoctl() -> Allocate HECI Ioctl Data [%p]\n", (void*)data); } -// data->reserved = Duktape_PushBuffer(ctx, sizeof(void*)); // [heci][stash][array][ptr] -// duk_put_prop_index(ctx, -2, 0); // [heci][stash][array] -// if (outBufferLen > 0) -// { // [heci][stash][array][buffer] -// duk_dup(ctx, 2); -// } -// else -// { -// duk_push_null(ctx); // [heci][stash][array][buffer] -// } -// -// duk_put_prop_index(ctx, -2, 1); // [heci][stash][array] -// data->ctx = ctx; -// -// duk_dup(ctx, cbx); // [heci][stash][array][callback] -// duk_put_prop_index(ctx, -2, 2); // [heci][stash][array] -// -//#ifdef WIN32 -// duk_get_prop_string(ctx, -3, ILibDuktape_HECI_IoctlWaitHandle); // [heci][stash][array][handle] -// data->v.hEvent = (HANDLE)duk_get_pointer(ctx, -1); -// duk_pop(ctx); // [heci][stash][array] -//#endif -// -// duk_get_prop_string(ctx, -3, ILibDuktape_HECI_Descriptor); // [heci][stash][array][descriptor] -//#ifdef WIN32 -// data->device = (HANDLE)duk_get_pointer(ctx, -1); -//#elif defined(_POSIX) -// data->device = duk_get_int(ctx, -1); -//#endif -// duk_pop(ctx); // [heci][stash][array] -// data->chain = Duktape_GetChain(ctx); -// data->Q = Q; -// data->code = code; -// data->outBuffer = outBuffer; -// data->outBufferLen = outBufferLen; -// data->heciObject = duk_get_heapptr(ctx, -3); -// data->bufferLen = bufferLen; -// data->data = duk_get_heapptr(ctx, -1); -// memcpy_s(data->buffer, bufferLen, buffer, bufferLen); -// -// for (i = cbx + 1; i < nargs; ++i) -// { -// duk_dup(ctx, i); // [heci][stash][array][object] -// duk_put_prop_index(ctx, -2, i-1); // [heci][stash][array] -// } -// duk_put_prop_string(ctx, -2, Duktape_GetStashKey(duk_get_heapptr(ctx, -1))); // [heci][stash] -// -// -//#ifdef WIN32 -// duk_get_prop_string(ctx, -2, ILibDuktape_HECI_ChildProcess); // [heci][stash][childProcess] -// duk_get_prop_string(ctx, -1, ILibDuktape_ChildProcess_Manager); // [heci][stash][childProcess][manager] -// data->pipeManager = (ILibProcessPipe_Manager)duk_get_pointer(ctx, -1); -// -// ILibDuktape_HECI_apc_AddIoctl((ULONG_PTR)data); -//#elif defined(_POSIX) -// ILibDuktape_HECI_AddIoctl(data); -//#endif -// -// return(0); -} - -#ifdef WIN32 -void __stdcall ILibDuktape_HECI_Finalizer2(ULONG_PTR obj) -{ - ILibQueue Q = (ILibQueue)obj; - - if (ILibQueue_IsEmpty(Q) != 0) - { - if (ILibDuktape_HECI_Debug) { printf("<== Queue: %p Destroyed in Finalizer ==>\n", (void*)Q); } - ILibQueue_Destroy(Q); - } - else - { - void *node = ILibLinkedList_GetNode_Head((void*)Q); - while (node != NULL) - { - ((ILibDuktape_HECI_ioctl_data*)ILibLinkedList_GetDataFromNode(node))->abort = 1; - node = ILibLinkedList_GetNextNode(node); - } - } -} -#endif - -duk_ret_t ILibDuktape_HECI_Finalizer(duk_context *ctx) -{ - if (ILibDuktape_HECI_Debug) { printf("ILibDuktape_HECI_Finalizer()\n"); } -#ifdef WIN32 - HANDLE h = Duktape_GetPointerProperty(ctx, 0, ILibDuktape_HECI_Descriptor); - if (h != NULL) { CloseHandle(h); } - OVERLAPPED *p = (OVERLAPPED*)Duktape_GetBufferProperty(ctx, 0, ILibDuktape_HECI_OVERLAPPED); - if (p->hEvent != NULL) - { - ILibChain_RemoveWaitHandle(duk_ctx_chain(ctx), p->hEvent); - CloseHandle(p->hEvent); - } -#endif - - -#ifdef _POSIX - if (duk_has_prop_string(ctx, 0, ILibDuktape_HECI_ChainLink)) - { - duk_get_prop_string(ctx, 0, ILibDuktape_HECI_ChainLink); - HECI_chainLink *h = (HECI_chainLink*)duk_get_pointer(ctx, -1); - h->ctx = NULL; - h->heciObject = NULL; - if (h->link.ParentChain != NULL) { ILibChain_SafeRemove(h->link.ParentChain, h); } - } -#endif - - return(0); -} -#if !defined(WIN32) && !defined(__APPLE__) -void ILibDuktape_HECI_PreSelect(void* object, fd_set *readset, fd_set *writeset, fd_set *errorset, int* blocktime) -{ - - int result; - HECI_chainLink *h = (HECI_chainLink*)object; - //printf("h = %p, descriptor = %d, paused = %d, session = %p\n", (void*)h, h->descriptor, h->paused, (void*)h->session); - - if (h->descriptor <= 0) { return; } - if (h->paused == 0 && h->session != NULL) { FD_SET(h->descriptor, readset); } - if (h->session != NULL && ILibMemory_CanaryOK(h->session) && ILibQueue_GetCount(h->session->PendingWrites) > 0) { FD_SET(h->descriptor, writeset); } - - while (ILibQueue_GetCount(h->Q) > 0 && h->paused == 0) - { - ILibDuktape_HECI_ioctl_data *data = (ILibDuktape_HECI_ioctl_data*)ILibQueue_DeQueue(h->Q); - switch (data->code) - { - case 0x00: - break; - case 0x01: - case 0x02: - case 0x03: - result = ioctl(h->descriptor, _IOC(_IOC_READ | _IOC_WRITE, 'H', data->code, data->outBufferLen), data->outBuffer); - data->code = result ? errno : 0; - ILibDuktape_HECI_IoctlHandler_Dispatch(NULL, data); - break; - default: - break; - } - } -} -void ILibDuktape_HECI_PostSelect(void* object, int slct, fd_set *readset, fd_set *writeset, fd_set *errorset) -{ - HECI_chainLink *h = (HECI_chainLink*)object; - if (h->descriptor <= 0) { return; } - - if (FD_ISSET(h->descriptor, readset)) - { - //printf("session = %p\n", (void*)h->session); - //printf("Attempting to read: %d bytes from %p\n", h->session->bufferSize, (void*)h->session->buffer); - int bytesRead = read(h->descriptor, h->session->buffer, h->session->bufferSize); - if (bytesRead >= 0) - { - ILibDuktape_DuplexStream_WriteData(h->session->stream, h->session->buffer, bytesRead); - } - else if(h->ctx != NULL && ILibMemory_CanaryOK(h->session)) - { - ILibDuktape_EventEmitter_SetupEmit(h->ctx, h->session->stream->ParentObject, "error"); // [emit][this][error] - duk_push_string(h->ctx, "HECI Read Error"); // [emit][this][error][msg] - duk_pcall_method(h->ctx, 2); duk_pop(h->ctx); // ... - - duk_push_heapptr(h->ctx, h->session->stream->ParentObject); // [heci] - duk_del_prop_string(h->ctx, -1, ILibDuktape_HECI_Descriptor); - duk_pop(h->ctx); // ... - - int td = h->descriptor; - h->descriptor = -1; - close(td); - } - } - if (FD_ISSET(h->descriptor, writeset)) - { - ILibDuktape_HECI_Session_WriteHandler_Process(h->session); - } -} -void ILibDuktape_HECI_Destroy(void *object) -{ - HECI_chainLink *h = (HECI_chainLink*)object; - if (h->ctx != NULL && h->heciObject != NULL) - { - duk_push_heapptr(h->ctx, h->heciObject); // [heci] - duk_del_prop_string(h->ctx, -1, ILibDuktape_HECI_ChainLink); - duk_pop(h->ctx); // ... - } - close(h->descriptor); -} -#endif - -duk_ret_t ILibDuktape_HECI_reset(duk_context *ctx) -{ - duk_push_this(ctx); -#ifdef WIN32 - if (ILibDuktape_HECI_Debug) { printf("ILibDuktape_HECI_reset()\n"); } - - HANDLE h = (HANDLE)Duktape_GetPointerProperty(ctx, -1, ILibDuktape_HECI_Descriptor); - if (h != NULL) - { - CloseHandle(h); - } - h = ILibDuktape_HECI_windowsInit(); - duk_push_pointer(ctx, h); - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_Descriptor); -#endif - return(0); -} - -void ILibDuktape_HECI_Push(duk_context *ctx, void *chain) -{ - duk_push_object(ctx); // [HECI] - ILibDuktape_WriteID(ctx, "heci"); - ILibDuktape_CreateFinalizer(ctx, ILibDuktape_HECI_Finalizer); - -#ifdef WIN32 - HANDLE h = ILibDuktape_HECI_windowsInit(); - if (h == NULL) { ILibDuktape_Error(ctx, "Error initializing HECI"); return; } - duk_push_pointer(ctx, h); // [HECI][HANDLE] - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_Descriptor); // [HECI] - - if (duk_peval_string(ctx, "require('child_process');") != 0) // [HECI][child_process] - { - ILibDuktape_Error(ctx, "Error instantiating dependency 'child_process'"); - return; - } - - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_ChildProcess); // [HECI] - OVERLAPPED *v = Duktape_PushBuffer(ctx, sizeof(OVERLAPPED)); // [HECI][OVERLAPPED] - v->hEvent = CreateEvent(NULL, TRUE, FALSE, NULL); - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_OVERLAPPED); // [HECI] - void **ptrs = (void**)Duktape_PushBuffer(ctx, 2*sizeof(void*)); duk_put_prop_string(ctx, -2, ILibDuktape_HECI_CTX_PTRS); - ptrs[0] = ctx; - ptrs[1] = duk_get_heapptr(ctx, -1); - -#elif defined(_POSIX) && !defined(__APPLE__) - int h = ILibDuktape_HECI_linuxInit(); - if (h < 0) { ILibDuktape_Error(ctx, "error initializing HECI"); return; } - duk_push_int(ctx, h); // [HECI][descriptor] - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_Descriptor); // [HECI] - - HECI_chainLink *hlink = ILibMemory_Allocate(sizeof(HECI_chainLink), 0, NULL, NULL); - hlink->link.MetaData = "ILibDuktape_HECI"; - hlink->ctx = ctx; - hlink->descriptor = h; - hlink->link.PreSelectHandler = ILibDuktape_HECI_PreSelect; - hlink->link.PostSelectHandler = ILibDuktape_HECI_PostSelect; - hlink->link.DestroyHandler = ILibDuktape_HECI_Destroy; - duk_push_pointer(ctx, hlink); // [HECI][link] - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_ChainLink); // [HECI] - - ILibChain_SafeAdd(Duktape_GetChain(ctx), hlink); -#endif - if (chain != NULL) { ILibDuktape_CreateInstanceMethod(ctx, "create", ILibDuktape_HECI_create, 0); } - ILibDuktape_CreateInstanceMethod(ctx, "doIoctl", ILibDuktape_HECI_doIoctl, DUK_VARARGS); - ILibDuktape_CreateInstanceMethod(ctx, "disconnect", ILibDuktape_HECI_Session_close, 0); - ILibDuktape_CreateInstanceMethod(ctx, "reset", ILibDuktape_HECI_reset, 0); - - duk_queue_create(ctx); - duk_put_prop_string(ctx, -2, ILibDuktape_HECI_Q); // [HECI] - duk_push_object(ctx); -#ifdef WIN32 - ILibDuktape_CreateReadonlyProperty_int(ctx, "HECI_VERSION", (int)(CTL_CODE(0x8000, 0x800, METHOD_BUFFERED, FILE_READ_ACCESS | FILE_WRITE_ACCESS))); - ILibDuktape_CreateReadonlyProperty_int(ctx, "CLIENT_CONNECT", (int)(CTL_CODE(0x8000, 0x801, METHOD_BUFFERED, FILE_READ_ACCESS | FILE_WRITE_ACCESS))); -#elif defined(_POSIX) - ILibDuktape_CreateReadonlyProperty_int(ctx, "HECI_VERSION", (int)0x00); - ILibDuktape_CreateReadonlyProperty_int(ctx, "CLIENT_CONNECT", (int)0x01); -#endif - - ILibDuktape_CreateReadonlyProperty(ctx, "IOCTL"); - duk_push_object(ctx); - duk_peval_string(ctx, "Buffer.from('DBA4336776047B4EB3AFBCFC29BEE7A7', 'hex');"); - ILibDuktape_CreateReadonlyProperty(ctx, "LME"); - duk_peval_string(ctx, "Buffer.from('2800F812B7B42D4BACA846E0FF65814C', 'hex');"); - ILibDuktape_CreateReadonlyProperty(ctx, "AMT"); - ILibDuktape_CreateReadonlyProperty(ctx, "GUIDS"); - -} // KLOCKWORK: We are not losing reference to created Event... It is freed in the object finalizer 'ILibDuktape_HECI_Finalizer' -void ILibDuktape_HECI_Init(duk_context *ctx) -{ - ILibDuktape_ModSearch_AddHandler(ctx, "heci", ILibDuktape_HECI_Push); -} - -#endif - diff --git a/microscript/ILibDuktape_HECI.h b/microscript/ILibDuktape_HECI.h deleted file mode 100644 index 9a361dc..0000000 --- a/microscript/ILibDuktape_HECI.h +++ /dev/null @@ -1,25 +0,0 @@ -/* -Copyright 2006 - 2018 Intel Corporation - -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. -*/ - -#ifndef __DUKTAPEHECI__ -#define __DUKTAPEHECI__ - -#include "duktape.h" - -void ILibDuktape_HECI_Init(duk_context *ctx); - - -#endif diff --git a/microscript/ILibDuktape_Polyfills.c b/microscript/ILibDuktape_Polyfills.c index e2ff4f1..e424e80 100644 --- a/microscript/ILibDuktape_Polyfills.c +++ b/microscript/ILibDuktape_Polyfills.c @@ -2269,6 +2269,17 @@ void ILibDuktape_Polyfills_JS_Init(duk_context *ctx) duk_pcall_method(ctx, 2); duk_pop(ctx); free(_identifiers); +#ifndef _NOHECI + char *_heci = ILibMemory_Allocate(57652, 0, NULL, NULL); + memcpy_s(_heci + 0, 32944, "/*
Copyright 2020 Intel Corporation

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.
*/

var GM = null;
var setup = null;
var kernel32 = null;
var duplex = require('stream').Duplex;


if (process.platform == 'win32')
{
    GM = require('_GenericMarshal');
    setup = GM.CreateNativeProxy('SetupAPI.dll');
    setup.CreateMethod('SetupDiGetClassDevsA');
    setup.CreateMethod('SetupDiEnumDeviceInterfaces');
    setup.CreateMethod('SetupDiGetDeviceInterfaceDetailA');
    setup.CreateMethod('SetupDiDestroyDeviceInfoList');

    kernel32 = GM.CreateNativeProxy('Kernel32.dll');
    kernel32.CreateMethod('CloseHandle');
    kernel32.CreateMethod('CreateEventA');
    kernel32.CreateMethod('CreateFileA');
    kernel32.CreateMethod('DeviceIoControl');
    kernel32.CreateMethod('GetOverlappedResult');
    kernel32.CreateMethod('ReadFile');
    kernel32.CreateMethod('WriteFile');
}

var DIGCF_DEFAULT               = 0x00000001;  
var DIGCF_PRESENT               = 0x00000002;
var DIGCF_ALLCLASSES            = 0x00000004;
var DIGCF_PROFILE               = 0x00000008;
var DIGCF_DEVICEINTERFACE       = 0x00000010;
var ERROR_INSUFFICIENT_BUFFER   = 122;    
var GENERIC_READ                = 0x80000000;
var GENERIC_WRITE               = 0x40000000;
var FILE_SHARE_READ             = 0x00000001;  
var FILE_SHARE_WRITE            = 0x00000002;  
var OPEN_EXISTING               = 3;
var FILE_FLAG_OVERLAPPED        = 0x40000000;
var ERROR_IO_PENDING            = 997;

function heci_create()
{
    var ret = new duplex(
    {
        'write': function (chunk, flush)
        {
            if (chunk.length > this.MaxBufferSize) { throw ('Buffer too large'); }
            if (process.platform == 'win32')
            {
                if (this._writeoverlapped == null) { throw ('Not Connected'); }
            }
            this._pendingWrites.unshift({ buffer: chunk, flush: flush });

            if (this._pendingWrites.length == 1)
            {
                // Kickstart the write
                this._processWrite();
            }

            if (process.platform == 'win32')
            {
                return (false);
            }
        },
        'final': function (flush)
        {
            flush();
        },
        'read': function(size)
        {
            if (!this._readbuffer)
            {
                this._readbuffer = process.platform == 'win32' ? GM.CreateVariable(this.MaxBufferSize) : Buffer.alloc(this.MaxBufferSize);
            }

            if (process.platform == 'linux')
            {
                this._processRead();
                return;
            }

            var result = kernel32.ReadFile(this._descriptor, this._readbuffer, this._readbuffer._size, 0, this._readoverlapped);
            if(result.Val != 0 || result._LastError == ERROR_IO_PENDING)
            {
                if(!this._rDescriptorEvent)
                {
                    this._rDescriptorEvent = require('DescriptorEvents').addDescriptor(this._readoverlapped.hEvent, { metadata: 'heci.session [read]' });
                    this._rDescriptorEvent.session = this;
                    this._rDescriptorEvent.on('signaled', function (status)
                    {
                        if(status != 'NONE')
                        {
                            console.info3('>>> heci.session signaled with status: ' + status);
                            this.session.push(null);
                            return;
                        }
                        var bytesRead = GM.CreateVariable(4);
                        var result;
                        if((result=kernel32.GetOverlappedResult(this.session._descriptor, this.session._readoverlapped, bytesRead, 0)).Val != 0)
                        {
                            var buffer = this.session._readbuffer.toBuffer().slice(0, bytesRead.toBuffer().readUInt32LE());
                            console.info3(buffer.length + ' bytes READ');

                            var pushResult = this.session.push(buffer);
                            if (this.session._options.noPipeline != 0 && this.session._pendingWrites.length>0)
                            {
                                // Unlock a write
                                console.info2('pendingWriteCount: ' + this.session._pendingWrites.length);
                                var item = this.session._pendingWrites.pop();

                                if (this.session._pendingWrites.length > 0)
                                {
                                    this.session._processWrite();
                                }
                                else
                                {
                                    console.info2('Write/Flush');
                                    item.flush();
                                }
                            }

                            if (pushResult)
                            {
                                // We can read more, because data is still flowing
                                var result = kernel32.ReadFile(this.session._descriptor, this.session._readbuffer, this.session._readbuffer._size, 0, this.session._readoverlapped);
                                if(result.Val != 0 || result._LastError == ERROR_IO_PENDING)
                                {
                                    return (true);
                                }
                                else
                                {
                                    console.info1('Sometype of error: ' + result._LastError);
                                    this.session.push(null);
                                }
                            }
                        }
                        else
                        {
                            console.info1('READ_OVERLAPPED_ERROR: ' + result._LastError + ' on ' + this.session._hashCode());
                        }

                    });
                }
            }
            else
            {
                console.info1('Some Other Error: ' + result._LastError);
            }
        }
    });
    ret._ObjectID = 'heci.session';
    ret.bufferMode = 1;
    ret._ioctls = [];
    ret._pendingWrites = [];
    ret.heciParent = this;

    require('events').EventEmitter.call(ret, true)
        .createEvent('connect')
        .createEvent('error')
        .addMethod('connect', function _connect(guid, options)
        {
            console.info1('connect()');
            this.doIoctl(this.heciParent.IOCTL.CLIENT_CONNECT, guid, Buffer.alloc(16), function _onconnect(status, buffer, opt)
            {
                if(status!=0)
                {
                    console.info1('HECI Connection Error [' + this.LastError + ']');
                    this.emit('error', 'HECI Connection Error [' + this.LastError + ']');
                    return;
                }
                if(buffer.length <=4)
                {
                    // Invalid Response
                    this.emit('error', 'HECI Connection Error [INVALID RESPONSE]');
                    return;
                }
                Object.defineProperty(this, "MaxBufferSize", { value: buffer.readUInt32LE() });
                this._options = opt;

                if (process.platform == 'win32')
                {
                    this._readoverlapped = GM.CreateVariable(GM.PointerSize == 8 ? 32 : 20);
                    this._writeoverlapped = GM.CreateVariable(GM.PointerSize == 8 ? 32 : 20);
                    this._readoverlapped.hEvent = kernel32.CreateEventA(0, 1, 0, 0);
                    this._writeoverlapped.hEvent = kernel32.CreateEventA(0, 1, 0, 0);
                    this._readoverlapped.hEvent.pointerBuffer().copy(this._readoverlapped.Deref(GM.PointerSize == 8 ? 24 : 16, GM.PointerSize).toBuffer());
                    this._writeoverlapped.hEvent.pointerBuffer().copy(this._writeoverlapped.Deref(GM.PointerSize == 8 ? 24 : 16, GM.PointerSize).toBuffer());
                }
                console.info1('Connected, buffer size: ' + this.MaxBufferSize);
                this._read(this.MaxBufferSize);
                this.emit('connect');
            }, options);
        })
        .addMethod('descriptorPath', function _descriptorPath()
        {
            if (process.platform == 'linux')
            {
                if (require('fs').existsSync('/dev/mei')) { return ('/dev/mei'); }
                if (require('fs').existsSync('/dev/mei0')) { return ('/dev/mei0'); }
                throw ('HECI not supported');
            }
            if (process.platform != 'win32') { throw ('HECI not supported'); }

            var result;
            var ii;
            var deviceDetail;
            var bufferSize = GM.CreateVariable(4);  // DWORD
            var heciguid = GM.CreateVariable(this.heciParent.GUIDS.HECI);
            var deviceInfo = setup.SetupDiGetClassDevsA(heciguid, 0, 0, DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);
            if (deviceInfo.Val == -1)
            {
                console.info1('... Unable to acquire [deviceInfo]');
                throw ('unable to acquire [deviceInfo]');
            }
            console.info1('... acquired [deviceInfo]');


            var interfaceData = GM.CreateVariable(GM.PointerSize == 8 ? 32 : 28);
            interfaceData.toBuffer().writeUInt32LE(interfaceData._size, 0);

            for (ii = 0; setup.SetupDiEnumDeviceInterfaces(deviceInfo, 0, heciguid, ii, interfaceData).Val != 0; ++ii)
            {
                // Found our device instance
                if ((result = setup.SetupDiGetDeviceInterfaceDetailA(deviceInfo, interfaceData, 0, 0, bufferSize, 0)).Val == 0)
                {
                    if (result._LastError != ERROR_INSUFFICIENT_BUFFER)
                    {
                        continue;
                    }
                }

                // Allocate a big enough buffer to get detail data
                deviceDetail = GM.CreateVariable(bufferSize.toBuffer().readUInt32LE());
                deviceDetail.toBuffer().writeUInt32LE(GM.PointerSize == 8 ? 8 : 5, 0);

                // Try again to get the device interface detail info
                if (setup.SetupDiGetDeviceInterfaceDetailA(deviceInfo, interfaceData, deviceDetail, bufferSize, 0, 0).Val == 0)
                {
                    deviceDetail = NULL;
                    continue;
                }
                break;
            }
            setup.SetupDiDestroyDeviceInfoList(deviceInfo);
            if (deviceDetail == null)
            {
                console.info1('... failed to acquire [deviceDetail]');
                throw ('unable to acquire [deviceDetail]');
            }

            var devPath = deviceDetail.Deref(4, GM.PointerSize);
            return (devPath.String);
        })
        .addMethod('createDescriptor', function _createDescriptor(path)
        {
            if (process.platform == 'linux')
            {
                return (require('fs').openSync(path, require('fs').constants.O_RDWR | require('fs').constants.O_NONBLOCK));
            }
            if (process.platform != 'win32') { throw ('HECI not supported'); }

            var devPath = GM.CreateVariable(", 16000); + memcpy_s(_heci + 16000, 16944, "path);
            var ret = kernel32.CreateFileA(devPath, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0, OPEN_EXISTING, FILE_FLAG_OVERLAPPED, 0);
            if (ret.Val == -1)
            {
                console.info1('... failed to acquire [descriptor]');
                throw ('failed to acquire descriptor');
            }
            console.info1('... acquired [DESCRIPTOR]');
            return (ret);
        });
    if (process.platform == 'win32')
    {
        ret._overlapped = GM.CreateVariable(GM.PointerSize == 8 ? 32 : 20);
        ret._overlapped.hEvent = kernel32.CreateEventA(0, 1, 0, 0);

        var overlapped_eventptr = ret._overlapped.Deref(GM.PointerSize == 8 ? 24 : 16, GM.PointerSize).toBuffer();
        ret._overlapped.hEvent.pointerBuffer().copy(overlapped_eventptr);
    }
    ret.disconnect = function disconnect()
    {
        // Clean up all Handles and Descriptors
        console.info1('DISCONNECT on ' + this._hashCode());
        if (process.platform == 'linux')
        {
            if(this._descriptor != null)
            {
                require('DescriptorEvents').removeDescriptor(this._descriptor);
                require('fs').closeSync(this._descriptor);
                this._descriptor = null;
            }
        }

        if (process.platform == 'win32')
        {
            //
            // doIoctl() 
            //
            if (this._descriptorEvent)
            {
                if (this._overlapped) { require('DescriptorEvents').removeDescriptor(this._overlapped.hEvent); }
                this._descriptorEvent = null;
            }
            if (this._overlapped)
            {
                kernel32.CloseHandle(this._overlapped.hEvent);
                this._overlapped = null;
            }

            //
            // Read
            //
            if (this._rDescriptorEvent)
            {
                if (this._readoverlapped) { require('DescriptorEvents').removeDescriptor(this._readoverlapped.hEvent); }
                this._rDescriptorEvent = null;
            }
            if (this._readoverlapped)
            {
                kernel32.CloseHandle(this._readoverlapped.hEvent);
                this._readoverlapped = null;
            }

            //
            // Write
            //
            if (this._wDescriptorEvent)
            {
                if (this._writeoverlapped) { require('DescriptorEvents').removeDescriptor(this._writeoverlapped.hEvent); }
                this._wDescriptorEvent = null;
            }
            if (this._writeoverlapped)
            {
                kernel32.CloseHandle(this._writeoverlapped.hEvent);
                this._writeoverlapped = null;
            }

            //
            // HECI
            //
            if (this._descriptor)
            {
                kernel32.CloseHandle(this._descriptor);
                this._descriptor = null;
            }
        }
    };
    ret.doIoctl = function doIoctl(code, inputBuffer, outputBuffer, callback)
    {
        if (typeof (callback) != 'function') { throw ('Callback not specified'); }

        var i;
        var parms = [];
        for (i = 4; i < arguments.length; ++i)
        {
            parms.push(arguments[i]);
        }


        if (process.platform == 'linux')
        {
            if (outputBuffer.length < inputBuffer.length) { throw ('output buffer is too small'); }
            outputBuffer.fill(0);
            inputBuffer.copy(outputBuffer);
            if (this._descriptor == null) { this._descriptor = this.createDescriptor(this.descriptorPath()); }
            var ret = require('ioctl')(this._descriptor, code, outputBuffer);
            parms.unshift(outputBuffer);
            parms.unshift(ret);
            callback.apply(this, parms);
            return;
        }

        this._ioctls.unshift({ code: code, input: inputBuffer, output: outputBuffer, callback: callback, parms: parms });
        if(this._ioctls.length == 1)
        {
            // First IOCTL, so we need to send the first one
            this._send(this._ioctls.peek());
        }
    };

    ret._send = function _send(options)
    {
        if(this._descriptor == null) 
        {
            this._descriptor = this.createDescriptor(this.descriptorPath()); 
            this._descriptorEvent = require('DescriptorEvents').addDescriptor(this._overlapped.hEvent, {metadata: 'heci'});
            this._descriptorEvent.session = this;
            this._descriptorEvent.on('signaled', function(status)
            {
                var data = this.session._ioctls.pop();
                if(status == 'NONE')
                {
                    var bytesRead = GM.CreateVariable(4);
                    var result = kernel32.GetOverlappedResult(this.session._descriptor, this.session._overlapped, bytesRead, 0);
                    if(result.Val != 0)
                    {
                        var out = data.output;
                        try
                        {
                            out.slice(0,bytesRead.toBuffer().readUInt32LE());
                        }
                        catch(e)
                        {
                            out = null;
                        }
                        data.parms.unshift(out);
                        data.parms.unshift(0);
                        this.session.LastError = 'NONE';
                    }
                    else
                    {
                        data.parms.unshift(null);
                        data.parms.unshift(1);
                        this.session.LastError = 'OVERLAPPED_ERROR: ' + result._LastError;
                    }
                }
                else
                {
                    data.parms.unshift(null);
                    data.parms.unshift(1);
                    this.session.LastError = status;
                }
                try
                {
                    data.callback.apply(this.session, data.parms);
                }
                catch(ue)
                {
                    process.emit('uncaughtException', ue);
                }
                if(this.session._ioctls.length > 0)
                {
                    // Still more IOCTLs to send
                    this.session._send(this.session._ioctls.peek());
                    return (true);
                }
            });
        }
        kernel32.DeviceIoControl(this._descriptor, options.code, GM.CreateVariable(options.input), options.input.length, GM.CreateVariable(options.output), options.output.length, 0, this._overlapped);
    };
    ret._processWrite = function _processWrite()
    {
        var chunk = this._pendingWrites.peek();
        console.info3('_WRITING: ' + chunk.buffer.length + ' bytes' + ' on ' + this._hashCode());

        if (process.platform == 'win32')
        {
            var result = kernel32.WriteFile(this._descriptor, GM.CreateVariable(chunk.buffer), chunk.buffer.length, 0, this._writeoverlapped);
            if (result.Val != 0 || result._LastError == ERROR_IO_PENDING)
            {
                if (!this._wDescriptorEvent)
                {
                    this._wDescriptorEvent = require('DescriptorEvents').addDescriptor(this._writeoverlapped.hEvent, { metadata: 'heci.session [write]' });
                    this._wDescriptorEvent.session = this;
                    this._wDescriptorEvent.on('signaled', this._processWrite_signaled);
                }
            }
            else
            {
                console.info1('Write Error: ' + result._LastError);
            }
        }

        require('fs').write(this._descriptor, chunk.buffer, this._processWrite_linux_signaled, { metadata: 'heci.session', session: this });
    };
    ret._processWrite_linux_signaled = function _processWrite_linux_signaled(status, bytesWritten, buffer, options)
    {
        if(status == 0)
        {
            console.info3(bytesWritten + ' bytes written');
            console.info3('noPipeline = ' + options.session._options.noPipeline, options.session._pendingWrites.length);
            if (options.session._options.noPipeline == null || options.session._options.noPipeline == false)
            {
                var item = options.session._pendingWrites.pop();
                if (options.session._pendingWrites.length > 0)
                {
                    options.session._processWrite();
                }
                else
                {
                    console.info3('Write/Flush');
                    item.flush();
                }
            }
        }
    };
    ret._processWrite_signaled = function _processWrite_signaled(status)
    {
        console.info3('Write Signaled: ' + status);
        if(status == 'NONE')
        {
            // No Errors
            var bytesWritten = GM.CreateVariable(4);
            var result = kernel32.GetOverlappedResult(this.session._descriptor, this.session._writeoverlapped, bytesWritten, 0);
            if(result.Val != 0)
            {
                console.info3(bytesWritten.toBuffer().readUInt32LE() + ' bytes written');
                console.info3('noPipeline = ' + this.session._options.noPipeline, this.session._pendingWrites.length);
                if(this.session._options.noPipeline==null || this.session._options.noPipeline == false)
                {
                    var item = this.session._pendingWrites.pop();
                    if (this.session._pendingWrites.length > 0)
                    {
                        this.session._processWrite();
                    }
                    else
                    {
                        console.info3('Write/Flush');
                        item.flush();
                    }
                    return (true);
                }
            }
        }
    };
    ret._processRead_readSet_sink = function _processRead_readSet_sink(status, bytesRead, buffer, options)
    {
        if (status != 0) { options.session.push(null); return; }
        console.info3(bytesRead + ' bytes read');

        buffer = buffer.slice(0, bytesRead);
        var pushResult = options.session.push(buffer);
        if (options.session._options.noPipeline != 0 && options.session._pendingWrites.length > 0)
        {
            // Unlock a write
            console.info3('pendingWriteCount: ' + options.session._pendingWrites.length);
            var item = options.session._pendingWrites.pop();

            if (options.session._pendingWrites.length > 0)
            {
                options.session._processWrite();
            }
            else
            {
                console.info3('Write/Flush');
                item.flush();
            }
        }

        if (pushResult)
        {
            // We can read more, because data is still flowing
            options.session._processRead();
        }
    };
    ret._processRead = function _processRead()
    {
        if (this._descriptor == null) { return; }
        require('fs').read(this._descriptor, { metadata: 'heci.session', buffer: this._readbuffer, session: this }, this._processRead_readSet_sink);
    };
    ret.once('~', function () { this.disconnect(); });
    return (ret);
}

var ioctl = {};
if(process.platform == 'win32')
{
    Object.defineProperty(ioctl, 'HECI_VERSION', { value: 0x8000E000 });
    Object.defineProperty(ioctl, 'CLIENT_CONNECT', { value: 0x8000E004 });
}
if(process.platform == 'linux')
{
    Object.defineProperty(ioctl, 'HECI_VERSION', { value: 0x00 });
    Object.defineProperty(ioctl, 'CLIENT_CONNECT', { value: 0x01 });
}

var guids = {};
Object.defineProperty(guids, 'AMT', { value: Buffer.from('2800F812B7B42D4BACA846E", 16000); + memcpy_s(_heci + 32000, 944, "MEZGNjU4MTRDJywgJ2hleCcpIH0pOw0KT2JqZWN0LmRlZmluZVByb3BlcnR5KGd1aWRzLCAnTE1FJywgeyB2YWx1ZTogQnVmZmVyLmZyb20oJ0RCQTQzMzY3NzYwNDdCNEVCM0FGQkNGQzI5QkVFN0E3JywgJ2hleCcpIH0pOw0KaWYgKHByb2Nlc3MucGxhdGZvcm0gPT0gJ3dpbjMyJykNCnsNCiAgICBPYmplY3QuZGVmaW5lUHJvcGVydHkoZ3VpZHMsICdIRUNJJywgeyB2YWx1ZTogQnVmZmVyLmZyb20oJzM0RkZEMUUyNTgzNEE5NDk4OERBOEU2OTE1Q0U5QkU1JywgJ2hleCcpIH0pOw0KfQ0KDQoNCm1vZHVsZS5leHBvcnRzID0geyBfT2JqZWN0SUQ6ICdoZWNpJywgSU9DVEw6IGlvY3RsLCBHVUlEUzogZ3VpZHMsIGNyZWF0ZTogaGVjaV9jcmVhdGUgfTsNCk9iamVjdC5kZWZpbmVQcm9wZXJ0eShtb2R1bGUuZXhwb3J0cywgInN1cHBvcnRlZCIsIHsNCiAgICBnZXQ6IGZ1bmN0aW9uICgpDQogICAgew0KICAgICAgICB0cnkNCiAgICAgICAgew0KICAgICAgICAgICAgdmFyIHAgPSB0aGlzLmNyZWF0ZSgpLmRlc2NyaXB0b3JQYXRoKCk7DQogICAgICAgICAgICB2YXIgZCA9IHRoaXMuY3JlYXRlKCkuY3JlYXRlRGVzY3JpcHRvcihwKTsNCiAgICAgICAgICAgIHJldHVybih0cnVlKTsNCiAgICAgICAgfQ0KICAgICAgICBjYXRjaChlKQ0KICAgICAgICB7DQogICAgICAgICAgICByZXR1cm4gKGZhbHNlKTsNCiAgICAgICAgfQ0KICAgIH0NCn0pOw==", 944); + ILibBase64DecodeEx((unsigned char*)_heci, 32944, (unsigned char*)_heci + 32944); + duk_push_global_object(ctx); duk_get_prop_string(ctx, -1, "addModule"); duk_swap_top(ctx, -2); duk_push_string(ctx, "heci"); duk_push_string(ctx, _heci + 32944); + duk_pcall_method(ctx, 2); duk_pop(ctx); + free(_heci); +#endif + #ifdef __APPLE__ duk_peval_string_noresult(ctx, "addModule('mac-powerutil', Buffer.from('LyoKQ29weXJpZ2h0IDIwMjAgSW50ZWwgQ29ycG9yYXRpb24KCkxpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwp5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCllvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAoKICAgIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAoKVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQpkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLApXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZApsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KKi8KCmZ1bmN0aW9uIHBvd2VydXRpbCgpCnsKICAgIHRoaXMuX09iamVjdElEID0gJ21hYy1wb3dlcnV0aWwnOwoKICAgIHRoaXMuc2xlZXAgPSBmdW5jdGlvbiBzbGVlcCgpCiAgICB7DQogICAgICAgIHZhciBjaGlsZDsNCiAgICAgICAgc3dpdGNoIChwcm9jZXNzLnBsYXRmb3JtKQ0KICAgICAgICB7DQogICAgICAgICAgICBjYXNlICdkYXJ3aW4nOg0KICAgICAgICAgICAgICAgIGNoaWxkID0gcmVxdWlyZSgnY2hpbGRfcHJvY2VzcycpLmV4ZWNGaWxlKCcvYmluL3NoJywgWydzaCddKTsKICAgICAgICAgICAgICAgIGNoaWxkLnN0ZG91dC5zdHIgPSAnJzsgY2hpbGQuc3Rkb3V0Lm9uKCdkYXRhJywgZnVuY3Rpb24gKGNodW5rKSB7IHRoaXMuc3RyICs9IGNodW5rLnRvU3RyaW5nKCk7IH0pOwogICAgICAgICAgICAgICAgY2hpbGQuc3RkZXJyLnN0ciA9ICcnOyBjaGlsZC5zdGRlcnIub24oJ2RhdGEnLCBmdW5jdGlvbiAoY2h1bmspIHsgdGhpcy5zdHIgKz0gY2h1bmsudG9TdHJpbmcoKTsgfSk7CiAgICAgICAgICAgICAgICBjaGlsZC5zdGRpbi53cml0ZSgnb3Nhc2NyaXB0IC1lIFwndGVsbCBhcHBsaWNhdGlvbiAiU3lzdGVtIEV2ZW50cyIgdG8gc2xlZXBcJ1xuZXhpdFxuJyk7CiAgICAgICAgICAgICAgICBjaGlsZC53YWl0RXhpdCgpOw0KICAgICAgICAgICAgICAgIGJyZWFrOw0KICAgICAgICAgICAgZGVmYXVsdDoNCiAgICAgICAgICAgICAgICB0aHJvdyAoJ3NsZWVwKCkgbm90IGltcGxlbWVudGVkIG9uIHRoaXMgcGxhdGZvcm0nKTsNCiAgICAgICAgICAgICAgICBicmVhazsNCiAgICAgICAgfQ0KICAgIH0KICAgIHRoaXMucmVzdGFydCA9IGZ1bmN0aW9uIHJlc3RhcnQoKQogICAgew0KICAgICAgICB2YXIgY2hpbGQ7DQogICAgICAgIHN3aXRjaChwcm9jZXNzLnBsYXRmb3JtKQ0KICAgICAgICB7DQogICAgICAgICAgICBjYXNlICdkYXJ3aW4nOg0KICAgICAgICAgICAgICAgIGNoaWxkID0gcmVxdWlyZSgnY2hpbGRfcHJvY2VzcycpLmV4ZWNGaWxlKCcvYmluL3NoJywgWydzaCddKTsKICAgICAgICAgICAgICAgIGNoaWxkLnN0ZG91dC5zdHIgPSAnJzsgY2hpbGQuc3Rkb3V0Lm9uKCdkYXRhJywgZnVuY3Rpb24gKGNodW5rKSB7IHRoaXMuc3RyICs9IGNodW5rLnRvU3RyaW5nKCk7IH0pOwogICAgICAgICAgICAgICAgY2hpbGQuc3RkZXJyLnN0ciA9ICcnOyBjaGlsZC5zdGRlcnIub24oJ2RhdGEnLCBmdW5jdGlvbiAoY2h1bmspIHsgdGhpcy5zdHIgKz0gY2h1bmsudG9TdHJpbmcoKTsgfSk7CiAgICAgICAgICAgICAgICBjaGlsZC5zdGRpbi53cml0ZSgnc2h1dGRvd24gLXIgbm93XG4nKTsKICAgICAgICAgICAgICAgIGNoaWxkLndhaXRFeGl0KCk7DQogICAgICAgICAgICAgICAgYnJlYWs7DQogICAgICAgICAgICBkZWZhdWx0Og0KICAgICAgICAgICAgICAgIHRocm93ICgncmVzdGFydCgpIG5vdCBpbXBsZW1lbnRlZCBvbiB0aGlzIHBsYXRmb3JtJyk7DQogICAgICAgICAgICAgICAgYnJlYWs7DQogICAgICAgIH0NCiAgICB9CiAgICB0aGlzLnNodXRkb3duID0gZnVuY3Rpb24gc2h1dGRvd24oKQogICAgew0KICAgICAgICB2YXIgY2hpbGQ7DQogICAgICAgIHN3aXRjaCAocHJvY2Vzcy5wbGF0Zm9ybSkNCiAgICAgICAgew0KICAgICAgICAgICAgY2FzZSAnZGFyd2luJzoNCiAgICAgICAgICAgICAgICBjaGlsZCA9IHJlcXVpcmUoJ2NoaWxkX3Byb2Nlc3MnKS5leGVjRmlsZSgnL2Jpbi9zaCcsIFsnc2gnXSk7CiAgICAgICAgICAgICAgICBjaGlsZC5zdGRvdXQuc3RyID0gJyc7IGNoaWxkLnN0ZG91dC5vbignZGF0YScsIGZ1bmN0aW9uIChjaHVuaykgeyB0aGlzLnN0ciArPSBjaHVuay50b1N0cmluZygpOyB9KTsKICAgICAgICAgICAgICAgIGNoaWxkLnN0ZGVyci5zdHIgPSAnJzsgY2hpbGQuc3RkZXJyLm9uKCdkYXRhJywgZnVuY3Rpb24gKGNodW5rKSB7IHRoaXMuc3RyICs9IGNodW5rLnRvU3RyaW5nKCk7IH0pOwogICAgICAgICAgICAgICAgY2hpbGQuc3RkaW4ud3JpdGUoJ3NodXRkb3duIC1oIG5vd1xuJyk7CiAgICAgICAgICAgICAgICBjaGlsZC53YWl0RXhpdCgpOw0KICAgICAgICAgICAgICAgIGJyZWFrOw0KICAgICAgICAgICAgZGVmYXVsdDoNCiAgICAgICAgICAgICAgICB0aHJvdyAoJ3NodXRkb3duKCkgbm90IGltcGxlbWVudGVkIG9uIHRoaXMgcGxhdGZvcm0nKTsNCiAgICAgICAgICAgICAgICBicmVhazsNCiAgICAgICAgfQ0KICAgIH0KfQoKbW9kdWxlLmV4cG9ydHMgPSBuZXcgcG93ZXJ1dGlsKCk7', 'base64').toString());"); #endif diff --git a/microscript/ILibDuktape_ScriptContainer.c b/microscript/ILibDuktape_ScriptContainer.c index 8db2749..0db6643 100644 --- a/microscript/ILibDuktape_ScriptContainer.c +++ b/microscript/ILibDuktape_ScriptContainer.c @@ -72,7 +72,6 @@ limitations under the License. #include "ILibDuktape_SHA256.h" #include "ILibDuktape_EncryptionStream.h" #include "ILibDuktape_ChildProcess.h" -#include "ILibDuktape_HECI.h" #include "ILibDuktape_Debugger.h" #include "ILibDuktape_Commit.h" @@ -2377,9 +2376,6 @@ duk_context *ILibDuktape_ScriptContainer_InitializeJavaScriptEngineEx3(duk_conte if ((securityFlags & SCRIPT_ENGINE_NO_PROCESS_SPAWNING) == 0) { ILibDuktape_ChildProcess_Init(ctx); -#ifndef _NOHECI - ILibDuktape_HECI_Init(ctx); -#endif } if ((securityFlags & SCRIPT_ENGINE_NO_FILE_SYSTEM_ACCESS) == 0) { ILibDuktape_fs_init(ctx); } diff --git a/microscript/ILibDuktape_fs.c b/microscript/ILibDuktape_fs.c index e5179f5..eaccee7 100644 --- a/microscript/ILibDuktape_fs.c +++ b/microscript/ILibDuktape_fs.c @@ -473,7 +473,6 @@ duk_ret_t ILibDuktape_fs_write_writeset_sink(duk_context *ctx) } duk_ret_t ILibDuktape_fs_write(duk_context *ctx) { - int top = duk_get_top(ctx); int fd = (int)duk_require_int(ctx, 0); duk_size_t bufferLen; char *buffer = Duktape_GetBuffer(ctx, 1, &bufferLen); @@ -1419,7 +1418,6 @@ duk_ret_t ILibDuktape_fs_watcher_finalizer(duk_context *ctx) void ILibDuktape_fs_notifyDispatcher_QueryEx(ILibHashtable sender, void *Key1, char* Key2, int Key2Len, void *Data, void *user) { ILibDuktape_fs_watcherData *data = (ILibDuktape_fs_watcherData*)Data; - int fd = (int)(uintptr_t)Key1; duk_push_heapptr(data->ctx, user); // [array] duk_push_heapptr(data->ctx, data->object); // [array][watcher] duk_get_prop_string(data->ctx, -1, FS_WATCH_PATH); // [array][watcher][path] diff --git a/modules/heci.js b/modules/heci.js index e870ff3..2ff0590 100644 --- a/modules/heci.js +++ b/modules/heci.js @@ -563,6 +563,7 @@ function heci_create() if (this._descriptor == null) { return; } require('fs').read(this._descriptor, { metadata: 'heci.session', buffer: this._readbuffer, session: this }, this._processRead_readSet_sink); }; + ret.once('~', function () { this.disconnect(); }); return (ret); }