Submitted new SerialComms module

This commit is contained in:
tomch 2006-05-02 21:42:45 +00:00
parent 04c439ed87
commit 570f8eb9b9
2 changed files with 635 additions and 188 deletions

View File

@ -21,14 +21,28 @@ along with AppleWin; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/ */
/* Description: Serial port emulation: not SSC, but similar (same?) regs /* Description: Super Serial Card emulation
* *
* Author: Various * Author: Various
*/ */
// TO DO:
// . Enable & test Tx IRQ
// . DIP switch read values
//
// Refs:
// (1) "Super Serial Card (SSC) Memory Locations for Programmers" - Aaron Heiss
// (2) SSC recv IRQ example: http://www.wright.edu/~john.matthews/ssc.html#lst - John B. Matthews, 5/13/87
// (3) WaitCommEvent, etc: http://mail.python.org/pipermail/python-list/2002-November/131437.html
// (4) SY6551 info: http://www.axess.com/twilight/sock/rs232pak.html
//
#include "StdAfx.h" #include "StdAfx.h"
#pragma hdrstop #pragma hdrstop
//#define SUPPORT_MODEM
static DWORD baudrate = CBR_19200; static DWORD baudrate = CBR_19200;
static BYTE bytesize = 8; static BYTE bytesize = 8;
static BYTE commandbyte = 0x00; static BYTE commandbyte = 0x00;
@ -36,238 +50,669 @@ static HANDLE commhandle = INVALID_HANDLE_VALUE;
static DWORD comminactivity = 0; static DWORD comminactivity = 0;
static BYTE controlbyte = 0x1F; static BYTE controlbyte = 0x1F;
static BYTE parity = NOPARITY; static BYTE parity = NOPARITY;
static BYTE recvbuffer[9];
static DWORD recvbytes = 0;
DWORD serialport = 0; DWORD serialport = 0;
static BYTE stopbits = ONESTOPBIT; static BYTE stopbits = ONESTOPBIT;
static void UpdateCommState (); //
static CRITICAL_SECTION g_CriticalSection; // To guard /g_vRecvBytes/
static BYTE g_RecvBuffer[uRecvBufferSize]; // NB: More work required if >1 is used
static volatile DWORD g_vRecvBytes = 0;
//
static bool g_bTxIrqEnabled = false;
static bool g_bRxIrqEnabled = false;
static bool g_bWrittenTx = false;
//
static volatile bool g_vbCommIRQ = false;
static HANDLE g_hCommThread = NULL;
enum {COMMEVT_WAIT=0, COMMEVT_ACK, COMMEVT_TERM, COMMEVT_MAX};
static HANDLE g_hCommEvent[COMMEVT_MAX] = {NULL};
static OVERLAPPED o;
//=========================================================================== //===========================================================================
static BOOL CheckComm () {
comminactivity = 0; static void UpdateCommState ()
if ((commhandle == INVALID_HANDLE_VALUE) && serialport) { {
TCHAR portname[8]; if (commhandle == INVALID_HANDLE_VALUE)
wsprintf(portname, return;
TEXT("COM%u"),
serialport); DCB dcb;
commhandle = CreateFile(portname, ZeroMemory(&dcb,sizeof(DCB));
GENERIC_READ | GENERIC_WRITE, dcb.DCBlength = sizeof(DCB);
0, GetCommState(commhandle,&dcb);
(LPSECURITY_ATTRIBUTES)NULL, dcb.BaudRate = baudrate;
OPEN_EXISTING, dcb.ByteSize = bytesize;
0, dcb.Parity = parity;
NULL); dcb.StopBits = stopbits;
if (commhandle != INVALID_HANDLE_VALUE) {
UpdateCommState(); SetCommState(commhandle,&dcb);
COMMTIMEOUTS ct;
ZeroMemory(&ct,sizeof(COMMTIMEOUTS));
ct.ReadIntervalTimeout = MAXDWORD;
SetCommTimeouts(commhandle,&ct);
}
}
return (commhandle != INVALID_HANDLE_VALUE);
} }
//=========================================================================== //===========================================================================
static void CheckReceive () {
if (recvbytes || (commhandle == INVALID_HANDLE_VALUE)) static BOOL CheckComm ()
return; {
ReadFile(commhandle,recvbuffer,8,&recvbytes,NULL); comminactivity = 0;
if ((commhandle == INVALID_HANDLE_VALUE) && serialport)
{
TCHAR portname[8];
wsprintf(portname, TEXT("COM%u"), serialport);
commhandle = CreateFile(portname,
GENERIC_READ | GENERIC_WRITE,
0, // exclusive access
(LPSECURITY_ATTRIBUTES)NULL, // default security attributes
OPEN_EXISTING,
FILE_FLAG_OVERLAPPED, // required for WaitCommEvent()
NULL);
if (commhandle != INVALID_HANDLE_VALUE)
{
UpdateCommState();
COMMTIMEOUTS ct;
ZeroMemory(&ct,sizeof(COMMTIMEOUTS));
ct.ReadIntervalTimeout = MAXDWORD;
SetCommTimeouts(commhandle,&ct);
CommThInit();
}
else
{
DWORD uError = GetLastError();
}
}
return (commhandle != INVALID_HANDLE_VALUE);
} }
//=========================================================================== //===========================================================================
static void CloseComm () {
if (commhandle != INVALID_HANDLE_VALUE) static void CloseComm ()
CloseHandle(commhandle); {
commhandle = INVALID_HANDLE_VALUE; CommThUninit(); // Kill CommThread before closing COM handle
comminactivity = 0;
if (commhandle != INVALID_HANDLE_VALUE)
CloseHandle(commhandle);
commhandle = INVALID_HANDLE_VALUE;
comminactivity = 0;
} }
//=========================================================================== //===========================================================================
static void UpdateCommState () {
if (commhandle == INVALID_HANDLE_VALUE) // EG. 0x09 = Enable IRQ, No parity [Ref.2]
return;
DCB dcb; BYTE __stdcall CommCommand (WORD, BYTE, BYTE write, BYTE value, ULONG)
ZeroMemory(&dcb,sizeof(DCB)); {
dcb.DCBlength = sizeof(DCB); if (!CheckComm())
GetCommState(commhandle,&dcb); return 0;
dcb.BaudRate = baudrate;
dcb.ByteSize = bytesize; if (write && (value != commandbyte))
dcb.Parity = parity; {
dcb.StopBits = stopbits; commandbyte = value;
SetCommState(commhandle,&dcb);
// UPDATE THE PARITY
if (commandbyte & 0x20)
{
switch (commandbyte & 0xC0)
{
case 0x00 : parity = ODDPARITY; break;
case 0x40 : parity = EVENPARITY; break;
case 0x80 : parity = MARKPARITY; break;
case 0xC0 : parity = SPACEPARITY; break;
}
}
else
{
parity = NOPARITY;
}
if (commandbyte & 0x10) // Receiver mode echo [0=no echo, 1=echo]
{
}
switch (commandbyte & 0x0C) // transmitter interrupt control
{
// Note: the RTS signal must be set 'low' in order to receive any
// incoming data from the serial device
case 0<<2: // set RTS high and transmit no interrupts
g_bTxIrqEnabled = false;
break;
case 1<<2: // set RTS low and transmit interrupts
g_bTxIrqEnabled = true;
break;
case 2<<2: // set RTS low and transmit no interrupts
g_bTxIrqEnabled = false;
break;
case 3<<2: // set RTS low and transmit break signals instead of interrupts
g_bTxIrqEnabled = false;
break;
}
// interrupt request disable [0=enable receiver interrupts]
g_bRxIrqEnabled = ((commandbyte & 0x02) == 0);
if (commandbyte & 0x01) // Data Terminal Ready (DTR) setting [0=set DTR high (indicates 'not ready')]
{
// Note that, although the DTR is generally not used in the SSC (it may actually not
// be connected!), it must be set to 'low' in order for the 6551 to function correctly.
}
UpdateCommState();
}
return commandbyte;
} }
//=========================================================================== //===========================================================================
BYTE __stdcall CommCommand (WORD, BYTE, BYTE write, BYTE value, ULONG) {
if (!CheckComm())
return 0;
if (write && (value != commandbyte)) {
commandbyte = value;
// UPDATE THE PARITY BYTE __stdcall CommControl (WORD, BYTE, BYTE write, BYTE value, ULONG)
if (commandbyte & 0x20) {
switch (commandbyte & 0xC0) { if (!CheckComm())
case 0x00 : parity = ODDPARITY; break; return 0;
case 0x40 : parity = EVENPARITY; break;
case 0x80 : parity = MARKPARITY; break;
case 0xC0 : parity = SPACEPARITY; break;
}
else
parity = NOPARITY;
UpdateCommState(); if (write && (value != controlbyte))
} {
return commandbyte; controlbyte = value;
}
//=========================================================================== // UPDATE THE BAUD RATE
BYTE __stdcall CommControl (WORD, BYTE, BYTE write, BYTE value, ULONG) { switch (controlbyte & 0x0F)
if (!CheckComm()) {
return 0; // Note that 1 MHz Apples (everything other than the Apple IIgs and //c
if (write && (value != controlbyte)) { // Plus running in "fast" mode) cannot handle 19.2 kbps, and even 9600
controlbyte = value; // bps on these machines requires either some highly optimised code or
// a decent buffer in the device being accessed. The faster Apples
// have no difficulty with this speed, however.
// UPDATE THE BAUD RATE case 0x00: // fall through [16x external clock]
switch (controlbyte & 0x0F) { case 0x01: // fall through [50 bps]
case 0x00: // fall through case 0x02: // fall through [75 bps]
case 0x01: // fall through case 0x03: // fall through [109.92 bps]
case 0x02: // fall through case 0x04: // fall through [134.58 bps]
case 0x03: // fall through case 0x05: baudrate = CBR_110; break; // [150 bps]
case 0x04: // fall through case 0x06: baudrate = CBR_300; break;
case 0x05: baudrate = CBR_110; break; case 0x07: baudrate = CBR_600; break;
case 0x06: baudrate = CBR_300; break; case 0x08: baudrate = CBR_1200; break;
case 0x07: baudrate = CBR_600; break; case 0x09: // fall through [1800 bps]
case 0x08: baudrate = CBR_1200; break; case 0x0A: baudrate = CBR_2400; break;
case 0x09: // fall through case 0x0B: // fall through [3600 bps]
case 0x0A: baudrate = CBR_2400; break; case 0x0C: baudrate = CBR_4800; break;
case 0x0B: // fall through case 0x0D: // fall through [7200 bps]
case 0x0C: baudrate = CBR_4800; break; case 0x0E: baudrate = CBR_9600; break;
case 0x0D: // fall through case 0x0F: baudrate = CBR_19200; break;
case 0x0E: baudrate = CBR_9600; break; }
case 0x0F: baudrate = CBR_19200; break;
}
// UPDATE THE BYTE SIZE if (controlbyte & 0x10)
switch (controlbyte & 0x60) { {
case 0x00: bytesize = 8; break; // receiver clock source [0= external, 1= internal]
case 0x20: bytesize = 7; break; }
case 0x40: bytesize = 6; break;
case 0x60: bytesize = 5; break;
}
// UPDATE THE NUMBER OF STOP BITS // UPDATE THE BYTE SIZE
if (controlbyte & 0x80) { switch (controlbyte & 0x60)
if ((bytesize == 8) && (parity == NOPARITY)) {
stopbits = ONESTOPBIT; case 0x00: bytesize = 8; break;
else if ((bytesize == 5) && (parity == NOPARITY)) case 0x20: bytesize = 7; break;
stopbits = ONE5STOPBITS; case 0x40: bytesize = 6; break;
else case 0x60: bytesize = 5; break;
stopbits = TWOSTOPBITS; }
}
else // UPDATE THE NUMBER OF STOP BITS
stopbits = ONESTOPBIT; if (controlbyte & 0x80)
{
if ((bytesize == 8) && (parity == NOPARITY))
stopbits = ONESTOPBIT;
else if ((bytesize == 5) && (parity == NOPARITY))
stopbits = ONE5STOPBITS;
else
stopbits = TWOSTOPBITS;
}
else
{
stopbits = ONESTOPBIT;
}
UpdateCommState();
}
UpdateCommState();
}
return controlbyte; return controlbyte;
} }
//=========================================================================== //===========================================================================
void CommDestroy () {
if ((baudrate != CBR_19200) || BYTE __stdcall CommReceive (WORD, BYTE, BYTE, BYTE, ULONG)
(bytesize != 8) || {
(parity != NOPARITY) || if (!CheckComm())
(stopbits != ONESTOPBIT)) { return 0;
CommReset();
CheckComm(); BYTE result = 0;
} if (g_vRecvBytes)
CloseComm(); {
// Don't need critical section in here as CommThread is waiting for ACK
result = g_RecvBuffer[0];
--g_vRecvBytes;
if (g_vbCommIRQ && !g_vRecvBytes)
{
// Read last byte, so get CommThread to call WaitCommEvent() again
OutputDebugString("CommRecv: SetEvent - ACK\n");
SetEvent(g_hCommEvent[COMMEVT_ACK]);
}
}
return result;
} }
//=========================================================================== //===========================================================================
BYTE __stdcall CommDipSw (WORD, BYTE, BYTE, BYTE, ULONG) {
// note: determine what values a real SSC returns BYTE __stdcall CommTransmit (WORD, BYTE, BYTE, BYTE value, ULONG)
return 0; {
if (!CheckComm())
return 0;
DWORD uBytesWritten;
WriteFile(commhandle, &value, 1, &uBytesWritten, &o);
g_bWrittenTx = true; // Transmit done
// TO DO:
// 1) Use CommThread determine when transmit is complete
// 2) OR do this:
//if (g_bTxIrqEnabled)
// CpuIrqAssert(IS_SSC);
return 0;
} }
//=========================================================================== //===========================================================================
void CommSetSerialPort (HWND window, DWORD newserialport) {
if (commhandle == INVALID_HANDLE_VALUE) // 6551 ACIA Status Register ($C089+s0)
serialport = newserialport; // ------------------------------------
else // Bit Value Meaning
MessageBox(window, // 0 1 Parity error
TEXT("You cannot change the serial port while it is ") // 1 1 Framing error
TEXT("in use."), // 2 1 Overrun error
TEXT("Configuration"), // 3 1 Receive register full
MB_ICONEXCLAMATION | MB_SETFOREGROUND); // 4 1 Transmit register empty
// 5 0 Data Carrier Detect (DCD) true [0=DCD low (detected), 1=DCD high (not detected)]
// 6 0 Data Set Ready (DSR) true [0=DSR low (ready), 1=DSR high (not ready)]
// 7 1 Interrupt (IRQ) true (cleared by reading status reg [Ref.4])
enum { ST_PARITY_ERR = 1<<0,
ST_FRAMING_ERR = 1<<1,
ST_OVERRUN_ERR = 1<<2,
ST_RX_FULL = 1<<3,
ST_TX_EMPTY = 1<<4,
ST_DCD = 1<<5,
ST_DSR = 1<<6,
ST_IRQ = 1<<7
};
BYTE __stdcall CommStatus (WORD, BYTE, BYTE, BYTE, ULONG)
{
if (!CheckComm())
return ST_DSR | ST_DCD | ST_TX_EMPTY;
#ifdef SUPPORT_MODEM
DWORD modemstatus = 0;
GetCommModemStatus(commhandle,&modemstatus); // Returns 0x30 = MS_DSR_ON|MS_CTS_ON
#endif
//
// TO DO - ST_TX_EMPTY:
// . IRQs enabled : set after WaitCommEvent has signaled that TX has completed
// . IRQs disabled : always set it [Currently done]
//
// So that /g_vRecvBytes/ doesn't change midway (from 0 to 1):
// . bIRQ=false, but uStatus.ST_RX_FULL=1
EnterCriticalSection(&g_CriticalSection);
bool bIRQ = false;
if (g_bTxIrqEnabled && g_bWrittenTx)
{
bIRQ = true;
}
if (g_bRxIrqEnabled && g_vRecvBytes)
{
bIRQ = true;
}
g_bWrittenTx = false; // Read status reg always clears IRQ
//
BYTE uStatus = ST_TX_EMPTY
| (g_vRecvBytes ? ST_RX_FULL : 0x00)
#ifdef SUPPORT_MODEM
| ((modemstatus & MS_RLSD_ON) ? 0x00 : ST_DCD) // Need 0x00 to allow ZLink to start up
| ((modemstatus & MS_DSR_ON) ? 0x00 : ST_DSR)
#endif
| (bIRQ ? ST_IRQ : 0x00);
LeaveCriticalSection(&g_CriticalSection);
CpuIrqDeassert(IS_SSC);
return uStatus;
} }
//=========================================================================== //===========================================================================
void CommUpdate (DWORD totalcycles) {
if (commhandle == INVALID_HANDLE_VALUE) BYTE __stdcall CommDipSw (WORD, BYTE addr, BYTE, BYTE, ULONG)
return; {
if ((comminactivity += totalcycles) > 1000000) { // TO DO: determine what values a real SSC returns
static DWORD lastcheck = 0; BYTE sw = 0;
if ((comminactivity > 2000000) || (comminactivity-lastcheck > 99950)) { return sw;
DWORD modemstatus = 0;
GetCommModemStatus(commhandle,&modemstatus);
if ((modemstatus & MS_RLSD_ON) || DiskIsSpinning())
comminactivity = 0;
}
if (comminactivity > 2000000)
CloseComm();
}
} }
//=========================================================================== //===========================================================================
BYTE __stdcall CommReceive (WORD, BYTE, BYTE, BYTE, ULONG) {
if (!CheckComm()) void CommReset ()
return 0; {
if (!recvbytes) CloseComm();
CheckReceive();
BYTE result = 0; baudrate = CBR_19200;
if (recvbytes) { bytesize = 8;
result = recvbuffer[0]; commandbyte = 0x00;
if (--recvbytes) controlbyte = 0x1F;
MoveMemory(recvbuffer,recvbuffer+1,recvbytes); parity = NOPARITY;
} g_vRecvBytes = 0;
return result; stopbits = ONESTOPBIT;
} }
//=========================================================================== //===========================================================================
void CommReset () {
CloseComm(); void CommDestroy ()
baudrate = CBR_19200; {
bytesize = 8; if ((baudrate != CBR_19200) ||
commandbyte = 0x00; (bytesize != 8) ||
controlbyte = 0x1F; (parity != NOPARITY) ||
parity = NOPARITY; (stopbits != ONESTOPBIT))
recvbytes = 0; {
stopbits = ONESTOPBIT; CommReset();
}
CloseComm();
} }
//=========================================================================== //===========================================================================
BYTE __stdcall CommStatus (WORD, BYTE, BYTE, BYTE, ULONG) {
if (!CheckComm()) void CommSetSerialPort (HWND window, DWORD newserialport)
return 0x70; {
if (!recvbytes) if (commhandle == INVALID_HANDLE_VALUE)
CheckReceive(); {
DWORD modemstatus = 0; serialport = newserialport;
GetCommModemStatus(commhandle,&modemstatus); }
return 0x10 | (recvbytes ? 0x08 : 0x00) else
| ((modemstatus & MS_RLSD_ON) ? 0x00 : 0x20) {
| ((modemstatus & MS_DSR_ON) ? 0x00 : 0x40); MessageBox(window,
TEXT("You cannot change the serial port while it is ")
TEXT("in use."),
TEXT("Configuration"),
MB_ICONEXCLAMATION | MB_SETFOREGROUND);
}
} }
//=========================================================================== //===========================================================================
BYTE __stdcall CommTransmit (WORD, BYTE, BYTE, BYTE value, ULONG) {
if (!CheckComm()) void CommUpdate (DWORD totalcycles)
return 0; {
DWORD byteswritten; if (commhandle == INVALID_HANDLE_VALUE)
WriteFile(commhandle,&value,1,&byteswritten,NULL); return;
return 0;
if ((comminactivity += totalcycles) > 1000000)
{
static DWORD lastcheck = 0;
if ((comminactivity > 2000000) || (comminactivity-lastcheck > 99950))
{
#ifdef SUPPORT_MODEM
DWORD modemstatus = 0;
GetCommModemStatus(commhandle,&modemstatus);
if ((modemstatus & MS_RLSD_ON) || DiskIsSpinning())
comminactivity = 0;
#else
if (DiskIsSpinning())
comminactivity = 0;
#endif
}
//if (comminactivity > 2000000)
// CloseComm();
}
}
//===========================================================================
static void CheckCommEvent(DWORD dwEvtMask)
{
if (dwEvtMask & EV_RXCHAR)
{
EnterCriticalSection(&g_CriticalSection);
ReadFile(commhandle, g_RecvBuffer, 1, (DWORD*)&g_vRecvBytes, &o);
LeaveCriticalSection(&g_CriticalSection);
if (g_bRxIrqEnabled && g_vRecvBytes)
{
g_vbCommIRQ = true;
CpuIrqAssert(IS_SSC);
}
}
//else if (dwEvtMask & EV_TXEMPTY)
//{
// if (g_bTxIrqEnabled)
// {
// g_vbCommIRQ = true;
// CpuIrqAssert(IS_SSC);
// }
//}
}
static DWORD WINAPI CommThread(LPVOID lpParameter)
{
char szDbg[100];
// BOOL bRes = SetCommMask(commhandle, EV_TXEMPTY | EV_RXCHAR);
BOOL bRes = SetCommMask(commhandle, EV_RXCHAR); // Just RX
if (!bRes)
return -1;
//
const UINT nNumEvents = 2;
#if 1
HANDLE hCommEvent_Wait[nNumEvents] = {g_hCommEvent[COMMEVT_WAIT], g_hCommEvent[COMMEVT_TERM]};
HANDLE hCommEvent_Ack[nNumEvents] = {g_hCommEvent[COMMEVT_ACK], g_hCommEvent[COMMEVT_TERM]};
#else
HANDLE hCommEvent_Wait[nNumEvents];
HANDLE hCommEvent_Ack[nNumEvents];
hCommEvent_Wait[0] = g_hCommEvent[COMMEVT_WAIT];
hCommEvent_Wait[1] = g_hCommEvent[COMMEVT_TERM];
hCommEvent_Ack[0] = g_hCommEvent[COMMEVT_ACK];
hCommEvent_Ack[1] = g_hCommEvent[COMMEVT_TERM];
#endif
while(1)
{
DWORD dwEvtMask = 0;
DWORD dwWaitResult;
bRes = WaitCommEvent(commhandle, &dwEvtMask, &o); // Will return immediately (probably with ERROR_IO_PENDING)
_ASSERT(!bRes);
if (!bRes)
{
DWORD dwRet = GetLastError();
_ASSERT(dwRet == ERROR_IO_PENDING);
if (dwRet != ERROR_IO_PENDING)
return -1;
//
// Wait for comm event
//
while(1)
{
OutputDebugString("CommThread: Wait1\n");
dwWaitResult = WaitForMultipleObjects(
nNumEvents, // number of handles in array
hCommEvent_Wait, // array of event handles
FALSE, // wait until any one is signaled
INFINITE);
// On very 1st wait *only*: get a false signal (when not running via debugger)
if ((dwWaitResult == WAIT_OBJECT_0) && (dwEvtMask == 0))
continue;
if ((dwWaitResult >= WAIT_OBJECT_0) && (dwWaitResult <= WAIT_OBJECT_0+nNumEvents-1))
break;
}
dwWaitResult -= WAIT_OBJECT_0; // Determine event # that signaled
sprintf(szDbg, "CommThread: GotEvent1: %d\n", dwWaitResult); OutputDebugString(szDbg);
if (dwWaitResult == (nNumEvents-1))
break; // Termination event
}
// Comm event
CheckCommEvent(dwEvtMask);
if (g_vbCommIRQ)
{
//
// Wait for ack
//
while(1)
{
OutputDebugString("CommThread: Wait2\n");
dwWaitResult = WaitForMultipleObjects(
nNumEvents, // number of handles in array
hCommEvent_Ack, // array of event handles
FALSE, // wait until any one is signaled
INFINITE);
if ((dwWaitResult >= WAIT_OBJECT_0) && (dwWaitResult <= WAIT_OBJECT_0+nNumEvents-1))
break;
}
dwWaitResult -= WAIT_OBJECT_0; // Determine event # that signaled
sprintf(szDbg, "CommThread: GotEvent2: %d\n", dwWaitResult); OutputDebugString(szDbg);
if (dwWaitResult == (nNumEvents-1))
break; // Termination event
g_vbCommIRQ = false;
}
}
return 0;
}
bool CommThInit()
{
_ASSERT(g_hCommThread == NULL);
_ASSERT(commhandle);
if ((g_hCommEvent[0] == NULL) && (g_hCommEvent[1] == NULL) && (g_hCommEvent[2] == NULL))
{
// Create an event object for use by WaitCommEvent
o.hEvent = CreateEvent(
NULL, // default security attributes
FALSE, // auto reset event (bManualReset)
FALSE, // not signaled (bInitialState)
NULL // no name
);
// Initialize the rest of the OVERLAPPED structure to zero
o.Internal = 0;
o.InternalHigh = 0;
o.Offset = 0;
o.OffsetHigh = 0;
//
g_hCommEvent[COMMEVT_WAIT] = o.hEvent;
g_hCommEvent[COMMEVT_ACK] = CreateEvent(NULL, // lpEventAttributes
FALSE, // bManualReset (FALSE = auto-reset)
FALSE, // bInitialState (FALSE = non-signaled)
NULL); // lpName
g_hCommEvent[COMMEVT_TERM] = CreateEvent(NULL, // lpEventAttributes
FALSE, // bManualReset (FALSE = auto-reset)
FALSE, // bInitialState (FALSE = non-signaled)
NULL); // lpName
if ((g_hCommEvent[0] == NULL) || (g_hCommEvent[1] == NULL) || (g_hCommEvent[2] == NULL))
{
if(g_fh) fprintf(g_fh, "Comm: CreateEvent failed\n");
return false;
}
}
//
if (g_hCommThread == NULL)
{
DWORD dwThreadId;
g_hCommThread = CreateThread(NULL, // lpThreadAttributes
0, // dwStackSize
CommThread,
NULL, // lpParameter
0, // dwCreationFlags : 0 = Run immediately
&dwThreadId); // lpThreadId
InitializeCriticalSection(&g_CriticalSection);
}
return true;
}
void CommThUninit()
{
if (g_hCommThread)
{
SetEvent(g_hCommEvent[COMMEVT_TERM]); // Signal to thread that it should exit
do
{
DWORD dwExitCode;
if(GetExitCodeThread(g_hCommThread, &dwExitCode))
{
if(dwExitCode == STILL_ACTIVE)
Sleep(10);
else
break;
}
}
while(1);
CloseHandle(g_hCommThread);
g_hCommThread = NULL;
DeleteCriticalSection(&g_CriticalSection);
}
//
for (UINT i=0; i<COMMEVT_MAX; i++)
{
if(g_hCommEvent[i])
{
CloseHandle(g_hCommEvent[i]);
g_hCommEvent[i] = NULL;
}
}
} }
//=========================================================================== //===========================================================================
@ -280,8 +725,8 @@ DWORD CommGetSnapshot(SS_IO_Comms* pSS)
pSS->comminactivity = comminactivity; pSS->comminactivity = comminactivity;
pSS->controlbyte = controlbyte; pSS->controlbyte = controlbyte;
pSS->parity = parity; pSS->parity = parity;
memcpy(pSS->recvbuffer, recvbuffer, 9); memcpy(pSS->recvbuffer, g_RecvBuffer, uRecvBufferSize);
pSS->recvbytes = recvbytes; pSS->recvbytes = g_vRecvBytes;
pSS->stopbits = stopbits; pSS->stopbits = stopbits;
return 0; return 0;
} }
@ -294,8 +739,8 @@ DWORD CommSetSnapshot(SS_IO_Comms* pSS)
comminactivity = pSS->comminactivity; comminactivity = pSS->comminactivity;
controlbyte = pSS->controlbyte; controlbyte = pSS->controlbyte;
parity = pSS->parity; parity = pSS->parity;
memcpy(recvbuffer, pSS->recvbuffer, 9); memcpy(g_RecvBuffer, pSS->recvbuffer, uRecvBufferSize);
recvbytes = pSS->recvbytes; g_vRecvBytes = pSS->recvbytes;
stopbits = pSS->stopbits; stopbits = pSS->stopbits;
return 0; return 0;
} }

View File

@ -6,6 +6,8 @@ void CommDestroy ();
void CommReset (); void CommReset ();
void CommSetSerialPort (HWND,DWORD); void CommSetSerialPort (HWND,DWORD);
void CommUpdate (DWORD); void CommUpdate (DWORD);
bool CommThInit();
void CommThUninit();
DWORD CommGetSnapshot(SS_IO_Comms* pSS); DWORD CommGetSnapshot(SS_IO_Comms* pSS);
DWORD CommSetSnapshot(SS_IO_Comms* pSS); DWORD CommSetSnapshot(SS_IO_Comms* pSS);