834 lines
31 KiB
C
834 lines
31 KiB
C
/*********************************************************************
|
|
* Copyright (c) Intel Corporation 2011 - 2020
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
**********************************************************************/
|
|
|
|
/*
|
|
Real LMS code can be found here: http://software.intel.com/en-us/articles/download-the-latest-intel-amt-open-source-drivers
|
|
*/
|
|
|
|
#if !defined(_NOHECI)
|
|
|
|
#if defined(WIN32) && !defined(_MINCORE)
|
|
#define _CRTDBG_MAP_ALLOC
|
|
#include <stdlib.h>
|
|
#include <crtdbg.h>
|
|
#else
|
|
#include <stdlib.h>
|
|
#endif
|
|
|
|
#if defined(WINSOCK2)
|
|
#include <winsock2.h>
|
|
#include <ws2ipdef.h>
|
|
#elif defined(WINSOCK1)
|
|
#include <winsock.h>
|
|
#include <wininet.h>
|
|
#endif
|
|
|
|
#include "ILibParsers.h"
|
|
#include "ILibAsyncSocket.h"
|
|
#include "ILibAsyncServerSocket.h"
|
|
|
|
#ifdef WIN32
|
|
#include "../heci/heciwin.h"
|
|
#include "WinBase.h"
|
|
#endif
|
|
#ifdef _POSIX
|
|
#include "../heci/HECILinux.h"
|
|
#define UNREFERENCED_PARAMETER(P) (P)
|
|
#endif
|
|
#include "../heci/PTHICommand.h"
|
|
#include "../heci/LMEConnection.h"
|
|
|
|
#define INET_SOCKADDR_LENGTH(x) ((x==AF_INET6?sizeof(struct sockaddr_in6):sizeof(struct sockaddr_in)))
|
|
#define LMS_MAX_CONNECTIONS 16 // Maximum is 32
|
|
#define LMS_MAX_SESSIONS 32
|
|
// #define _DEBUGLMS
|
|
#ifdef _DEBUGLMS
|
|
#define LMSDEBUG(...) printf(__VA_ARGS__);
|
|
#else
|
|
#define LMSDEBUG(...)
|
|
#endif
|
|
|
|
char * g_SelfLMSDir=NULL;
|
|
int LastId = 2;
|
|
|
|
struct ILibLMS_StateModule* IlibExternLMS = NULL; // Glocal pointer to the LMS module. Since we can only have one of these modules running, a global pointer is sometimes useful.
|
|
void ILibLMS_SetupConnection(struct ILibLMS_StateModule* module, int i);
|
|
void ILibLMS_LaunchHoldingSessions(struct ILibLMS_StateModule* module);
|
|
|
|
// Each LMS session can be in one of these states.
|
|
enum LME_CHANNEL_STATUS {
|
|
LME_CS_FREE = 0, // The session slot is free, can be used at any time.
|
|
LME_CS_PENDING_CONNECT = 1, // A connection as been made to LMS and a notice has been send to Intel AMT asking for a CHANNEL OPEN.
|
|
LME_CS_CONNECTED = 2, // Intel AMT confirmed the connection and data can flow in both directions.
|
|
LME_CS_PENDING_LMS_DISCONNECT = 3, // The connection to LMS was disconnected, Intel AMT has been notified and we are waitting for Intel AMT confirmation.
|
|
LME_CS_PENDING_AMT_DISCONNECT = 4, // Intel AMT decided to close the connection. We are disconnecting the LMS TCP connection.
|
|
LME_CS_HOLDING = 5, // We got too much data from the LMS TCP connection, more than Intel AMT can handle. We are holding the connection until AMT can handle more.
|
|
LME_CS_CONNECTION_WAIT = 6 // A TCP connection to LMS was made, but there all LMS connections are currently busy. Waitting for one to free up.
|
|
};
|
|
|
|
// This is the structure for a session
|
|
struct LMEChannel
|
|
{
|
|
int ourid; // The iddentifier of this channel on our side, this is the same as the index in the "Channel Sessions[LMS_MAX_SESSIONS];" array below.
|
|
int amtid; // The Intel AMT identifier for this channel.
|
|
enum LME_CHANNEL_STATUS status; // Current status of the channel.
|
|
int sockettype; // Type of connected socket: 0 = TCP, 1 = WebSocket
|
|
void* socketmodule; // The microstack associated with the LMS connection.
|
|
int txwindow; // Transmit window.
|
|
int rxwindow; // Receive window.
|
|
unsigned short localport; // The Intel AMT connection port.
|
|
int errorcount; // Open channel error count.
|
|
char* pending; // Buffer pointing to pending data that needs to be sent to Intel AMT, used for websocket only.
|
|
int pendingcount; // Buffer pointing to pending data size that needs to be sent to Intel AMT, used for websocket only.
|
|
int pendingptr; // Pointer to start of the pending buffer.
|
|
};
|
|
|
|
enum LME_VERSION_HANDSHAKING {
|
|
LME_NOT_INITIATED,
|
|
LME_INITIATED,
|
|
LME_AGREED
|
|
};
|
|
|
|
enum LME_SERVICE_STATUS {
|
|
LME_NOT_STARTED,
|
|
LME_STARTED
|
|
};
|
|
|
|
// This is the Intel AMT LMS chain module structure
|
|
struct ILibLMS_StateModule
|
|
{
|
|
ILibChain_Link ChainLink;
|
|
void *Server1; // Microstack TCP server for port 16992
|
|
void *Server2; // Microstack TCP server for port 16993
|
|
void *ControlSocket; // Pointer to the LMS control web socket, NULL if not connected.
|
|
|
|
struct LMEConnection MeConnection; // The MEI connection structure
|
|
struct LMEChannel Sessions[LMS_MAX_SESSIONS]; // List of sessions
|
|
//ILibWebServer_ServerToken WebServer; // LMS Web Server
|
|
enum LME_VERSION_HANDSHAKING handshakingStatus;
|
|
enum LME_SERVICE_STATUS pfwdService;
|
|
unsigned int AmtProtVersionMajor; // Intel AMT MEI major version
|
|
unsigned int AmtProtVersionMinor; // Intel AMT MEI minor version
|
|
sem_t Lock; // Global lock, this is needed because MEI listener is a different thread from the microstack thread
|
|
};
|
|
|
|
/*
|
|
// Convert a block of data to HEX
|
|
// The "out" must have (len*2)+1 free space.
|
|
char utils_HexTable[16] = { '0','1','2','3','4','5','6','7','8','9','a','b','c','d','e','f' };
|
|
char* __fastcall util_tohex(char* data, int len, char* out)
|
|
{
|
|
int i;
|
|
char *p = out;
|
|
if (data == NULL || len == 0) { *p = 0; return NULL; }
|
|
for (i = 0; i < len; i++)
|
|
{
|
|
*(p++) = utils_HexTable[((unsigned char)data[i]) >> 4];
|
|
*(p++) = utils_HexTable[((unsigned char)data[i]) & 0x0F];
|
|
}
|
|
*p = 0;
|
|
return out;
|
|
}
|
|
*/
|
|
|
|
int GetFreeSlot(struct ILibLMS_StateModule* module){
|
|
int i;
|
|
int cur = LastId;
|
|
// Look for an empty session
|
|
for (i = 0; i < LMS_MAX_SESSIONS-3; i++) {
|
|
cur++;
|
|
if (cur >= LMS_MAX_SESSIONS){
|
|
cur = 3;
|
|
}
|
|
if (module->Sessions[cur].status == LME_CS_FREE) {
|
|
LastId = cur;
|
|
return cur;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
//int GetFreeSlot(struct ILibLMS_StateModule* module){
|
|
// int i;
|
|
// for (i = 0; i < LMS_MAX_SESSIONS - 3; i++) {
|
|
// if (cur >= LMS_MAX_SESSIONS){
|
|
// cur = 3;
|
|
// }
|
|
// if (module->Sessions[cur].status == LME_CS_FREE) {
|
|
// LastId = cur;
|
|
// return cur;
|
|
// }
|
|
// }
|
|
// return -1;
|
|
//}
|
|
|
|
// This function is called each time a MEI message is received from Intel AMT
|
|
void ILibLMS_MEICallback(struct LMEConnection* lmemodule, void *param, void *rxBuffer, unsigned int len)
|
|
{
|
|
struct ILibLMS_StateModule* module = (struct ILibLMS_StateModule*)param;
|
|
int disconnects = 0; // This is a counter of how many sessions are freed up. We use this at the end to place HOLDING sessions into free slots.
|
|
|
|
//ILibAsyncServerSocket_ServerModule newModule;
|
|
//ILibLinkedList list;
|
|
//void* node;
|
|
//ILibTransport* curModule;
|
|
|
|
// Happens when the chain is being destroyed, don't call anything chain related.
|
|
if (rxBuffer == NULL) return;
|
|
|
|
sem_wait(&(module->Lock));
|
|
LMSDEBUG("ILibLMS_MEICallback %d\r\n", ((unsigned char*)rxBuffer)[0]);
|
|
|
|
switch (((unsigned char*)rxBuffer)[0])
|
|
{
|
|
case APF_GLOBAL_REQUEST: // 80
|
|
{
|
|
|
|
int request = 0;
|
|
unsigned char *pCurrent;
|
|
APF_GENERIC_HEADER *pHeader = (APF_GENERIC_HEADER *)rxBuffer;
|
|
|
|
pHeader->StringLength = ntohl(pHeader->StringLength);
|
|
|
|
if (pHeader->StringLength == APF_STR_SIZE_OF(APF_GLOBAL_REQUEST_STR_TCP_FORWARD_REQUEST) && memcmp(pHeader->String, APF_GLOBAL_REQUEST_STR_TCP_FORWARD_REQUEST, APF_STR_SIZE_OF(APF_GLOBAL_REQUEST_STR_TCP_FORWARD_REQUEST)) == 0) { request = 1; }
|
|
else if (pHeader->StringLength == APF_STR_SIZE_OF(APF_GLOBAL_REQUEST_STR_TCP_FORWARD_CANCEL_REQUEST) && memcmp(pHeader->String, APF_GLOBAL_REQUEST_STR_TCP_FORWARD_CANCEL_REQUEST, APF_STR_SIZE_OF(APF_GLOBAL_REQUEST_STR_TCP_FORWARD_CANCEL_REQUEST)) == 0) { request = 2; }
|
|
else if (pHeader->StringLength == APF_STR_SIZE_OF(APF_GLOBAL_REQUEST_STR_UDP_SEND_TO) && memcmp(pHeader->String, APF_GLOBAL_REQUEST_STR_UDP_SEND_TO, APF_STR_SIZE_OF(APF_GLOBAL_REQUEST_STR_UDP_SEND_TO)) == 0) { request = 3; }
|
|
|
|
|
|
if (request == 1 || request == 2)
|
|
{
|
|
int port = 0;
|
|
unsigned int len2;
|
|
unsigned int bytesRead = len;
|
|
unsigned int hsize=0;
|
|
if (request==1)
|
|
hsize = sizeof(APF_GENERIC_HEADER) + APF_STR_SIZE_OF(APF_GLOBAL_REQUEST_STR_TCP_FORWARD_REQUEST) + sizeof(UINT8);
|
|
else if (request==2)
|
|
hsize = sizeof(APF_GENERIC_HEADER) + APF_STR_SIZE_OF(APF_GLOBAL_REQUEST_STR_TCP_FORWARD_CANCEL_REQUEST) + sizeof(UINT8);
|
|
|
|
pCurrent = (unsigned char*)rxBuffer + hsize;
|
|
bytesRead -= hsize;
|
|
if (bytesRead < sizeof(unsigned int)) {
|
|
LME_Deinit(lmemodule);
|
|
sem_post(&(module->Lock));
|
|
return;
|
|
}
|
|
|
|
len2 = ntohl(*((unsigned int *)pCurrent));
|
|
pCurrent += sizeof(unsigned int);
|
|
if (bytesRead < (sizeof(unsigned int) + len2 + sizeof(unsigned int))) {
|
|
LME_Deinit(lmemodule);
|
|
sem_post(&(module->Lock));
|
|
return;
|
|
}
|
|
|
|
// addr = (char*)pCurrent;
|
|
pCurrent += len2;
|
|
port = ntohl(*((unsigned int *)pCurrent));
|
|
|
|
if (request == 1)
|
|
{
|
|
//newModule = ILibCreateAsyncServerSocketModule(module->Chain, LMS_MAX_CONNECTIONS, port, 4096, 0, &ILibLMS_OnConnect, &ILibLMS_OnDisconnect, &ILibLMS_OnReceive, NULL, &ILibLMS_OnSendOK);
|
|
//if (newModule){
|
|
// ILibAsyncServerSocket_SetTag(newModule, module);
|
|
// LME_TcpForwardReplySuccess(lmemodule, port);
|
|
//}
|
|
//else{
|
|
// LME_TcpForwardReplyFailure(lmemodule);
|
|
//}
|
|
|
|
// New forwarding request
|
|
if (port == 16992 || port == 16993)
|
|
LME_TcpForwardReplySuccess(lmemodule, port);
|
|
else
|
|
LME_TcpForwardReplyFailure(lmemodule);
|
|
}
|
|
else
|
|
{
|
|
// Cancel a forwarding request
|
|
|
|
|
|
|
|
//list = ILibGetModuleList(module->Chain);
|
|
//node = ILibLinkedList_GetNode_Head(list);
|
|
//while (node != NULL){
|
|
// curModule = (ILibTransport*)ILibLinkedList_GetDataFromNode(node);
|
|
// if (curModule->IdentifierFlags == ILibTransports_ILibAsyncServerSocket){
|
|
// if (ILibAsyncServerSocket_GetPortNumber(curModule)){
|
|
// ILibAsyncServerSocket_Close(curModule);
|
|
// }
|
|
// }
|
|
// node = ILibLinkedList_GetNextNode(node);
|
|
//}
|
|
|
|
|
|
LME_TcpForwardCancelReplySuccess(lmemodule);
|
|
}
|
|
}
|
|
else if (request == 3)
|
|
{
|
|
// Send a UDP packet
|
|
// TODO: Send UDP
|
|
}
|
|
else{
|
|
// do nothing
|
|
}
|
|
}
|
|
break;
|
|
case APF_CHANNEL_OPEN: // (90) Sent by Intel AMT when a channel needs to be open from Intel AMT. This is not common, but WSMAN events are a good example of channel coming from AMT.
|
|
{
|
|
APF_GENERIC_HEADER *pHeader = (APF_GENERIC_HEADER *)rxBuffer;
|
|
|
|
if (pHeader->StringLength == APF_STR_SIZE_OF(APF_OPEN_CHANNEL_REQUEST_DIRECT) && memcmp((char*)pHeader->String, APF_OPEN_CHANNEL_REQUEST_DIRECT, APF_STR_SIZE_OF(APF_OPEN_CHANNEL_REQUEST_DIRECT)) == 0)
|
|
{
|
|
unsigned int len2;
|
|
unsigned char *pCurrent;
|
|
struct LMEChannelOpenRequestMessage channelOpenRequest;
|
|
|
|
if (len < sizeof(APF_GENERIC_HEADER) + ntohl(pHeader->StringLength) + 7 + (5 * sizeof(UINT32))) { LME_Deinit(lmemodule); sem_post(&(module->Lock)); return; }
|
|
pCurrent = (unsigned char*)rxBuffer + sizeof(APF_GENERIC_HEADER) + APF_STR_SIZE_OF(APF_OPEN_CHANNEL_REQUEST_DIRECT);
|
|
|
|
channelOpenRequest.ChannelType = APF_CHANNEL_DIRECT;
|
|
channelOpenRequest.SenderChannel = ntohl(*((UINT32 *)pCurrent));
|
|
pCurrent += sizeof(UINT32);
|
|
channelOpenRequest.InitialWindow = ntohl(*((UINT32 *)pCurrent));
|
|
pCurrent += 2 * sizeof(UINT32);
|
|
len2 = ntohl(*((UINT32 *)pCurrent));
|
|
pCurrent += sizeof(UINT32);
|
|
channelOpenRequest.Address = (char*)pCurrent;
|
|
pCurrent += len2;
|
|
channelOpenRequest.Port = ntohl(*((UINT32 *)pCurrent));
|
|
pCurrent += sizeof(UINT32);
|
|
|
|
LME_ChannelOpenReplyFailure(lmemodule, channelOpenRequest.SenderChannel, OPEN_FAILURE_REASON_CONNECT_FAILED);
|
|
}
|
|
}
|
|
break;
|
|
case APF_DISCONNECT: // (1) Intel AMT wants to completely disconnect. Not sure when this happens.
|
|
{
|
|
// First, we decode the message.
|
|
struct LMEDisconnectMessage disconnectMessage;
|
|
disconnectMessage.MessageType = APF_DISCONNECT;
|
|
disconnectMessage.ReasonCode = ((APF_DISCONNECT_REASON_CODE)ntohl(((APF_DISCONNECT_MESSAGE *)rxBuffer)->ReasonCode));
|
|
LMSDEBUG("LME requested to disconnect with reason code 0x%08x\r\n", disconnectMessage.ReasonCode);
|
|
LME_Deinit(lmemodule);
|
|
}
|
|
break;
|
|
case APF_SERVICE_REQUEST: // (5)
|
|
{
|
|
|
|
int service = 0;
|
|
APF_SERVICE_REQUEST_MESSAGE *pMessage = (APF_SERVICE_REQUEST_MESSAGE *)rxBuffer;
|
|
pMessage->ServiceNameLength = ntohl(pMessage->ServiceNameLength);
|
|
if (pMessage->ServiceNameLength == 18) {
|
|
if (memcmp(pMessage->ServiceName, "pfwd@amt.intel.com", 18) == 0) service = 1;
|
|
else if (memcmp(pMessage->ServiceName, "auth@amt.intel.com", 18) == 0) service = 2;
|
|
}
|
|
|
|
if (service > 0)
|
|
{
|
|
if (service == 1)
|
|
{
|
|
LME_ServiceAccept(lmemodule, "pfwd@amt.intel.com");
|
|
module->pfwdService = LME_STARTED;
|
|
}
|
|
else if (service == 2)
|
|
{
|
|
LME_ServiceAccept(lmemodule, "auth@amt.intel.com");
|
|
}
|
|
} else {
|
|
LMSDEBUG("APF_SERVICE_REQUEST - APF_DISCONNECT_SERVICE_NOT_AVAILABLE\r\n");
|
|
LME_Disconnect(lmemodule, APF_DISCONNECT_SERVICE_NOT_AVAILABLE);
|
|
LME_Deinit(lmemodule);
|
|
sem_post(&(module->Lock));
|
|
return;
|
|
}
|
|
}
|
|
break;
|
|
case APF_CHANNEL_OPEN_CONFIRMATION: // (91) Intel AMT confirmation to an APF_CHANNEL_OPEN request.
|
|
{
|
|
|
|
// First, we decode the message.
|
|
struct LMEChannel* channel;
|
|
APF_CHANNEL_OPEN_CONFIRMATION_MESSAGE *pMessage = (APF_CHANNEL_OPEN_CONFIRMATION_MESSAGE *)rxBuffer;
|
|
|
|
|
|
struct LMEChannelOpenReplySuccessMessage channelOpenReply;
|
|
channelOpenReply.RecipientChannel = ntohl(pMessage->RecipientChannel); // This is the identifier on our side.
|
|
channelOpenReply.SenderChannel = ntohl(pMessage->SenderChannel); // This is the identifier on the Intel AMT side.
|
|
channelOpenReply.InitialWindow = ntohl(pMessage->InitialWindowSize); // This is the starting window size for flow control.
|
|
channel = &(module->Sessions[channelOpenReply.RecipientChannel]); // Get the current session for this message.
|
|
|
|
if (channel == NULL)
|
|
break; // This should never happen.
|
|
|
|
LMSDEBUG("MEI OPEN OK OUR:%d AMT:%d\r\n", channelOpenReply.RecipientChannel, channelOpenReply.SenderChannel);
|
|
|
|
if (channel->status == LME_CS_PENDING_CONNECT) // If the channel is in PENDING_CONNECT mode, move the session to connected state.
|
|
{
|
|
channel->amtid = channelOpenReply.SenderChannel; // We have to set the Intel AMT identifier for this session.
|
|
channel->txwindow = channelOpenReply.InitialWindow; // Set the session txwindow.
|
|
channel->status = LME_CS_CONNECTED; // Now set the session as CONNECTED.
|
|
LMSDEBUG("Channel %d now CONNECTED by AMT %d\r\n", channel->ourid, channel->socketmodule);
|
|
if (channel->sockettype == 0) // If the current socket is PAUSED, lets resume it so data can start flowing again.
|
|
{
|
|
ILibAsyncSocket_Resume(channel->socketmodule); // TCP socket resume
|
|
}
|
|
}
|
|
else if (channel->status == LME_CS_PENDING_LMS_DISCONNECT) // If the channel is in PENDING_DISCONNECT, we have to disconnect the session now. Happens when we disconnect while connection is pending. We don't want to stop a channel during connection, that is bad.
|
|
{
|
|
channel->amtid = channelOpenReply.SenderChannel; // We have to set the Intel AMT identifier for this session.
|
|
LME_ChannelClose(&(module->MeConnection), channel->amtid, channel->ourid); // Send the Intel AMT close. We keep the channel in LME_CS_PENDING_LMS_DISCONNECT state until the close is confirmed.
|
|
LMSDEBUG("Channel %d now CONNECTED by AMT %d, but CLOSING it now\r\n", channel->ourid, channel->socketmodule);
|
|
}
|
|
else
|
|
{
|
|
// Here, we get an APF_CHANNEL_OPEN in an unexpected state, this should never happen.
|
|
LMSDEBUG("Channel %d, unexpected CONNECTED by AMT %d\r\n", channel->ourid, channel->socketmodule);
|
|
}
|
|
}
|
|
break;
|
|
case APF_CHANNEL_OPEN_FAILURE: // (92) Intel AMT rejected our connection attempt.
|
|
{
|
|
|
|
// First, we decode the message.
|
|
struct LMEChannel* channel;
|
|
APF_CHANNEL_OPEN_FAILURE_MESSAGE *pMessage = (APF_CHANNEL_OPEN_FAILURE_MESSAGE *)rxBuffer;
|
|
struct LMEChannelOpenReplyFailureMessage channelOpenReply;
|
|
channelOpenReply.RecipientChannel = ntohl(pMessage->RecipientChannel); // This is the identifier on our side.
|
|
|
|
channelOpenReply.ReasonCode = (OPEN_FAILURE_REASON)(ntohl(pMessage->ReasonCode)); // Get the error reason code.
|
|
channel = &(module->Sessions[channelOpenReply.RecipientChannel]); // Get the current session for this message.
|
|
if (channel == NULL) break; // This should never happen.
|
|
|
|
LMSDEBUG("**OPEN FAIL OUR:%d ERR:%d\r\n", channelOpenReply.RecipientChannel, channelOpenReply.ReasonCode);
|
|
|
|
if (channel->errorcount++ >= 0 || channel->status == LME_CS_PENDING_LMS_DISCONNECT)
|
|
{
|
|
// Fail connection
|
|
channel->status = LME_CS_FREE;
|
|
LMSDEBUG("Channel %d now FREE by AMT\r\n", channel->ourid);
|
|
sem_post(&(module->Lock));
|
|
ILibAsyncSocket_Disconnect(channel->socketmodule);
|
|
//ILibLMS_LaunchHoldingSessions(module);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
// Try again
|
|
//ILibLMS_SetupConnection(module, channel->ourid);
|
|
}
|
|
}
|
|
break;
|
|
case APF_CHANNEL_CLOSE: // (97) Intel AMT is closing this channel, we need to disconnect the LMS TCP connection
|
|
{
|
|
|
|
// First, we decode the message.
|
|
struct LMEChannel* channel;
|
|
APF_CHANNEL_CLOSE_MESSAGE *pMessage = (APF_CHANNEL_CLOSE_MESSAGE *)rxBuffer;
|
|
struct LMEChannelCloseMessage channelClose;
|
|
channelClose.RecipientChannel = ntohl(pMessage->RecipientChannel); // This is the identifier on our side.
|
|
channel = &(module->Sessions[channelClose.RecipientChannel]); // Get the current session for this message.
|
|
if (channel == NULL) break; // This should never happen.
|
|
|
|
//LMSDEBUG("CLOSE OUR:%d\r\n", channelClose.RecipientChannel);
|
|
|
|
if (channel->status == LME_CS_CONNECTED)
|
|
{
|
|
if (ILibAsyncSocket_IsConnected(channel->socketmodule))
|
|
{
|
|
channel->status = LME_CS_PENDING_AMT_DISCONNECT;
|
|
LMSDEBUG("Channel %d now PENDING_AMT_DISCONNECT by AMT, calling microstack disconnect %d\r\n", channel->ourid, channel->socketmodule);
|
|
LME_ChannelClose(lmemodule, channel->amtid, channel->ourid);
|
|
sem_post(&(module->Lock));
|
|
if (channel->sockettype == 0) // If the current socket is PAUSED, lets resume it so data can start flowing again.
|
|
{
|
|
ILibAsyncSocket_Disconnect(channel->socketmodule); // TCP socket close
|
|
}
|
|
|
|
sem_wait(&(module->Lock));
|
|
channel->status = LME_CS_FREE;
|
|
disconnects++;
|
|
LMSDEBUG("Channel %d now FREE by AMT\r\n", channel->ourid);
|
|
}
|
|
else
|
|
{
|
|
channel->status = LME_CS_FREE;
|
|
disconnects++;
|
|
LMSDEBUG("Channel %d now FREE by AMT\r\n", channel->ourid);
|
|
}
|
|
}
|
|
else if (channel->status == LME_CS_PENDING_LMS_DISCONNECT)
|
|
{
|
|
channel->status = LME_CS_FREE;
|
|
disconnects++;
|
|
LMSDEBUG("Channel %d now FREE by AMT\r\n", channel->ourid);
|
|
}
|
|
else
|
|
{
|
|
LMSDEBUG("Channel %d CLOSE, UNEXPECTED STATE %d\r\n", channel->ourid, channel->status);
|
|
}
|
|
}
|
|
break;
|
|
case APF_CHANNEL_DATA: // (94) Intel AMT is sending data that we must relay into an LMS TCP connection.
|
|
{
|
|
//sem_wait(&(module->Lock));
|
|
|
|
struct LMEChannel* channel;
|
|
APF_CHANNEL_DATA_MESSAGE *pMessage = (APF_CHANNEL_DATA_MESSAGE *)rxBuffer;
|
|
struct LMEChannelDataMessage channelData;
|
|
enum ILibAsyncSocket_SendStatus r;
|
|
channelData.MessageType = APF_CHANNEL_DATA;
|
|
channelData.RecipientChannel = ntohl(pMessage->RecipientChannel);
|
|
channelData.DataLength = ntohl(pMessage->DataLength);
|
|
channelData.Data = (unsigned char*)rxBuffer + sizeof(APF_CHANNEL_DATA_MESSAGE);
|
|
channel = &(module->Sessions[channelData.RecipientChannel]);
|
|
|
|
|
|
if (channel == NULL || channel->socketmodule == NULL){
|
|
sem_post(&(module->Lock));
|
|
break;
|
|
}
|
|
|
|
if (channel->sockettype == 0)
|
|
{
|
|
r = ILibAsyncSocket_Send(channel->socketmodule, (char*)(channelData.Data), channelData.DataLength, ILibAsyncSocket_MemoryOwnership_USER); // TCP socket
|
|
}
|
|
|
|
channel->rxwindow += channelData.DataLength;
|
|
if (r == ILibAsyncSocket_ALL_DATA_SENT && channel->rxwindow > 1024) {
|
|
LME_ChannelWindowAdjust(lmemodule, channel->amtid, channel->rxwindow); channel->rxwindow = 0;
|
|
}
|
|
//sem_post(&(module->Lock));
|
|
}
|
|
break;
|
|
case APF_CHANNEL_WINDOW_ADJUST: // 93
|
|
{
|
|
//sem_wait(&(module->Lock));
|
|
|
|
struct LMEChannel* channel;
|
|
APF_WINDOW_ADJUST_MESSAGE *pMessage = (APF_WINDOW_ADJUST_MESSAGE *)rxBuffer;
|
|
struct LMEChannelWindowAdjustMessage channelWindowAdjust;
|
|
channelWindowAdjust.MessageType = APF_CHANNEL_WINDOW_ADJUST;
|
|
channelWindowAdjust.RecipientChannel = ntohl(pMessage->RecipientChannel);
|
|
channelWindowAdjust.BytesToAdd = ntohl(pMessage->BytesToAdd);
|
|
channel = &(module->Sessions[channelWindowAdjust.RecipientChannel]);
|
|
if (channel == NULL || channel->status == LME_CS_FREE){
|
|
sem_post(&(module->Lock));
|
|
break;
|
|
}
|
|
channel->txwindow += channelWindowAdjust.BytesToAdd;
|
|
if (channel->sockettype == 0)
|
|
{
|
|
ILibAsyncSocket_Resume(channel->socketmodule); // TCP socket
|
|
}
|
|
//sem_post(&(module->Lock));
|
|
}
|
|
break;
|
|
case APF_PROTOCOLVERSION: // 192
|
|
{
|
|
APF_PROTOCOL_VERSION_MESSAGE *pMessage = (APF_PROTOCOL_VERSION_MESSAGE *)rxBuffer;
|
|
struct LMEProtocolVersionMessage protVersion;
|
|
protVersion.MajorVersion = ntohl(pMessage->MajorVersion);
|
|
protVersion.MinorVersion = ntohl(pMessage->MinorVersion);
|
|
protVersion.TriggerReason = (APF_TRIGGER_REASON)ntohl(pMessage->TriggerReason);
|
|
|
|
switch (module->handshakingStatus)
|
|
{
|
|
case LME_AGREED:
|
|
case LME_NOT_INITIATED:
|
|
{
|
|
LME_ProtocolVersion(lmemodule, 1, 0, protVersion.TriggerReason);
|
|
}
|
|
case LME_INITIATED:
|
|
if (protVersion.MajorVersion != 1 || protVersion.MinorVersion != 0)
|
|
{
|
|
LMSDEBUG("LME Version %d.%d is not supported.\r\n", protVersion.MajorVersion, protVersion.MinorVersion);
|
|
LME_Disconnect(lmemodule, APF_DISCONNECT_PROTOCOL_VERSION_NOT_SUPPORTED);
|
|
LME_Deinit(lmemodule);
|
|
sem_post(&(module->Lock));
|
|
return;
|
|
}
|
|
module->AmtProtVersionMajor = protVersion.MajorVersion;
|
|
module->AmtProtVersionMinor = protVersion.MinorVersion;
|
|
module->handshakingStatus = LME_AGREED;
|
|
break;
|
|
default:
|
|
LME_Disconnect(lmemodule, APF_DISCONNECT_BY_APPLICATION);
|
|
LME_Deinit(lmemodule);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case APF_USERAUTH_REQUEST: // 50
|
|
{
|
|
// _lme.UserAuthSuccess();
|
|
}
|
|
break;
|
|
default:
|
|
// Unknown request.
|
|
LMSDEBUG("**Unknown LME command: %d\r\n", ((unsigned char*)rxBuffer)[0]);
|
|
LME_Disconnect(lmemodule, APF_DISCONNECT_PROTOCOL_ERROR);
|
|
LME_Deinit(lmemodule);
|
|
break;
|
|
}
|
|
|
|
sem_post(&(module->Lock));
|
|
//if (disconnects > 0) ILibLMS_LaunchHoldingSessions(module); // If disconnects is set to anything, we have free session slots we can fill up.
|
|
}
|
|
|
|
void ILibLMS_OnReceive(ILibAsyncServerSocket_ServerModule AsyncServerSocketModule, ILibAsyncServerSocket_ConnectionToken ConnectionToken, char* buffer, int *p_beginPointer, int endPointer, ILibAsyncServerSocket_OnInterrupt *OnInterrupt, void **user, int *PAUSE)
|
|
{
|
|
int r, maxread = endPointer;
|
|
struct ILibLMS_StateModule* module = (struct ILibLMS_StateModule*)ILibAsyncServerSocket_GetTag(AsyncServerSocketModule);
|
|
struct LMEChannel* channel = (struct LMEChannel*)*user;
|
|
|
|
UNREFERENCED_PARAMETER( AsyncServerSocketModule );
|
|
UNREFERENCED_PARAMETER( ConnectionToken );
|
|
UNREFERENCED_PARAMETER( OnInterrupt );
|
|
UNREFERENCED_PARAMETER( PAUSE );
|
|
|
|
if (channel == NULL) return;
|
|
sem_wait(&(module->Lock));
|
|
if (channel->socketmodule != ConnectionToken) { sem_post(&(module->Lock)); ILibAsyncSocket_Disconnect(ConnectionToken); return; }
|
|
if (channel->txwindow < endPointer) maxread = channel->txwindow;
|
|
if (channel->status != LME_CS_CONNECTED || maxread == 0) { *PAUSE = 1; sem_post(&(module->Lock)); return; }
|
|
//printf("%.*s", maxread, buffer);
|
|
r = LME_ChannelData(&(module->MeConnection), channel->amtid, maxread, (unsigned char*)buffer);
|
|
//LMSDEBUG("ILibLMS_OnReceive, status = %d, txwindow = %d, endPointer = %d, r = %d\r\n", channel->status, channel->txwindow, endPointer, r);
|
|
if (r != maxread)
|
|
{
|
|
//LMSDEBUG("ILibLMS_OnReceive, DISCONNECT %d\r\n", channel->ourid);
|
|
sem_post(&(module->Lock));
|
|
ILibAsyncSocket_Disconnect(ConnectionToken); // Drop the connection
|
|
return;
|
|
}
|
|
channel->txwindow -= maxread;
|
|
*p_beginPointer = maxread;
|
|
sem_post(&(module->Lock));
|
|
}
|
|
|
|
void ILibLMS_SetupConnection(struct ILibLMS_StateModule* module, int i)
|
|
{
|
|
int rport = 0;
|
|
char* laddr = NULL;
|
|
struct sockaddr_in6 remoteAddress;
|
|
|
|
if (module->Sessions[i].sockettype == 0)
|
|
{
|
|
// Fetch the socket remote TCP address
|
|
ILibAsyncSocket_GetRemoteInterface(module->Sessions[i].socketmodule, (struct sockaddr*)&remoteAddress);
|
|
if (remoteAddress.sin6_family == AF_INET6)
|
|
{
|
|
laddr = "::1"; // TODO: decode this properly into a string
|
|
rport = ntohs(remoteAddress.sin6_port);
|
|
}
|
|
else
|
|
{
|
|
laddr = "127.0.0.1"; // TODO: decode this properly into a string
|
|
rport = ntohs(((struct sockaddr_in*)(&remoteAddress))->sin_port);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Fetch the socket remote web socket address
|
|
laddr = "127.0.0.1"; // TODO: decode this properly into a string
|
|
rport = 123; // TODO: decode the remote port
|
|
}
|
|
|
|
// Setup a new LME session
|
|
LME_ChannelOpenForwardedRequest(&(module->MeConnection), (unsigned int)i, laddr, module->Sessions[i].localport, laddr, rport);
|
|
//LMSDEBUG("ILibLMS_OnReceive, CONNECT %d\r\n", i);
|
|
}
|
|
|
|
|
|
int OnConnect = 0;
|
|
void ILibLMS_OnConnect(ILibAsyncServerSocket_ServerModule AsyncServerSocketModule, ILibAsyncServerSocket_ConnectionToken ConnectionToken, void **user)
|
|
{
|
|
OnConnect++;
|
|
LMSDEBUG("OnCOnnect:%d", OnConnect);
|
|
int i, sessionid = -1, activecount = 0;
|
|
struct ILibLMS_StateModule* module = (struct ILibLMS_StateModule*)ILibAsyncServerSocket_GetTag(AsyncServerSocketModule);
|
|
|
|
sem_wait(&(module->Lock));
|
|
|
|
// Count the number of active sessions
|
|
activecount = LMS_MAX_SESSIONS;
|
|
for (i = 0; i < LMS_MAX_SESSIONS; i++) {
|
|
if (module->Sessions[i].status == LME_CS_FREE || module->Sessions[i].status == LME_CS_CONNECTION_WAIT) {
|
|
activecount--;
|
|
}
|
|
}
|
|
|
|
sessionid=GetFreeSlot(module);
|
|
|
|
// Look for an empty session
|
|
//for (i = 1; i < LMS_MAX_SESSIONS; i++) {
|
|
// if (module->Sessions[i].status == LME_CS_FREE) {
|
|
// sessionid = i;
|
|
// break;
|
|
// }
|
|
//}
|
|
if (sessionid == -1)
|
|
{
|
|
sem_post(&(module->Lock));
|
|
LMSDEBUG("ILibLMS_OnConnect NO SESSION SLOTS AVAILABLE\r\n");
|
|
ILibAsyncSocket_Disconnect(ConnectionToken); // Drop the connection
|
|
return;
|
|
}
|
|
i = sessionid;
|
|
|
|
// LMSDEBUG("USING SLOT OUR:%d\r\n", i);
|
|
module->Sessions[i].amtid = -1;
|
|
module->Sessions[i].ourid = i;
|
|
module->Sessions[i].sockettype = 0; // TCP Type
|
|
module->Sessions[i].socketmodule = ConnectionToken;
|
|
module->Sessions[i].rxwindow = 0;
|
|
module->Sessions[i].txwindow = 0;
|
|
module->Sessions[i].localport = ILibAsyncServerSocket_GetPortNumber(AsyncServerSocketModule);
|
|
module->Sessions[i].errorcount = 0;
|
|
*user = &(module->Sessions[i]);
|
|
|
|
/*
|
|
if (activecount >= LMS_MAX_CONNECTIONS)
|
|
{
|
|
module->Sessions[i].status = LME_CS_CONNECTION_WAIT;
|
|
LMSDEBUG("ILibLMS_OnConnect %d (HOLDING)\r\n", i);
|
|
}
|
|
else
|
|
*/
|
|
{
|
|
module->Sessions[i].status = LME_CS_PENDING_CONNECT;
|
|
LMSDEBUG("ILibLMS_OnConnect %d\r\n", i);
|
|
ILibLMS_SetupConnection(module, i);
|
|
}
|
|
|
|
sem_post(&(module->Lock));
|
|
}
|
|
|
|
void ILibLMS_OnDisconnect(ILibAsyncServerSocket_ServerModule AsyncServerSocketModule, ILibAsyncServerSocket_ConnectionToken ConnectionToken, void *user)
|
|
{
|
|
int disconnects = 0;
|
|
struct ILibLMS_StateModule* module = (struct ILibLMS_StateModule*)ILibAsyncServerSocket_GetTag(AsyncServerSocketModule);
|
|
struct LMEChannel* channel = (struct LMEChannel*)user;
|
|
UNREFERENCED_PARAMETER( ConnectionToken );
|
|
|
|
sem_wait(&(module->Lock));
|
|
|
|
if (channel == NULL || channel->socketmodule != ConnectionToken)
|
|
{
|
|
LMSDEBUG("****ILibLMS_OnDisconnect EXIT\r\n");
|
|
sem_post(&(module->Lock));
|
|
return;
|
|
}
|
|
|
|
LMSDEBUG("ILibLMS_OnDisconnect, Channel %d, %d\r\n", channel->ourid, ConnectionToken);
|
|
|
|
if (channel->status == LME_CS_CONNECTED)
|
|
{
|
|
channel->status = LME_CS_PENDING_LMS_DISCONNECT;
|
|
if (channel->amtid!=-1 && !LME_ChannelClose(&(module->MeConnection), channel->amtid, channel->ourid))
|
|
{
|
|
channel->status = LME_CS_FREE;
|
|
disconnects++;
|
|
LMSDEBUG("Channel %d now FREE by LMS because of failed close\r\n", channel->ourid);
|
|
}
|
|
else
|
|
{
|
|
LMSDEBUG("Channel %d now PENDING_LMS_DISCONNECT by LMS\r\n", channel->ourid);
|
|
//LMSDEBUG("LME_ChannelClose OK\r\n");
|
|
}
|
|
}
|
|
else if (channel->status == LME_CS_PENDING_CONNECT)
|
|
{
|
|
channel->status = LME_CS_PENDING_LMS_DISCONNECT;
|
|
LMSDEBUG("Channel %d now PENDING_DISCONNECT by LMS\r\n", channel->ourid);
|
|
}
|
|
else if (channel->status == LME_CS_CONNECTION_WAIT)
|
|
{
|
|
channel->status = LME_CS_FREE;
|
|
disconnects++;
|
|
LMSDEBUG("Channel %d now FREE by LMS\r\n", channel->ourid);
|
|
}
|
|
//LMSDEBUG("ILibLMS_OnDisconnect, DISCONNECT %d, status = %d\r\n", channel->ourid, channel->status);
|
|
sem_post(&(module->Lock));
|
|
|
|
//if (disconnects > 0) ILibLMS_LaunchHoldingSessions(module);
|
|
}
|
|
|
|
void ILibLMS_OnSendOK(ILibAsyncServerSocket_ServerModule AsyncServerSocketModule, ILibAsyncServerSocket_ConnectionToken ConnectionToken, void *user)
|
|
{
|
|
struct ILibLMS_StateModule* module = (struct ILibLMS_StateModule*)ILibAsyncServerSocket_GetTag(AsyncServerSocketModule);
|
|
struct LMEChannel* channel = (struct LMEChannel*)user;
|
|
UNREFERENCED_PARAMETER( ConnectionToken );
|
|
|
|
// Ok to send more on this socket, adjust the window
|
|
sem_wait(&(module->Lock));
|
|
if (channel->rxwindow != 0)
|
|
{
|
|
//LMSDEBUG("LME_ChannelWindowAdjust id=%d, rxwindow=%d\r\n", channel->amtid, channel->rxwindow);
|
|
LME_ChannelWindowAdjust(&(module->MeConnection), channel->amtid, channel->rxwindow);
|
|
channel->rxwindow = 0;
|
|
}
|
|
sem_post(&(module->Lock));
|
|
}
|
|
// Private method called when the chain is destroyed, we want to do our cleanup here
|
|
void ILibLMS_Destroy(void *object)
|
|
{
|
|
struct ILibLMS_StateModule* module = (struct ILibLMS_StateModule*)object;
|
|
UNREFERENCED_PARAMETER( object );
|
|
|
|
sem_wait(&(module->Lock));
|
|
LME_Disconnect(&(module->MeConnection), APF_DISCONNECT_BY_APPLICATION);
|
|
LME_Exit(&(module->MeConnection));
|
|
sem_destroy(&(module->Lock));
|
|
if (IlibExternLMS == module) { IlibExternLMS = NULL; } // Clear the global reference to the the LMS module.
|
|
}
|
|
|
|
// Create a new Reflector module.
|
|
struct ILibLMS_StateModule *ILibLMS_Create(void *Chain)
|
|
{
|
|
struct ILibLMS_StateModule *module;
|
|
|
|
// Allocate the new module
|
|
module = (struct ILibLMS_StateModule*)malloc(sizeof(struct ILibLMS_StateModule));
|
|
if (module == NULL) { ILIBCRITICALEXIT(254); }
|
|
memset(module, 0, sizeof(struct ILibLMS_StateModule));
|
|
|
|
// Setup MEI with LMS interface, if we can't return null
|
|
if (LME_Init(&(module->MeConnection), &ILibLMS_MEICallback, module) == 0) {
|
|
free(module);
|
|
return NULL;
|
|
}
|
|
|
|
// Setup the module
|
|
module->ChainLink.DestroyHandler = &ILibLMS_Destroy;
|
|
module->ChainLink.ParentChain = Chain;
|
|
sem_init(&(module->Lock), 0, 1);
|
|
|
|
// TCP servers
|
|
module->Server1 = ILibCreateAsyncServerSocketModule(Chain, LMS_MAX_CONNECTIONS, 16992, 4096, 0, &ILibLMS_OnConnect, &ILibLMS_OnDisconnect, &ILibLMS_OnReceive, NULL, &ILibLMS_OnSendOK);
|
|
if (module->Server1 == NULL) { sem_destroy(&(module->Lock)); free(module); return NULL; }
|
|
ILibAsyncServerSocket_SetTag(module->Server1, module);
|
|
|
|
module->Server2 = ILibCreateAsyncServerSocketModule(Chain, LMS_MAX_CONNECTIONS, 16993, 4096, 0, &ILibLMS_OnConnect, &ILibLMS_OnDisconnect, &ILibLMS_OnReceive, NULL, &ILibLMS_OnSendOK);
|
|
if (module->Server2 == NULL) { sem_destroy(&(module->Lock)); free(module->Server1); free(module); return NULL; }
|
|
ILibAsyncServerSocket_SetTag(module->Server2, module);
|
|
|
|
|
|
|
|
|
|
// TODO: US6986 - The meshcommander site should not be served on localhost:16994.
|
|
// This code should be commented out - in the future(late 2018) IMC could be put there instead.
|
|
// Web Server
|
|
//module->WebServer = ILibWebServer_CreateEx(Chain, 8, 16994, 2, &ILibLMS_WebServer_OnSession, module);
|
|
|
|
IlibExternLMS = module; // Set the global reference to the LMS module.
|
|
ILibAddToChain(Chain, module); // Add this module to the chain.
|
|
return module;
|
|
}
|
|
|
|
#endif
|