2014-04-22 19:04:07 +04:00
|
|
|
/**
|
|
|
|
* WinPR: Windows Portable Runtime
|
|
|
|
* Serial Communication API
|
|
|
|
*
|
|
|
|
* Copyright 2014 Hewlett-Packard Development Company, L.P.
|
|
|
|
*
|
|
|
|
* 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 <stdio.h>
|
|
|
|
|
|
|
|
#include <winpr/comm.h>
|
|
|
|
#include <winpr/crt.h>
|
|
|
|
|
2014-04-29 00:32:27 +04:00
|
|
|
#include "../comm.h"
|
2014-04-22 19:04:07 +04:00
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
static void init_empty_dcb(DCB *pDcb)
|
|
|
|
{
|
|
|
|
ZeroMemory(pDcb, sizeof(DCB));
|
|
|
|
pDcb->DCBlength = sizeof(DCB);
|
|
|
|
pDcb->XonChar = 1;
|
|
|
|
pDcb->XoffChar = 2;
|
|
|
|
}
|
|
|
|
|
2014-04-29 00:32:27 +04:00
|
|
|
static BOOL test_fParity(HANDLE hComm)
|
2014-04-22 19:04:07 +04:00
|
|
|
{
|
|
|
|
DCB dcb;
|
|
|
|
BOOL result;
|
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
init_empty_dcb(&dcb);
|
2014-04-22 19:04:07 +04:00
|
|
|
result = GetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* test 1 */
|
|
|
|
dcb.fParity = TRUE;
|
|
|
|
result = SetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "SetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
init_empty_dcb(&dcb);
|
2014-04-22 19:04:07 +04:00
|
|
|
result = GetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!dcb.fParity)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "unexpected fParity: %d instead of TRUE\n", dcb.fParity);
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* test 2 */
|
|
|
|
dcb.fParity = FALSE;
|
|
|
|
result = SetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "SetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
init_empty_dcb(&dcb);
|
2014-04-22 19:04:07 +04:00
|
|
|
result = GetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (dcb.fParity)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "unexpected fParity: %d instead of FALSE\n", dcb.fParity);
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* test 3 (redo test 1) */
|
|
|
|
dcb.fParity = TRUE;
|
|
|
|
result = SetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "SetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
init_empty_dcb(&dcb);
|
2014-04-22 19:04:07 +04:00
|
|
|
result = GetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!dcb.fParity)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "unexpected fParity: %d instead of TRUE\n", dcb.fParity);
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
2014-04-29 00:32:27 +04:00
|
|
|
return TRUE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
2014-04-29 00:32:27 +04:00
|
|
|
|
|
|
|
static BOOL test_SerialSys(HANDLE hComm)
|
2014-04-22 19:04:07 +04:00
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
DCB dcb;
|
2014-04-22 19:04:07 +04:00
|
|
|
BOOL result;
|
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
init_empty_dcb(&dcb);
|
2014-04-29 00:32:27 +04:00
|
|
|
result = GetCommState(hComm, &dcb);
|
2014-04-22 19:04:07 +04:00
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
2014-04-29 00:32:27 +04:00
|
|
|
/* Test 1 */
|
|
|
|
dcb.BaudRate = SERIAL_BAUD_115200;
|
|
|
|
result = SetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
2014-04-22 19:04:07 +04:00
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "SetCommState failure: 0x%0.8x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
init_empty_dcb(&dcb);
|
2014-04-22 19:04:07 +04:00
|
|
|
result = GetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
if (dcb.BaudRate != SERIAL_BAUD_115200)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "SetCommState failure: could not set BaudRate=%d (SERIAL_BAUD_115200)\n", SERIAL_BAUD_115200);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test 2 using a defferent baud rate */
|
|
|
|
|
|
|
|
dcb.BaudRate = SERIAL_BAUD_57600;
|
|
|
|
result = SetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "SetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
init_empty_dcb(&dcb);
|
2014-04-29 00:32:27 +04:00
|
|
|
result = GetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
if (dcb.BaudRate != SERIAL_BAUD_57600)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "SetCommState failure: could not set BaudRate=%d (SERIAL_BAUD_57600)\n", SERIAL_BAUD_57600);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test 3 using an unsupported baud rate on Linux */
|
|
|
|
dcb.BaudRate = SERIAL_BAUD_128K;
|
|
|
|
result = SetCommState(hComm, &dcb);
|
|
|
|
if (result)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "SetCommState failure: unexpected support of BaudRate=%d (SERIAL_BAUD_128K)\n", SERIAL_BAUD_128K);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL test_SerCxSys(HANDLE hComm)
|
|
|
|
{
|
|
|
|
DCB dcb;
|
|
|
|
BOOL result;
|
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
init_empty_dcb(&dcb);
|
2014-04-29 00:32:27 +04:00
|
|
|
result = GetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test 1 */
|
2014-04-22 19:04:07 +04:00
|
|
|
dcb.BaudRate = CBR_115200;
|
|
|
|
result = SetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "SetCommState failure: 0x%0.8x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
init_empty_dcb(&dcb);
|
2014-04-22 19:04:07 +04:00
|
|
|
result = GetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
if (dcb.BaudRate != CBR_115200)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "SetCommState failure: could not set BaudRate=%d (CBR_115200)\n", CBR_115200);
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Test 2 using a defferent baud rate */
|
|
|
|
|
|
|
|
dcb.BaudRate = CBR_57600;
|
|
|
|
result = SetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "SetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
init_empty_dcb(&dcb);
|
2014-04-22 19:04:07 +04:00
|
|
|
result = GetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
if (dcb.BaudRate != CBR_57600)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "SetCommState failure: could not set BaudRate=%d (CBR_57600)\n", CBR_57600);
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Test 3 using an unsupported baud rate on Linux */
|
|
|
|
dcb.BaudRate = CBR_128000;
|
|
|
|
result = SetCommState(hComm, &dcb);
|
|
|
|
if (result)
|
|
|
|
{
|
2014-04-29 00:32:27 +04:00
|
|
|
fprintf(stderr, "SetCommState failure: unexpected support of BaudRate=%d (CBR_128000)\n", CBR_128000);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static BOOL test_SerCx2Sys(HANDLE hComm)
|
|
|
|
{
|
|
|
|
/* as of today there is no difference */
|
|
|
|
return test_SerCxSys(hComm);
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL test_generic(HANDLE hComm)
|
|
|
|
{
|
|
|
|
DCB dcb, dcb2;
|
|
|
|
BOOL result;
|
2014-06-16 21:18:45 +04:00
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
init_empty_dcb(&dcb);
|
2014-04-29 00:32:27 +04:00
|
|
|
result = GetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Checks whether we get the same information before and after SetCommState */
|
|
|
|
memcpy(&dcb2, &dcb, sizeof(DCB));
|
|
|
|
result = SetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "SetCommState failure: 0x%0.8x\n", GetLastError());
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
result = GetCommState(hComm, &dcb);
|
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memcmp(&dcb, &dcb2, sizeof(DCB)) != 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "DCB is different after SetCommState() whereas it should have not changed\n");
|
|
|
|
return FALSE;
|
2014-04-22 19:04:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: a more complete and generic test using GetCommProperties()
|
|
|
|
|
|
|
|
/* TMP: TODO: fBinary tests */
|
|
|
|
|
|
|
|
/* fParity tests */
|
2014-04-29 00:32:27 +04:00
|
|
|
if (!test_fParity(hComm))
|
|
|
|
{
|
|
|
|
fprintf(stderr, "test_fParity failure\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2014-05-06 18:08:58 +04:00
|
|
|
|
2014-04-29 00:32:27 +04:00
|
|
|
int TestSetCommState(int argc, char* argv[])
|
|
|
|
{
|
|
|
|
BOOL result;
|
|
|
|
HANDLE hComm;
|
|
|
|
|
|
|
|
// TMP: FIXME: check if we can proceed with tests on the actual device, skip and warn otherwise but don't fail
|
|
|
|
result = DefineCommDevice("COM1", "/dev/ttyS0");
|
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "DefineCommDevice failure: 0x%x\n", GetLastError());
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
hComm = CreateFile("COM1",
|
2014-06-16 21:18:45 +04:00
|
|
|
GENERIC_READ | GENERIC_WRITE,
|
|
|
|
0, NULL, OPEN_EXISTING, 0, NULL);
|
2014-04-29 00:32:27 +04:00
|
|
|
if (hComm == INVALID_HANDLE_VALUE)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "CreateFileA failure: 0x%x\n", GetLastError());
|
2014-04-22 19:04:07 +04:00
|
|
|
return EXIT_FAILURE;
|
2014-04-29 00:32:27 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!test_generic(hComm))
|
|
|
|
{
|
2014-06-18 17:58:08 +04:00
|
|
|
fprintf(stderr, "test_generic failure (SerialDriverUnknown)\n");
|
2014-04-29 00:32:27 +04:00
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
2014-06-18 17:58:08 +04:00
|
|
|
_comm_setServerSerialDriver(hComm, SerialDriverSerialSys);
|
2014-04-29 00:32:27 +04:00
|
|
|
if (!test_generic(hComm))
|
|
|
|
{
|
2014-06-18 17:58:08 +04:00
|
|
|
fprintf(stderr, "test_generic failure (SerialDriverSerialSys)\n");
|
2014-04-29 00:32:27 +04:00
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
if (!test_SerialSys(hComm))
|
|
|
|
{
|
|
|
|
fprintf(stderr, "test_SerialSys failure\n");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-18 17:58:08 +04:00
|
|
|
_comm_setServerSerialDriver(hComm, SerialDriverSerCxSys);
|
2014-04-29 00:32:27 +04:00
|
|
|
if (!test_generic(hComm))
|
|
|
|
{
|
2014-06-18 17:58:08 +04:00
|
|
|
fprintf(stderr, "test_generic failure (SerialDriverSerCxSys)\n");
|
2014-04-29 00:32:27 +04:00
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
if (!test_SerCxSys(hComm))
|
|
|
|
{
|
|
|
|
fprintf(stderr, "test_SerCxSys failure\n");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
2014-06-18 17:58:08 +04:00
|
|
|
_comm_setServerSerialDriver(hComm, SerialDriverSerCx2Sys);
|
2014-04-29 00:32:27 +04:00
|
|
|
if (!test_generic(hComm))
|
|
|
|
{
|
2014-06-18 17:58:08 +04:00
|
|
|
fprintf(stderr, "test_generic failure (SerialDriverSerCx2Sys)\n");
|
2014-04-29 00:32:27 +04:00
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
if (!test_SerCx2Sys(hComm))
|
|
|
|
{
|
|
|
|
fprintf(stderr, "test_SerCx2Sys failure\n");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
2014-04-22 19:04:07 +04:00
|
|
|
|
2014-04-29 06:04:09 +04:00
|
|
|
if (!CloseHandle(hComm))
|
|
|
|
{
|
|
|
|
fprintf(stderr, "CloseHandle failure, GetLastError()=%0.8x\n", GetLastError());
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
2014-04-22 19:04:07 +04:00
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|