Commit a05a3435 authored by Eric Pouech's avatar Eric Pouech Committed by Alexandre Julliard

ntdll/kernel32: SetCommState & IOCTL_SET_CHARS

- implemented ntdll's serial IOCTL SET_CHARS - used this IOCTL in kernel32.SetCommState
parent b68203ea
...@@ -983,13 +983,10 @@ static void dump_dcb(const DCB* lpdcb) ...@@ -983,13 +983,10 @@ static void dump_dcb(const DCB* lpdcb)
*/ */
BOOL WINAPI SetCommState( HANDLE handle, LPDCB lpdcb) BOOL WINAPI SetCommState( HANDLE handle, LPDCB lpdcb)
{ {
struct termios port;
int fd;
BOOL ret;
SERIAL_BAUD_RATE sbr; SERIAL_BAUD_RATE sbr;
SERIAL_LINE_CONTROL slc; SERIAL_LINE_CONTROL slc;
SERIAL_HANDFLOW shf; SERIAL_HANDFLOW shf;
SERIAL_CHARS sc;
if (lpdcb == NULL) if (lpdcb == NULL)
{ {
...@@ -1040,44 +1037,24 @@ BOOL WINAPI SetCommState( HANDLE handle, LPDCB lpdcb) ...@@ -1040,44 +1037,24 @@ BOOL WINAPI SetCommState( HANDLE handle, LPDCB lpdcb)
shf.XonLimit = lpdcb->XonLim; shf.XonLimit = lpdcb->XonLim;
shf.XoffLimit = lpdcb->XoffLim; shf.XoffLimit = lpdcb->XoffLim;
sc.EofChar = lpdcb->EofChar;
sc.ErrorChar = lpdcb->ErrorChar;
sc.BreakChar = 0;
sc.EventChar = lpdcb->EvtChar;
sc.XonChar = lpdcb->XonChar;
sc.XoffChar = lpdcb->XoffChar;
/* note: change DTR/RTS lines after setting the comm attributes, /* note: change DTR/RTS lines after setting the comm attributes,
* so flow control does not interfere. * so flow control does not interfere.
*/ */
if (!DeviceIoControl(handle, IOCTL_SERIAL_SET_BAUD_RATE, return (DeviceIoControl(handle, IOCTL_SERIAL_SET_BAUD_RATE,
&sbr, sizeof(sbr), NULL, 0, NULL, NULL)) &sbr, sizeof(sbr), NULL, 0, NULL, NULL) &&
return FALSE; DeviceIoControl(handle, IOCTL_SERIAL_SET_LINE_CONTROL,
if (!DeviceIoControl(handle, IOCTL_SERIAL_SET_LINE_CONTROL, &slc, sizeof(slc), NULL, 0, NULL, NULL) &&
&slc, sizeof(slc), NULL, 0, NULL, NULL)) DeviceIoControl(handle, IOCTL_SERIAL_SET_HANDFLOW,
return FALSE; &shf, sizeof(shf), NULL, 0, NULL, NULL) &&
DeviceIoControl(handle, IOCTL_SERIAL_SET_CHARS,
if (!DeviceIoControl(handle, IOCTL_SERIAL_SET_HANDFLOW, &sc, sizeof(sc), NULL, 0, NULL, NULL));
&shf, sizeof(shf), NULL, 0, NULL, NULL))
return FALSE;
fd = get_comm_fd( handle, FILE_READ_DATA );
if (fd < 0) return FALSE;
if ((tcgetattr(fd,&port)) == -1) {
int save_error = errno;
release_comm_fd( handle, fd );
ERR("tcgetattr error '%s'\n", strerror(save_error));
return FALSE;
}
port.c_cc[VMIN] = 0;
port.c_cc[VTIME] = 1;
if (tcsetattr(fd,TCSANOW,&port)==-1) { /* otherwise it hangs with pending input*/
ERR("tcsetattr error '%s'\n", strerror(errno));
ret = FALSE;
} else {
ClearCommError(handle, NULL, NULL);
ret = TRUE;
}
release_comm_fd( handle, fd );
return ret;
} }
......
...@@ -540,6 +540,34 @@ static NTSTATUS set_line_control(int fd, const SERIAL_LINE_CONTROL* slc) ...@@ -540,6 +540,34 @@ static NTSTATUS set_line_control(int fd, const SERIAL_LINE_CONTROL* slc)
return STATUS_SUCCESS; return STATUS_SUCCESS;
} }
static NTSTATUS set_special_chars(int fd, const SERIAL_CHARS* sc)
{
struct termios port;
if (tcgetattr(fd, &port) == -1)
{
ERR("tcgetattr error '%s'\n", strerror(errno));
return FILE_GetNtStatus();
}
port.c_cc[VMIN ] = 0;
port.c_cc[VTIME ] = 1;
port.c_cc[VEOF ] = sc->EofChar;
/* FIXME: sc->ErrorChar is not supported */
/* FIXME: sc->BreakChar is not supported */
/* FIXME: sc->EventChar is not supported */
port.c_cc[VSTART] = sc->XonChar;
port.c_cc[VSTOP ] = sc->XoffChar;
if (tcsetattr(fd, TCSANOW, &port) == -1)
{
ERR("tcsetattr error '%s'\n", strerror(errno));
return FILE_GetNtStatus();
}
return STATUS_SUCCESS;
}
static NTSTATUS set_wait_mask(HANDLE hDevice, DWORD mask) static NTSTATUS set_wait_mask(HANDLE hDevice, DWORD mask)
{ {
NTSTATUS status; NTSTATUS status;
...@@ -658,6 +686,12 @@ NTSTATUS COMM_DeviceIoControl(HANDLE hDevice, ...@@ -658,6 +686,12 @@ NTSTATUS COMM_DeviceIoControl(HANDLE hDevice,
status = STATUS_NOT_SUPPORTED; status = STATUS_NOT_SUPPORTED;
#endif #endif
break; break;
case IOCTL_SERIAL_SET_CHARS:
if (lpInBuffer && nInBufferSize == sizeof(SERIAL_CHARS))
status = set_special_chars(fd, (const SERIAL_CHARS*)lpInBuffer);
else
status = STATUS_INVALID_PARAMETER;
break;
case IOCTL_SERIAL_SET_HANDFLOW: case IOCTL_SERIAL_SET_HANDFLOW:
if (lpInBuffer && nInBufferSize == sizeof(SERIAL_HANDFLOW)) if (lpInBuffer && nInBufferSize == sizeof(SERIAL_HANDFLOW))
status = set_handflow(fd, (const SERIAL_HANDFLOW*)lpInBuffer); status = set_handflow(fd, (const SERIAL_HANDFLOW*)lpInBuffer);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment