commit
86b4123cf0
@ -53,6 +53,7 @@ typedef struct _SERIAL_DEVICE SERIAL_DEVICE;
|
|||||||
struct _SERIAL_DEVICE
|
struct _SERIAL_DEVICE
|
||||||
{
|
{
|
||||||
DEVICE device;
|
DEVICE device;
|
||||||
|
BOOL permissive;
|
||||||
SERIAL_DRIVER_ID ServerSerialDriverId;
|
SERIAL_DRIVER_ID ServerSerialDriverId;
|
||||||
HANDLE* hComm;
|
HANDLE* hComm;
|
||||||
|
|
||||||
@ -182,11 +183,7 @@ static void serial_process_irp_create(SERIAL_DEVICE* serial, IRP* irp)
|
|||||||
|
|
||||||
_comm_setServerSerialDriver(serial->hComm, serial->ServerSerialDriverId);
|
_comm_setServerSerialDriver(serial->hComm, serial->ServerSerialDriverId);
|
||||||
|
|
||||||
/* FIXME: Appeared to be useful to setup some devices. Guess
|
_comm_set_permissive(serial->hComm, serial->permissive);
|
||||||
* the device driver asked to setup some unsupported feature
|
|
||||||
* that were not eventually used. TODO: collecting more
|
|
||||||
* details, a command line argument? */
|
|
||||||
/* _comm_set_permissive(serial->hComm, TRUE); */
|
|
||||||
|
|
||||||
/* NOTE: binary mode/raw mode required for the redirection. On
|
/* NOTE: binary mode/raw mode required for the redirection. On
|
||||||
* Linux, CommCreateFileA forces this setting.
|
* Linux, CommCreateFileA forces this setting.
|
||||||
@ -817,6 +814,21 @@ int DeviceServiceEntry(PDEVICE_SERVICE_ENTRY_POINTS pEntryPoints)
|
|||||||
serial->ServerSerialDriverId = SerialDriverSerCx2Sys;
|
serial->ServerSerialDriverId = SerialDriverSerCx2Sys;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (device->Permissive != NULL)
|
||||||
|
{
|
||||||
|
if (_stricmp(device->Permissive, "permissive") == 0)
|
||||||
|
{
|
||||||
|
serial->permissive = TRUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
WLog_Print(serial->log, WLOG_DEBUG, "Unknown flag: %s", device->Permissive);
|
||||||
|
assert(FALSE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
WLog_Print(serial->log, WLOG_DEBUG, "Server's serial driver: %s (id: %d)", driver, serial->ServerSerialDriverId);
|
WLog_Print(serial->log, WLOG_DEBUG, "Server's serial driver: %s (id: %d)", driver, serial->ServerSerialDriverId);
|
||||||
/* TODO: implement auto detection of the server's serial driver */
|
/* TODO: implement auto detection of the server's serial driver */
|
||||||
|
|
||||||
|
@ -256,8 +256,8 @@ int freerdp_client_print_command_line_help(int argc, char** argv)
|
|||||||
|
|
||||||
printf("Drive Redirection: /drive:home,/home/user\n");
|
printf("Drive Redirection: /drive:home,/home/user\n");
|
||||||
printf("Smartcard Redirection: /smartcard:<device>\n");
|
printf("Smartcard Redirection: /smartcard:<device>\n");
|
||||||
printf("Printer Redirection: /printer:<device>,<driver>\n");
|
printf("Serial Port Redirection: /serial:<name>,<device>,[SerCx2|SerCx|Serial],[permissive]\n");
|
||||||
printf("Serial Port Redirection: /serial:<device>\n");
|
printf("Serial Port Redirection: /serial:COM1,/dev/ttyS0\n");
|
||||||
printf("Parallel Port Redirection: /parallel:<device>\n");
|
printf("Parallel Port Redirection: /parallel:<device>\n");
|
||||||
printf("Printer Redirection: /printer:<device>,<driver>\n");
|
printf("Printer Redirection: /printer:<device>,<driver>\n");
|
||||||
printf("\n");
|
printf("\n");
|
||||||
@ -425,6 +425,9 @@ int freerdp_client_add_device_channel(rdpSettings* settings, int count, char** p
|
|||||||
if (count > 3)
|
if (count > 3)
|
||||||
serial->Driver = _strdup(params[3]);
|
serial->Driver = _strdup(params[3]);
|
||||||
|
|
||||||
|
if (count > 4)
|
||||||
|
serial->Permissive = _strdup(params[4]);
|
||||||
|
|
||||||
freerdp_device_collection_add(settings, (RDPDR_DEVICE*) serial);
|
freerdp_device_collection_add(settings, (RDPDR_DEVICE*) serial);
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
|
@ -472,6 +472,7 @@ struct _RDPDR_SERIAL
|
|||||||
char* Name;
|
char* Name;
|
||||||
char* Path;
|
char* Path;
|
||||||
char* Driver;
|
char* Driver;
|
||||||
|
char* Permissive;
|
||||||
};
|
};
|
||||||
typedef struct _RDPDR_SERIAL RDPDR_SERIAL;
|
typedef struct _RDPDR_SERIAL RDPDR_SERIAL;
|
||||||
|
|
||||||
|
@ -174,29 +174,6 @@
|
|||||||
#define BAUD_57600 ((DWORD)0x00040000)
|
#define BAUD_57600 ((DWORD)0x00040000)
|
||||||
#define BAUD_USER ((DWORD)0x10000000)
|
#define BAUD_USER ((DWORD)0x10000000)
|
||||||
|
|
||||||
/* Ntddser.h: http://msdn.microsoft.com/en-us/cc308432.aspx */
|
|
||||||
#define SERIAL_BAUD_075 ((ULONG)0x00000001)
|
|
||||||
#define SERIAL_BAUD_110 ((ULONG)0x00000002)
|
|
||||||
#define SERIAL_BAUD_134_5 ((ULONG)0x00000004)
|
|
||||||
#define SERIAL_BAUD_150 ((ULONG)0x00000008)
|
|
||||||
#define SERIAL_BAUD_300 ((ULONG)0x00000010)
|
|
||||||
#define SERIAL_BAUD_600 ((ULONG)0x00000020)
|
|
||||||
#define SERIAL_BAUD_1200 ((ULONG)0x00000040)
|
|
||||||
#define SERIAL_BAUD_1800 ((ULONG)0x00000080)
|
|
||||||
#define SERIAL_BAUD_2400 ((ULONG)0x00000100)
|
|
||||||
#define SERIAL_BAUD_4800 ((ULONG)0x00000200)
|
|
||||||
#define SERIAL_BAUD_7200 ((ULONG)0x00000400)
|
|
||||||
#define SERIAL_BAUD_9600 ((ULONG)0x00000800)
|
|
||||||
#define SERIAL_BAUD_14400 ((ULONG)0x00001000)
|
|
||||||
#define SERIAL_BAUD_19200 ((ULONG)0x00002000)
|
|
||||||
#define SERIAL_BAUD_38400 ((ULONG)0x00004000)
|
|
||||||
#define SERIAL_BAUD_56K ((ULONG)0x00008000)
|
|
||||||
#define SERIAL_BAUD_128K ((ULONG)0x00010000)
|
|
||||||
#define SERIAL_BAUD_115200 ((ULONG)0x00020000)
|
|
||||||
#define SERIAL_BAUD_57600 ((ULONG)0x00040000)
|
|
||||||
#define SERIAL_BAUD_USER ((ULONG)0x10000000)
|
|
||||||
|
|
||||||
|
|
||||||
#define DATABITS_5 ((WORD)0x0001)
|
#define DATABITS_5 ((WORD)0x0001)
|
||||||
#define DATABITS_6 ((WORD)0x0002)
|
#define DATABITS_6 ((WORD)0x0002)
|
||||||
#define DATABITS_7 ((WORD)0x0004)
|
#define DATABITS_7 ((WORD)0x0004)
|
||||||
|
@ -44,12 +44,17 @@ struct winpr_comm
|
|||||||
int fd_write_event; /* as of today, only used by _purge() */
|
int fd_write_event; /* as of today, only used by _purge() */
|
||||||
CRITICAL_SECTION WriteLock;
|
CRITICAL_SECTION WriteLock;
|
||||||
|
|
||||||
/* permissive mode on errors if TRUE (default is FALSE).
|
/* permissive mode on errors. If TRUE (default is FALSE)
|
||||||
|
* CommDeviceIoControl always return TRUE.
|
||||||
*
|
*
|
||||||
* Since not all features are supported, some devices and applications
|
* Not all features are supported yet and an error is then returned when
|
||||||
* can still be functional on such errors.
|
* an application turns them on (e.g: i/o buffers > 4096). It appeared
|
||||||
|
* though that devices and applications can be still functional on such
|
||||||
|
* errors.
|
||||||
*
|
*
|
||||||
* TODO: command line switch or getting rid of it.
|
* see also: comm_ioctl.c
|
||||||
|
*
|
||||||
|
* FIXME: getting rid of this flag once all features supported.
|
||||||
*/
|
*/
|
||||||
BOOL permissive;
|
BOOL permissive;
|
||||||
|
|
||||||
|
@ -163,10 +163,10 @@ SERIAL_DRIVER* SerCx2Sys_s()
|
|||||||
SERIAL_DRIVER* pSerialSys = SerialSys_s();
|
SERIAL_DRIVER* pSerialSys = SerialSys_s();
|
||||||
SERIAL_DRIVER* pSerCxSys = SerCxSys_s();
|
SERIAL_DRIVER* pSerCxSys = SerCxSys_s();
|
||||||
|
|
||||||
_SerCx2Sys.set_baud_rate = pSerCxSys->set_baud_rate;
|
_SerCx2Sys.set_baud_rate = pSerialSys->set_baud_rate;
|
||||||
_SerCx2Sys.get_baud_rate = pSerCxSys->get_baud_rate;
|
_SerCx2Sys.get_baud_rate = pSerialSys->get_baud_rate;
|
||||||
|
|
||||||
_SerCx2Sys.get_properties = pSerCxSys->get_properties;
|
_SerCx2Sys.get_properties = pSerialSys->get_properties;
|
||||||
|
|
||||||
_SerCx2Sys.set_line_control = pSerCxSys->set_line_control;
|
_SerCx2Sys.set_line_control = pSerCxSys->set_line_control;
|
||||||
_SerCx2Sys.get_line_control = pSerCxSys->get_line_control;
|
_SerCx2Sys.get_line_control = pSerCxSys->get_line_control;
|
||||||
|
@ -30,207 +30,6 @@
|
|||||||
#include "comm_serial_sys.h"
|
#include "comm_serial_sys.h"
|
||||||
|
|
||||||
|
|
||||||
/* 0: B* (Linux termios)
|
|
||||||
* 1: CBR_* or actual baud rate
|
|
||||||
* 2: BAUD_* (similar to SERIAL_BAUD_*)
|
|
||||||
*/
|
|
||||||
static const speed_t _SERCX_SYS_BAUD_TABLE[][3] = {
|
|
||||||
#ifdef B0
|
|
||||||
{B0, 0, 0}, /* hang up */
|
|
||||||
#endif
|
|
||||||
#ifdef B50
|
|
||||||
{B50, 50, 0},
|
|
||||||
#endif
|
|
||||||
#ifdef B75
|
|
||||||
{B75, 75, BAUD_075},
|
|
||||||
#endif
|
|
||||||
#ifdef B110
|
|
||||||
{B110, CBR_110, BAUD_110},
|
|
||||||
#endif
|
|
||||||
#ifdef B134
|
|
||||||
{B134, 134, 0 /*BAUD_134_5*/},
|
|
||||||
#endif
|
|
||||||
#ifdef B150
|
|
||||||
{B150, 150, BAUD_150},
|
|
||||||
#endif
|
|
||||||
#ifdef B200
|
|
||||||
{B200, 200, 0},
|
|
||||||
#endif
|
|
||||||
#ifdef B300
|
|
||||||
{B300, CBR_300, BAUD_300},
|
|
||||||
#endif
|
|
||||||
#ifdef B600
|
|
||||||
{B600, CBR_600, BAUD_600},
|
|
||||||
#endif
|
|
||||||
#ifdef B1200
|
|
||||||
{B1200, CBR_1200, BAUD_1200},
|
|
||||||
#endif
|
|
||||||
#ifdef B1800
|
|
||||||
{B1800, 1800, BAUD_1800},
|
|
||||||
#endif
|
|
||||||
#ifdef B2400
|
|
||||||
{B2400, CBR_2400, BAUD_2400},
|
|
||||||
#endif
|
|
||||||
#ifdef B4800
|
|
||||||
{B4800, CBR_4800, BAUD_4800},
|
|
||||||
#endif
|
|
||||||
/* {, ,BAUD_7200} */
|
|
||||||
#ifdef B9600
|
|
||||||
{B9600, CBR_9600, BAUD_9600},
|
|
||||||
#endif
|
|
||||||
/* {, CBR_14400, BAUD_14400}, /\* unsupported on Linux *\/ */
|
|
||||||
#ifdef B19200
|
|
||||||
{B19200, CBR_19200, BAUD_19200},
|
|
||||||
#endif
|
|
||||||
#ifdef B38400
|
|
||||||
{B38400, CBR_38400, BAUD_38400},
|
|
||||||
#endif
|
|
||||||
/* {, CBR_56000, BAUD_56K}, /\* unsupported on Linux *\/ */
|
|
||||||
#ifdef B57600
|
|
||||||
{B57600, CBR_57600, BAUD_57600},
|
|
||||||
#endif
|
|
||||||
#ifdef B115200
|
|
||||||
{B115200, CBR_115200, BAUD_115200},
|
|
||||||
#endif
|
|
||||||
/* {, CBR_128000, BAUD_128K}, /\* unsupported on Linux *\/ */
|
|
||||||
/* {, CBR_256000, BAUD_USER}, /\* unsupported on Linux *\/ */
|
|
||||||
#ifdef B230400
|
|
||||||
{B230400, 230400, BAUD_USER},
|
|
||||||
#endif
|
|
||||||
#ifdef B460800
|
|
||||||
{B460800, 460800, BAUD_USER},
|
|
||||||
#endif
|
|
||||||
#ifdef B500000
|
|
||||||
{B500000, 500000, BAUD_USER},
|
|
||||||
#endif
|
|
||||||
#ifdef B576000
|
|
||||||
{B576000, 576000, BAUD_USER},
|
|
||||||
#endif
|
|
||||||
#ifdef B921600
|
|
||||||
{B921600, 921600, BAUD_USER},
|
|
||||||
#endif
|
|
||||||
#ifdef B1000000
|
|
||||||
{B1000000, 1000000, BAUD_USER},
|
|
||||||
#endif
|
|
||||||
#ifdef B1152000
|
|
||||||
{B1152000, 1152000, BAUD_USER},
|
|
||||||
#endif
|
|
||||||
#ifdef B1500000
|
|
||||||
{B1500000, 1500000, BAUD_USER},
|
|
||||||
#endif
|
|
||||||
#ifdef B2000000
|
|
||||||
{B2000000, 2000000, BAUD_USER},
|
|
||||||
#endif
|
|
||||||
#ifdef B2500000
|
|
||||||
{B2500000, 2500000, BAUD_USER},
|
|
||||||
#endif
|
|
||||||
#ifdef B3000000
|
|
||||||
{B3000000, 3000000, BAUD_USER},
|
|
||||||
#endif
|
|
||||||
#ifdef B3500000
|
|
||||||
{B3500000, 3500000, BAUD_USER},
|
|
||||||
#endif
|
|
||||||
#ifdef B4000000
|
|
||||||
{B4000000, 4000000, BAUD_USER}, /* __MAX_BAUD */
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
static BOOL _get_properties(WINPR_COMM *pComm, COMMPROP *pProperties)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
SERIAL_DRIVER* pSerialSys = SerialSys_s();
|
|
||||||
|
|
||||||
if (!pSerialSys->get_properties(pComm, pProperties))
|
|
||||||
{
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* override some of the inherited properties from SerialSys ... */
|
|
||||||
|
|
||||||
pProperties->dwMaxBaud = BAUD_USER;
|
|
||||||
|
|
||||||
pProperties->dwSettableBaud = 0;
|
|
||||||
for (i=0; _SERCX_SYS_BAUD_TABLE[i][0]<=__MAX_BAUD; i++)
|
|
||||||
{
|
|
||||||
pProperties->dwSettableBaud |= _SERCX_SYS_BAUD_TABLE[i][2];
|
|
||||||
}
|
|
||||||
|
|
||||||
return TRUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static BOOL _set_baud_rate(WINPR_COMM *pComm, const SERIAL_BAUD_RATE *pBaudRate)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
speed_t newSpeed;
|
|
||||||
struct termios futureState;
|
|
||||||
|
|
||||||
ZeroMemory(&futureState, sizeof(struct termios));
|
|
||||||
if (tcgetattr(pComm->fd, &futureState) < 0) /* NB: preserves current settings not directly handled by the Communication Functions */
|
|
||||||
{
|
|
||||||
SetLastError(ERROR_IO_DEVICE);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (i=0; _SERCX_SYS_BAUD_TABLE[i][0]<=__MAX_BAUD; i++)
|
|
||||||
{
|
|
||||||
if (_SERCX_SYS_BAUD_TABLE[i][1] == pBaudRate->BaudRate)
|
|
||||||
{
|
|
||||||
newSpeed = _SERCX_SYS_BAUD_TABLE[i][0];
|
|
||||||
if (cfsetspeed(&futureState, newSpeed) < 0)
|
|
||||||
{
|
|
||||||
CommLog_Print(WLOG_WARN, "failed to set speed 0x%x (%lu)", newSpeed, pBaudRate->BaudRate);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(cfgetispeed(&futureState) == newSpeed);
|
|
||||||
|
|
||||||
if (_comm_ioctl_tcsetattr(pComm->fd, TCSANOW, &futureState) < 0)
|
|
||||||
{
|
|
||||||
CommLog_Print(WLOG_WARN, "_comm_ioctl_tcsetattr failure: last-error: 0x%lX", GetLastError());
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
return TRUE;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
CommLog_Print(WLOG_WARN, "could not find a matching speed for the baud rate %lu", pBaudRate->BaudRate);
|
|
||||||
SetLastError(ERROR_INVALID_DATA);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static BOOL _get_baud_rate(WINPR_COMM *pComm, SERIAL_BAUD_RATE *pBaudRate)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
speed_t currentSpeed;
|
|
||||||
struct termios currentState;
|
|
||||||
|
|
||||||
ZeroMemory(¤tState, sizeof(struct termios));
|
|
||||||
if (tcgetattr(pComm->fd, ¤tState) < 0)
|
|
||||||
{
|
|
||||||
SetLastError(ERROR_IO_DEVICE);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
currentSpeed = cfgetispeed(¤tState);
|
|
||||||
|
|
||||||
for (i=0; _SERCX_SYS_BAUD_TABLE[i][0]<=__MAX_BAUD; i++)
|
|
||||||
{
|
|
||||||
if (_SERCX_SYS_BAUD_TABLE[i][0] == currentSpeed)
|
|
||||||
{
|
|
||||||
pBaudRate->BaudRate = _SERCX_SYS_BAUD_TABLE[i][1];
|
|
||||||
return TRUE;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
CommLog_Print(WLOG_WARN, "could not find a matching baud rate for the speed 0x%x", currentSpeed);
|
|
||||||
SetLastError(ERROR_INVALID_DATA);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
static BOOL _set_handflow(WINPR_COMM *pComm, const SERIAL_HANDFLOW *pHandflow)
|
static BOOL _set_handflow(WINPR_COMM *pComm, const SERIAL_HANDFLOW *pHandflow)
|
||||||
{
|
{
|
||||||
SERIAL_HANDFLOW SerCxHandflow;
|
SERIAL_HANDFLOW SerCxHandflow;
|
||||||
@ -373,9 +172,9 @@ static SERIAL_DRIVER _SerCxSys =
|
|||||||
{
|
{
|
||||||
.id = SerialDriverSerCxSys,
|
.id = SerialDriverSerCxSys,
|
||||||
.name = _T("SerCx.sys"),
|
.name = _T("SerCx.sys"),
|
||||||
.set_baud_rate = _set_baud_rate,
|
.set_baud_rate = NULL,
|
||||||
.get_baud_rate = _get_baud_rate,
|
.get_baud_rate = NULL,
|
||||||
.get_properties = _get_properties,
|
.get_properties = NULL,
|
||||||
.set_serial_chars = NULL,
|
.set_serial_chars = NULL,
|
||||||
.get_serial_chars = NULL,
|
.get_serial_chars = NULL,
|
||||||
.set_line_control = NULL,
|
.set_line_control = NULL,
|
||||||
@ -412,6 +211,11 @@ SERIAL_DRIVER* SerCxSys_s()
|
|||||||
/* _SerCxSys completed with inherited functions from SerialSys */
|
/* _SerCxSys completed with inherited functions from SerialSys */
|
||||||
SERIAL_DRIVER* pSerialSys = SerialSys_s();
|
SERIAL_DRIVER* pSerialSys = SerialSys_s();
|
||||||
|
|
||||||
|
_SerCxSys.set_baud_rate = pSerialSys->set_baud_rate;
|
||||||
|
_SerCxSys.get_baud_rate = pSerialSys->get_baud_rate;
|
||||||
|
|
||||||
|
_SerCxSys.get_properties = pSerialSys->get_properties;
|
||||||
|
|
||||||
_SerCxSys.set_serial_chars = pSerialSys->set_serial_chars;
|
_SerCxSys.set_serial_chars = pSerialSys->set_serial_chars;
|
||||||
_SerCxSys.get_serial_chars = pSerialSys->get_serial_chars;
|
_SerCxSys.get_serial_chars = pSerialSys->get_serial_chars;
|
||||||
_SerCxSys.set_line_control = pSerialSys->set_line_control;
|
_SerCxSys.set_line_control = pSerialSys->set_line_control;
|
||||||
|
@ -43,76 +43,114 @@
|
|||||||
#define N_TTY_BUF_SIZE 4096
|
#define N_TTY_BUF_SIZE 4096
|
||||||
|
|
||||||
|
|
||||||
|
#define _BAUD_TABLE_END 0010020 /* __MAX_BAUD + 1 */
|
||||||
|
|
||||||
/*
|
/* 0: B* (Linux termios)
|
||||||
* Linux, Windows speeds
|
* 1: CBR_* or actual baud rate
|
||||||
*
|
* 2: BAUD_* (identical to SERIAL_BAUD_*)
|
||||||
*/
|
*/
|
||||||
static const speed_t _SERIAL_SYS_BAUD_TABLE[][2] = {
|
static const speed_t _BAUD_TABLE[][3] = {
|
||||||
#ifdef B0
|
#ifdef B0
|
||||||
{B0, 0}, /* hang up */
|
{B0, 0, 0}, /* hang up */
|
||||||
|
#endif
|
||||||
|
#ifdef B50
|
||||||
|
{B50, 50, 0},
|
||||||
#endif
|
#endif
|
||||||
/* #ifdef B50 */
|
|
||||||
/* {B50, }, /\* undefined by serial.sys *\/ */
|
|
||||||
/* #endif */
|
|
||||||
#ifdef B75
|
#ifdef B75
|
||||||
{B75, SERIAL_BAUD_075},
|
{B75, 75, BAUD_075},
|
||||||
#endif
|
#endif
|
||||||
#ifdef B110
|
#ifdef B110
|
||||||
{B110, SERIAL_BAUD_110},
|
{B110, CBR_110, BAUD_110},
|
||||||
|
#endif
|
||||||
|
#ifdef B134
|
||||||
|
{B134, 134, 0 /*BAUD_134_5*/},
|
||||||
#endif
|
#endif
|
||||||
/* #ifdef B134 */
|
|
||||||
/* {B134, SERIAL_BAUD_134_5}, /\* TODO: might be the same? *\/ */
|
|
||||||
/* #endif */
|
|
||||||
#ifdef B150
|
#ifdef B150
|
||||||
{B150, SERIAL_BAUD_150},
|
{B150, 150, BAUD_150},
|
||||||
|
#endif
|
||||||
|
#ifdef B200
|
||||||
|
{B200, 200, 0},
|
||||||
#endif
|
#endif
|
||||||
/* #ifdef B200 */
|
|
||||||
/* {B200, }, /\* undefined by serial.sys *\/ */
|
|
||||||
/* #endif */
|
|
||||||
#ifdef B300
|
#ifdef B300
|
||||||
{B300, SERIAL_BAUD_300},
|
{B300, CBR_300, BAUD_300},
|
||||||
#endif
|
#endif
|
||||||
#ifdef B600
|
#ifdef B600
|
||||||
{B600, SERIAL_BAUD_600},
|
{B600, CBR_600, BAUD_600},
|
||||||
#endif
|
#endif
|
||||||
#ifdef B1200
|
#ifdef B1200
|
||||||
{B1200, SERIAL_BAUD_1200},
|
{B1200, CBR_1200, BAUD_1200},
|
||||||
#endif
|
#endif
|
||||||
#ifdef B1800
|
#ifdef B1800
|
||||||
{B1800, SERIAL_BAUD_1800},
|
{B1800, 1800, BAUD_1800},
|
||||||
#endif
|
#endif
|
||||||
#ifdef B2400
|
#ifdef B2400
|
||||||
{B2400, SERIAL_BAUD_2400},
|
{B2400, CBR_2400, BAUD_2400},
|
||||||
#endif
|
#endif
|
||||||
#ifdef B4800
|
#ifdef B4800
|
||||||
{B4800, SERIAL_BAUD_4800},
|
{B4800, CBR_4800, BAUD_4800},
|
||||||
#endif
|
#endif
|
||||||
/* {, SERIAL_BAUD_7200} /\* undefined on Linux *\/ */
|
/* {, ,BAUD_7200} */
|
||||||
#ifdef B9600
|
#ifdef B9600
|
||||||
{B9600, SERIAL_BAUD_9600},
|
{B9600, CBR_9600, BAUD_9600},
|
||||||
#endif
|
#endif
|
||||||
/* {, SERIAL_BAUD_14400} /\* undefined on Linux *\/ */
|
/* {, CBR_14400, BAUD_14400}, /\* unsupported on Linux *\/ */
|
||||||
#ifdef B19200
|
#ifdef B19200
|
||||||
{B19200, SERIAL_BAUD_19200},
|
{B19200, CBR_19200, BAUD_19200},
|
||||||
#endif
|
#endif
|
||||||
#ifdef B38400
|
#ifdef B38400
|
||||||
{B38400, SERIAL_BAUD_38400},
|
{B38400, CBR_38400, BAUD_38400},
|
||||||
#endif
|
#endif
|
||||||
/* {, SERIAL_BAUD_56K}, /\* undefined on Linux *\/ */
|
/* {, CBR_56000, BAUD_56K}, /\* unsupported on Linux *\/ */
|
||||||
#ifdef B57600
|
#ifdef B57600
|
||||||
{B57600, SERIAL_BAUD_57600},
|
{B57600, CBR_57600, BAUD_57600},
|
||||||
#endif
|
#endif
|
||||||
/* {, SERIAL_BAUD_128K} /\* undefined on Linux *\/ */
|
|
||||||
#ifdef B115200
|
#ifdef B115200
|
||||||
{B115200, SERIAL_BAUD_115200}, /* _SERIAL_MAX_BAUD */
|
{B115200, CBR_115200, BAUD_115200},
|
||||||
#endif
|
#endif
|
||||||
|
/* {, CBR_128000, BAUD_128K}, /\* unsupported on Linux *\/ */
|
||||||
/* no greater speed defined by serial.sys */
|
/* {, CBR_256000, BAUD_USER}, /\* unsupported on Linux *\/ */
|
||||||
|
#ifdef B230400
|
||||||
|
{B230400, 230400, BAUD_USER},
|
||||||
|
#endif
|
||||||
|
#ifdef B460800
|
||||||
|
{B460800, 460800, BAUD_USER},
|
||||||
|
#endif
|
||||||
|
#ifdef B500000
|
||||||
|
{B500000, 500000, BAUD_USER},
|
||||||
|
#endif
|
||||||
|
#ifdef B576000
|
||||||
|
{B576000, 576000, BAUD_USER},
|
||||||
|
#endif
|
||||||
|
#ifdef B921600
|
||||||
|
{B921600, 921600, BAUD_USER},
|
||||||
|
#endif
|
||||||
|
#ifdef B1000000
|
||||||
|
{B1000000, 1000000, BAUD_USER},
|
||||||
|
#endif
|
||||||
|
#ifdef B1152000
|
||||||
|
{B1152000, 1152000, BAUD_USER},
|
||||||
|
#endif
|
||||||
|
#ifdef B1500000
|
||||||
|
{B1500000, 1500000, BAUD_USER},
|
||||||
|
#endif
|
||||||
|
#ifdef B2000000
|
||||||
|
{B2000000, 2000000, BAUD_USER},
|
||||||
|
#endif
|
||||||
|
#ifdef B2500000
|
||||||
|
{B2500000, 2500000, BAUD_USER},
|
||||||
|
#endif
|
||||||
|
#ifdef B3000000
|
||||||
|
{B3000000, 3000000, BAUD_USER},
|
||||||
|
#endif
|
||||||
|
#ifdef B3500000
|
||||||
|
{B3500000, 3500000, BAUD_USER},
|
||||||
|
#endif
|
||||||
|
#ifdef B4000000
|
||||||
|
{B4000000, 4000000, BAUD_USER}, /* __MAX_BAUD */
|
||||||
|
#endif
|
||||||
|
{_BAUD_TABLE_END, 0, 0}
|
||||||
};
|
};
|
||||||
|
|
||||||
#define _SERIAL_MAX_BAUD B115200
|
|
||||||
|
|
||||||
|
|
||||||
static BOOL _get_properties(WINPR_COMM *pComm, COMMPROP *pProperties)
|
static BOOL _get_properties(WINPR_COMM *pComm, COMMPROP *pProperties)
|
||||||
{
|
{
|
||||||
@ -142,7 +180,8 @@ static BOOL _get_properties(WINPR_COMM *pComm, COMMPROP *pProperties)
|
|||||||
pProperties->dwMaxTxQueue = N_TTY_BUF_SIZE;
|
pProperties->dwMaxTxQueue = N_TTY_BUF_SIZE;
|
||||||
pProperties->dwMaxRxQueue = N_TTY_BUF_SIZE;
|
pProperties->dwMaxRxQueue = N_TTY_BUF_SIZE;
|
||||||
|
|
||||||
pProperties->dwMaxBaud = SERIAL_BAUD_115200; /* _SERIAL_MAX_BAUD */
|
/* FIXME: to be probe on the device? */
|
||||||
|
pProperties->dwMaxBaud = BAUD_USER;
|
||||||
|
|
||||||
/* FIXME: what about PST_RS232? see also: serial_struct */
|
/* FIXME: what about PST_RS232? see also: serial_struct */
|
||||||
pProperties->dwProvSubType = PST_UNSPECIFIED;
|
pProperties->dwProvSubType = PST_UNSPECIFIED;
|
||||||
@ -156,9 +195,9 @@ static BOOL _get_properties(WINPR_COMM *pComm, COMMPROP *pProperties)
|
|||||||
pProperties->dwSettableParams = SP_BAUD | SP_DATABITS | SP_HANDSHAKING | SP_PARITY | SP_PARITY_CHECK | /*SP_RLSD |*/ SP_STOPBITS;
|
pProperties->dwSettableParams = SP_BAUD | SP_DATABITS | SP_HANDSHAKING | SP_PARITY | SP_PARITY_CHECK | /*SP_RLSD |*/ SP_STOPBITS;
|
||||||
|
|
||||||
pProperties->dwSettableBaud = 0;
|
pProperties->dwSettableBaud = 0;
|
||||||
for (i=0; _SERIAL_SYS_BAUD_TABLE[i][0]<=_SERIAL_MAX_BAUD; i++)
|
for (i=0; _BAUD_TABLE[i][0]<_BAUD_TABLE_END; i++)
|
||||||
{
|
{
|
||||||
pProperties->dwSettableBaud |= _SERIAL_SYS_BAUD_TABLE[i][1];
|
pProperties->dwSettableBaud |= _BAUD_TABLE[i][2];
|
||||||
}
|
}
|
||||||
|
|
||||||
pProperties->wSettableData = DATABITS_5 | DATABITS_6 | DATABITS_7 | DATABITS_8 /*| DATABITS_16 | DATABITS_16X*/;
|
pProperties->wSettableData = DATABITS_5 | DATABITS_6 | DATABITS_7 | DATABITS_8 /*| DATABITS_16 | DATABITS_16X*/;
|
||||||
@ -180,27 +219,29 @@ static BOOL _set_baud_rate(WINPR_COMM *pComm, const SERIAL_BAUD_RATE *pBaudRate)
|
|||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
speed_t newSpeed;
|
speed_t newSpeed;
|
||||||
struct termios upcomingTermios;
|
struct termios futureState;
|
||||||
|
|
||||||
ZeroMemory(&upcomingTermios, sizeof(struct termios));
|
ZeroMemory(&futureState, sizeof(struct termios));
|
||||||
if (tcgetattr(pComm->fd, &upcomingTermios) < 0)
|
if (tcgetattr(pComm->fd, &futureState) < 0) /* NB: preserves current settings not directly handled by the Communication Functions */
|
||||||
{
|
{
|
||||||
SetLastError(ERROR_IO_DEVICE);
|
SetLastError(ERROR_IO_DEVICE);
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i=0; _SERIAL_SYS_BAUD_TABLE[i][0]<=_SERIAL_MAX_BAUD; i++)
|
for (i=0; _BAUD_TABLE[i][0]<_BAUD_TABLE_END; i++)
|
||||||
{
|
{
|
||||||
if (_SERIAL_SYS_BAUD_TABLE[i][1] == pBaudRate->BaudRate)
|
if (_BAUD_TABLE[i][1] == pBaudRate->BaudRate)
|
||||||
{
|
{
|
||||||
newSpeed = _SERIAL_SYS_BAUD_TABLE[i][0];
|
newSpeed = _BAUD_TABLE[i][0];
|
||||||
if (cfsetspeed(&upcomingTermios, newSpeed) < 0)
|
if (cfsetspeed(&futureState, newSpeed) < 0)
|
||||||
{
|
{
|
||||||
CommLog_Print(WLOG_WARN, "failed to set speed %u (%lu)", newSpeed, pBaudRate->BaudRate);
|
CommLog_Print(WLOG_WARN, "failed to set speed 0x%x (%lu)", newSpeed, pBaudRate->BaudRate);
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_comm_ioctl_tcsetattr(pComm->fd, TCSANOW, &upcomingTermios) < 0)
|
assert(cfgetispeed(&futureState) == newSpeed);
|
||||||
|
|
||||||
|
if (_comm_ioctl_tcsetattr(pComm->fd, TCSANOW, &futureState) < 0)
|
||||||
{
|
{
|
||||||
CommLog_Print(WLOG_WARN, "_comm_ioctl_tcsetattr failure: last-error: 0x%lX", GetLastError());
|
CommLog_Print(WLOG_WARN, "_comm_ioctl_tcsetattr failure: last-error: 0x%lX", GetLastError());
|
||||||
return FALSE;
|
return FALSE;
|
||||||
@ -231,11 +272,11 @@ static BOOL _get_baud_rate(WINPR_COMM *pComm, SERIAL_BAUD_RATE *pBaudRate)
|
|||||||
|
|
||||||
currentSpeed = cfgetispeed(¤tState);
|
currentSpeed = cfgetispeed(¤tState);
|
||||||
|
|
||||||
for (i=0; _SERIAL_SYS_BAUD_TABLE[i][0]<=_SERIAL_MAX_BAUD; i++)
|
for (i=0; _BAUD_TABLE[i][0]<_BAUD_TABLE_END; i++)
|
||||||
{
|
{
|
||||||
if (_SERIAL_SYS_BAUD_TABLE[i][0] == currentSpeed)
|
if (_BAUD_TABLE[i][0] == currentSpeed)
|
||||||
{
|
{
|
||||||
pBaudRate->BaudRate = _SERIAL_SYS_BAUD_TABLE[i][1];
|
pBaudRate->BaudRate = _BAUD_TABLE[i][1];
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -245,6 +286,7 @@ static BOOL _get_baud_rate(WINPR_COMM *pComm, SERIAL_BAUD_RATE *pBaudRate)
|
|||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* NOTE: Only XonChar and XoffChar are plenty supported with the Linux
|
* NOTE: Only XonChar and XoffChar are plenty supported with the Linux
|
||||||
* N_TTY line discipline.
|
* N_TTY line discipline.
|
||||||
|
@ -132,76 +132,6 @@ static BOOL test_SerialSys(HANDLE hComm)
|
|||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Test 1 */
|
|
||||||
dcb.BaudRate = SERIAL_BAUD_115200;
|
|
||||||
result = SetCommState(hComm, &dcb);
|
|
||||||
if (!result)
|
|
||||||
{
|
|
||||||
fprintf(stderr, "SetCommState failure: 0x%08x\n", GetLastError());
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
init_empty_dcb(&dcb);
|
|
||||||
result = GetCommState(hComm, &dcb);
|
|
||||||
if (!result)
|
|
||||||
{
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
|
|
||||||
init_empty_dcb(&dcb);
|
|
||||||
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;
|
|
||||||
|
|
||||||
init_empty_dcb(&dcb);
|
|
||||||
result = GetCommState(hComm, &dcb);
|
|
||||||
if (!result)
|
|
||||||
{
|
|
||||||
fprintf(stderr, "GetCommState failure: 0x%x\n", GetLastError());
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Test 1 */
|
/* Test 1 */
|
||||||
dcb.BaudRate = CBR_115200;
|
dcb.BaudRate = CBR_115200;
|
||||||
result = SetCommState(hComm, &dcb);
|
result = SetCommState(hComm, &dcb);
|
||||||
@ -259,11 +189,16 @@ static BOOL test_SerCxSys(HANDLE hComm)
|
|||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static BOOL test_SerCxSys(HANDLE hComm)
|
||||||
|
{
|
||||||
|
/* as of today there is no difference */
|
||||||
|
return test_SerialSys(hComm);
|
||||||
|
}
|
||||||
|
|
||||||
static BOOL test_SerCx2Sys(HANDLE hComm)
|
static BOOL test_SerCx2Sys(HANDLE hComm)
|
||||||
{
|
{
|
||||||
/* as of today there is no difference */
|
/* as of today there is no difference */
|
||||||
return test_SerCxSys(hComm);
|
return test_SerialSys(hComm);
|
||||||
}
|
}
|
||||||
|
|
||||||
static BOOL test_generic(HANDLE hComm)
|
static BOOL test_generic(HANDLE hComm)
|
||||||
|
Loading…
Reference in New Issue
Block a user