/*========================================================================== * * Copyright (C) 1996-1997 Microsoft Corporation. All Rights Reserved. * * File: serial.c * Content: Routines for serial I/O * History: * Date By Reason * ==== == ====== * 6/10/96 kipo created it * 6/22/96 kipo added support for EnumConnectionData() * 6/25/96 kipo updated for DPADDRESS * 7/13/96 kipo added GetSerialAddress() * 7/16/96 kipo changed address types to be GUIDs instead of 4CC * 8/21/96 kipo move comport address into dplobby.h * 1/06/97 kipo updated for objects * 2/11/97 kipo pass player flags to GetAddress() * 2/18/97 kipo allow multiple instances of service provider * 3/17/97 kipo deal with errors returned by DialogBoxParam() * 5/07/97 kipo added support for modem choice list ***************************************************************************/ #include #include #include "dplaysp.h" #include "comport.h" #include "resource.h" // constants typedef enum { ASCII_XON = 0x11, ASCII_XOFF = 0x13 }; // serial object typedef struct { DPCOMPORT comPort; // base object globals BOOL bHaveSettings; // set to TRUE after settings dialog has been displayed DPCOMPORTADDRESS settings; // settings to use } DPSERIAL, *LPDPSERIAL; // dialog choices for serial port settings static DWORD gComPorts[] = { 1, 2, 3, 4 }; static DWORD gBaudRates[] = { CBR_110, CBR_300, CBR_600, CBR_1200, CBR_2400, CBR_4800, CBR_9600, CBR_14400, CBR_19200, CBR_38400, CBR_56000, CBR_57600, CBR_115200, CBR_128000, CBR_256000 }; static DWORD gStopBits[] = { ONESTOPBIT, ONE5STOPBITS, TWOSTOPBITS }; static DWORD gParities[] = { NOPARITY, EVENPARITY, ODDPARITY, MARKPARITY }; static DWORD gFlowControls[] = { DPCPA_NOFLOW, DPCPA_XONXOFFFLOW, DPCPA_RTSFLOW, DPCPA_DTRFLOW, DPCPA_RTSDTRFLOW }; // globals // this is defined in dllmain.c extern HINSTANCE ghInstance; // this is defined in dpserial.c extern GUID DPSERIAL_GUID; // prototypes static HRESULT DisposeSerial(LPDPCOMPORT baseObject); static HRESULT ConnectSerial(LPDPCOMPORT baseObject, BOOL bWaitForConnection, BOOL bReturnStatus); static HRESULT DisconnectSerial(LPDPCOMPORT baseObject); static HRESULT GetSerialAddress(LPDPCOMPORT baseObject, DWORD dwPlayerFlags, LPVOID lpAddress, LPDWORD lpdwAddressSize); static HRESULT GetSerialAddressChoices(LPDPCOMPORT baseObject, LPVOID lpAddress, LPDWORD lpdwAddressSize); static BOOL SetupConnection(HANDLE hCom, LPDPCOMPORTADDRESS portSettings); static BOOL FAR PASCAL EnumAddressData(REFGUID lpguidDataType, DWORD dwDataSize, LPCVOID lpData, LPVOID lpContext); static BOOL GetSerialSettings(HINSTANCE hInstance, HWND hWndParent, LPDPSERIAL globals); static UINT_PTR CALLBACK SettingsDialog(HWND hDlg, UINT msg, WPARAM wParam, LPARAM lParam); static void InitDialog(HWND hDlg, LPDPCOMPORTADDRESS settings); static void GetSettingsFromDialog(HWND hDlg, LPDPCOMPORTADDRESS settings); static int ValueToIndex(LPDWORD buf, int bufLen, DWORD value); static void FillComboBox(HWND hDlg, int dlgItem, int startStr, int stopStr); /* * NewSerial * * Create new serial port object. */ HRESULT NewSerial(LPVOID lpConnectionData, DWORD dwConnectionDataSize, LPDIRECTPLAYSP lpDPlay, LPREADROUTINE lpReadRoutine, LPDPCOMPORT *storage) { LPDPCOMPORT baseObject; LPDPSERIAL globals; HRESULT hr; // create base object with enough space for our globals hr = NewComPort(sizeof(DPSERIAL), lpDPlay, lpReadRoutine, &baseObject); if FAILED(hr) return (hr); // fill in methods we implement baseObject->Dispose = DisposeSerial; baseObject->Connect = ConnectSerial; baseObject->Disconnect = DisconnectSerial; baseObject->GetAddress = GetSerialAddress; baseObject->GetAddressChoices = GetSerialAddressChoices; // setup default settings globals = (LPDPSERIAL) baseObject; globals->settings.dwComPort = 1; // COM port to use (1-4) globals->settings.dwBaudRate = CBR_57600; // baud rate (100-256k) globals->settings.dwStopBits = ONESTOPBIT; // no. stop bits (1-2) globals->settings.dwParity = NOPARITY; // parity (none, odd, even, mark) globals->settings.dwFlowControl = DPCPA_RTSDTRFLOW; // flow control (none, xon/xoff, rts, dtr) // check for valid connection data if (lpConnectionData) { baseObject->lpDPlay->lpVtbl->EnumAddress(baseObject->lpDPlay, EnumAddressData, lpConnectionData, dwConnectionDataSize, globals); } // return object pointer *storage = baseObject; return (DP_OK); } /* * EnumConnectionData * * Search for valid connection data */ static BOOL FAR PASCAL EnumAddressData(REFGUID lpguidDataType, DWORD dwDataSize, LPCVOID lpData, LPVOID lpContext) { LPDPSERIAL globals = (LPDPSERIAL) lpContext; LPDPCOMPORTADDRESS settings = (LPDPCOMPORTADDRESS) lpData; // this is a com port chunk if ( IsEqualGUID(lpguidDataType, &DPAID_ComPort) && (dwDataSize == sizeof(DPCOMPORTADDRESS)) ) { // make sure it's valid! if ((ValueToIndex(gComPorts, sizeof(gComPorts), settings->dwComPort) >= 0) && (ValueToIndex(gBaudRates, sizeof(gBaudRates), settings->dwBaudRate) >= 0) && (ValueToIndex(gStopBits, sizeof(gStopBits), settings->dwStopBits) >= 0) && (ValueToIndex(gParities, sizeof(gParities), settings->dwParity) >= 0) && (ValueToIndex(gFlowControls, sizeof(gFlowControls), settings->dwFlowControl) >= 0)) { globals->settings = *settings; // copy the data globals->bHaveSettings = TRUE; // we have valid settings } } return (TRUE); } /* * DisposeSerial * * Dispose serial port object. */ static HRESULT DisposeSerial(LPDPCOMPORT baseObject) { LPDPSERIAL globals = (LPDPSERIAL) baseObject; // make sure we are disconnected DisconnectSerial(baseObject); // free object GlobalFreePtr((HGLOBAL) baseObject); return (DP_OK); } /* * ConnectSerial * * Open serial port and configure based on user settings. */ static HRESULT ConnectSerial(LPDPCOMPORT baseObject, BOOL bWaitForConnection, BOOL bReturnStatus) { LPDPSERIAL globals = (LPDPSERIAL) baseObject; HANDLE hCom; TCHAR portName[10]; HRESULT hr; // see if com port is already connected hCom = baseObject->GetHandle(baseObject); if (hCom) return (DP_OK); // ask user for settings if we have not already if (!globals->bHaveSettings) { if (!GetSerialSettings(ghInstance, GetForegroundWindow(), globals)) { hr = DPERR_USERCANCEL; goto Failure; } globals->bHaveSettings = TRUE; } // open specified com port CopyMemory(portName, "COM0", 5); portName[3] += (BYTE) globals->settings.dwComPort; hCom = CreateFile( portName, GENERIC_READ | GENERIC_WRITE, 0, /* comm devices must be opened w/exclusive-access */ NULL, /* no security attrs */ OPEN_EXISTING, /* comm devices must use OPEN_EXISTING */ FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED, // overlapped I/O NULL /* hTemplate must be NULL for comm devices */ ); if (hCom == INVALID_HANDLE_VALUE) { hCom = NULL; hr = HRESULT_FROM_WIN32(GetLastError()); goto Failure; } // configure com port to proper settings if (!SetupConnection(hCom, &globals->settings)) { hr = HRESULT_FROM_WIN32(GetLastError()); goto Failure; } // setup com port hr = baseObject->Setup(baseObject, hCom); if FAILED(hr) goto Failure; return (DP_OK); Failure: if (hCom) CloseHandle(hCom); return (hr); } /* * DisconnectSerial * * Close serial port. */ static HRESULT DisconnectSerial(LPDPCOMPORT baseObject) { HANDLE hCom; HRESULT hr; hCom = baseObject->GetHandle(baseObject); // com port is already disconnected if (hCom == NULL) return (DP_OK); // shut down com port hr = baseObject->Shutdown(baseObject); // close com port CloseHandle(hCom); return (hr); } /* * SetupConnection * * Configure serial port with specified settings. */ static BOOL SetupConnection(HANDLE hCom, LPDPCOMPORTADDRESS portSettings) { DCB dcb; dcb.DCBlength = sizeof(DCB); if (!GetCommState(hCom, &dcb)) return (FALSE); // setup various port settings dcb.fBinary = TRUE; dcb.BaudRate = portSettings->dwBaudRate; dcb.ByteSize = 8; dcb.StopBits = (BYTE) portSettings->dwStopBits; dcb.Parity = (BYTE) portSettings->dwParity; if (portSettings->dwParity == NOPARITY) dcb.fParity = FALSE; else dcb.fParity = TRUE; // setup hardware flow control if ((portSettings->dwFlowControl == DPCPA_DTRFLOW) || (portSettings->dwFlowControl == DPCPA_RTSDTRFLOW)) { dcb.fOutxDsrFlow = TRUE; dcb.fDtrControl = DTR_CONTROL_HANDSHAKE; } else { dcb.fOutxDsrFlow = FALSE; dcb.fDtrControl = DTR_CONTROL_ENABLE; } if ((portSettings->dwFlowControl == DPCPA_RTSFLOW) || (portSettings->dwFlowControl == DPCPA_RTSDTRFLOW)) { dcb.fOutxCtsFlow = TRUE; dcb.fRtsControl = RTS_CONTROL_HANDSHAKE; } else { dcb.fOutxCtsFlow = FALSE; dcb.fRtsControl = RTS_CONTROL_ENABLE; } // setup software flow control if (portSettings->dwFlowControl == DPCPA_XONXOFFFLOW) { dcb.fInX = TRUE; dcb.fOutX = TRUE; } else { dcb.fInX = FALSE; dcb.fOutX = FALSE; } dcb.XonChar = ASCII_XON; dcb.XoffChar = ASCII_XOFF; dcb.XonLim = 100; dcb.XoffLim = 100; if (!SetCommState( hCom, &dcb )) return (FALSE); return (TRUE); } /* * GetSerialAddress * * Return current serial port info if available. */ static HRESULT GetSerialAddress(LPDPCOMPORT baseObject, DWORD dwPlayerFlags, LPVOID lpAddress, LPDWORD lpdwAddressSize) { LPDPSERIAL globals = (LPDPSERIAL) baseObject; HRESULT hResult; // no settings yet if (!globals->bHaveSettings) return (DPERR_UNAVAILABLE); hResult = baseObject->lpDPlay->lpVtbl->CreateAddress(baseObject->lpDPlay, &DPSERIAL_GUID, &DPAID_ComPort, &globals->settings, sizeof(DPCOMPORTADDRESS), lpAddress, lpdwAddressSize); return (hResult); } /* * GetSerialAddressChoices * * Return current serial address choices */ static HRESULT GetSerialAddressChoices(LPDPCOMPORT baseObject, LPVOID lpAddress, LPDWORD lpdwAddressSize) { LPDPSERIAL globals = (LPDPSERIAL) baseObject; // currently the serial provider does not support any choices return (E_NOTIMPL); } /* * GetComPortSettings * * Displays a dialog to gather and return the COM port settings. */ static BOOL GetSerialSettings(HINSTANCE hInstance, HWND hWndParent, LPDPSERIAL globals) { INT_PTR iResult; iResult = (INT_PTR)DialogBoxParam(hInstance, MAKEINTRESOURCE(IDD_SETTINGSDIALOG), hWndParent, SettingsDialog, (LPARAM) globals); return (iResult > 0); } /* * SettingsDialog * * The dialog callback routine to display and edit the COM port settings. */ static UINT_PTR CALLBACK SettingsDialog(HWND hDlg, UINT msg, WPARAM wParam, LPARAM lParam) { LPDPSERIAL globals = (LPDPSERIAL) GetWindowLongPtr(hDlg, DWLP_USER); HWND hWndCtl; BOOL msgHandled = FALSE; switch (msg) { case WM_INITDIALOG: // serial info pointer passed in lParam globals = (LPDPSERIAL) lParam; // save the globals with the window SetWindowLongPtr(hDlg, DWLP_USER, (LONG_PTR) globals); hWndCtl = GetDlgItem(hDlg, IDC_COMPORT); // make sure our dialog item is there if (hWndCtl == NULL) { EndDialog(hDlg, FALSE); msgHandled = TRUE; } else { InitDialog(hDlg, &globals->settings); // setup our dialog SetFocus(hWndCtl); // focus on com port combo box msgHandled = FALSE; // keep windows from setting input focus for us } break; case WM_COMMAND: if (HIWORD(wParam) == 0) { switch (LOWORD(wParam)) { case IDOK: // return settings GetSettingsFromDialog(hDlg, &globals->settings); EndDialog(hDlg, TRUE); msgHandled = TRUE; break; case IDCANCEL: // cancel EndDialog(hDlg, FALSE); msgHandled = TRUE; break; } } break; } return (msgHandled); } /* * InitDialog * * Initialize the dialog controls to display the given COM port settings. */ static void InitDialog(HWND hDlg, LPDPCOMPORTADDRESS settings) { // fill dialog combo boxes with items from string table FillComboBox(hDlg, IDC_COMPORT, IDS_COM1, IDS_COM4); FillComboBox(hDlg, IDC_BAUDRATE, IDS_BAUD1, IDS_BAUD15); FillComboBox(hDlg, IDC_STOPBITS, IDS_STOPBIT1, IDS_STOPBIT3); FillComboBox(hDlg, IDC_PARITY, IDS_PARITY1, IDS_PARITY4); FillComboBox(hDlg, IDC_FLOW, IDS_FLOW1, IDS_FLOW5); // select default values in combo boxes SendDlgItemMessage(hDlg, IDC_COMPORT, CB_SETCURSEL, ValueToIndex(gComPorts, sizeof(gComPorts), settings->dwComPort), 0); SendDlgItemMessage(hDlg, IDC_BAUDRATE, CB_SETCURSEL, ValueToIndex(gBaudRates, sizeof(gBaudRates), settings->dwBaudRate), 0); SendDlgItemMessage(hDlg, IDC_STOPBITS, CB_SETCURSEL, ValueToIndex(gStopBits, sizeof(gStopBits), settings->dwStopBits), 0); SendDlgItemMessage(hDlg, IDC_PARITY, CB_SETCURSEL, ValueToIndex(gParities, sizeof(gParities), settings->dwParity), 0); SendDlgItemMessage(hDlg, IDC_FLOW, CB_SETCURSEL, ValueToIndex(gFlowControls, sizeof(gFlowControls), settings->dwFlowControl), 0); } /* * GetSettingsFromDialog * * Get the COM port settings from the dialog controls. */ static void GetSettingsFromDialog(HWND hDlg, LPDPCOMPORTADDRESS settings) { INT_PTR index; index = SendDlgItemMessage(hDlg, IDC_COMPORT, CB_GETCURSEL, 0, 0); if (index == CB_ERR) return; settings->dwComPort = gComPorts[index]; index = SendDlgItemMessage(hDlg, IDC_BAUDRATE, CB_GETCURSEL, 0, 0); if (index == CB_ERR) return; settings->dwBaudRate = gBaudRates[index]; index = SendDlgItemMessage(hDlg, IDC_STOPBITS, CB_GETCURSEL, 0, 0); if (index == CB_ERR) return; settings->dwStopBits = gStopBits[index]; index = SendDlgItemMessage(hDlg, IDC_PARITY, CB_GETCURSEL, 0, 0); if (index == CB_ERR) return; settings->dwParity = gParities[index]; index = SendDlgItemMessage(hDlg, IDC_FLOW, CB_GETCURSEL, 0, 0); if (index == CB_ERR) return; settings->dwFlowControl = gFlowControls[index]; } /* * FillComboBox * * Add the specified strings to the combo box. */ #define MAXSTRINGSIZE 200 static void FillComboBox(HWND hDlg, int dlgItem, int startStr, int stopStr) { int i; TCHAR str[MAXSTRINGSIZE]; for (i = startStr; i <= stopStr; i++) { if (LoadString(ghInstance, i, str, MAXSTRINGSIZE)) SendDlgItemMessage(hDlg, dlgItem, CB_ADDSTRING, (WPARAM) 0, (LPARAM) str); } } /* * ValueToIndex * * Convert a settings value to a combo box selection index. */ static int ValueToIndex(LPDWORD buf, int bufLen, DWORD value) { int i; bufLen /= sizeof(DWORD); for (i = 0; i < bufLen; i++) if (buf[i] == value) return (i); return (-1); }