reactos/base/applications/cmdutils/mode/mode.c

1338 lines
37 KiB
C

/*
* PROJECT: ReactOS Mode Utility
* LICENSE: GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+)
* PURPOSE: Provides fast mode setup for DOS devices.
* COPYRIGHT: Copyright 2002 Robert Dickenson
* Copyright 2016-2021 Hermes Belusca-Maito
*/
/*
* ReactOS mode console command
*
* mode.c
*
* Copyright (C) 2002 Robert Dickenson <robd@reactos.org>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include <stdio.h>
#include <stdlib.h>
#include <windef.h>
#include <winbase.h>
#include <winuser.h>
#include <wincon.h>
#include <conutils.h>
#include "resource.h"
#define MAX_PORTNAME_LEN 20
#define ASSERT(a)
/*** For fixes, see also network/net/main.c ***/
// VOID PrintPadding(...)
VOID
__cdecl
UnderlinedResPrintf(
IN PCON_STREAM Stream,
IN UINT uID,
...)
{
INT Len;
va_list args;
#define MAX_BUFFER_SIZE 4096
INT i;
WCHAR szMsgBuffer[MAX_BUFFER_SIZE];
va_start(args, uID);
Len = ConResPrintfV(Stream, uID, args);
va_end(args);
ConPuts(Stream, L"\n");
for (i = 0; i < Len; i++)
szMsgBuffer[i] = L'-';
szMsgBuffer[Len] = UNICODE_NULL;
ConStreamWrite(Stream, szMsgBuffer, Len);
}
int ShowParallelStatus(INT nPortNum)
{
WCHAR buffer[250];
WCHAR szPortName[MAX_PORTNAME_LEN];
swprintf(szPortName, L"LPT%d", nPortNum);
ConPuts(StdOut, L"\n");
UnderlinedResPrintf(StdOut, IDS_DEVICE_STATUS_HEADER, szPortName);
ConPuts(StdOut, L"\n");
if (QueryDosDeviceW(szPortName, buffer, ARRAYSIZE(buffer)))
{
PWSTR ptr = wcsrchr(buffer, L'\\');
if (ptr != NULL)
{
if (_wcsicmp(szPortName, ++ptr) == 0)
ConResPuts(StdOut, IDS_PRINTER_OUTPUT_NOT_REROUTED);
else
ConResPrintf(StdOut, IDS_PRINTER_OUTPUT_REROUTED_SERIAL, ptr);
return 0;
}
else
{
ConResPrintf(StdErr, IDS_ERROR_QUERY_DEVICES_FORM, szPortName, buffer);
}
}
else
{
ConPrintf(StdErr, L"ERROR: QueryDosDeviceW(%s) failed: 0x%lx\n", szPortName, GetLastError());
}
ConPuts(StdOut, L"\n");
return 1;
}
int SetParallelState(INT nPortNum)
{
WCHAR szPortName[MAX_PORTNAME_LEN];
WCHAR szTargetPath[MAX_PORTNAME_LEN];
swprintf(szPortName, L"LPT%d", nPortNum);
swprintf(szTargetPath, L"COM%d", nPortNum);
if (!DefineDosDeviceW(DDD_REMOVE_DEFINITION, szPortName, szTargetPath))
{
ConPrintf(StdErr, L"ERROR: SetParallelState(%d) - DefineDosDevice(%s) failed: 0x%lx\n", nPortNum, szPortName, GetLastError());
}
ShowParallelStatus(nPortNum);
return 0;
}
static PCWSTR
ParseNumber(PCWSTR argStr, PDWORD Number)
{
INT value, skip = 0;
value = swscanf(argStr, L"%lu%n", Number, &skip);
if (!value) return NULL;
argStr += skip;
return argStr;
}
/*
\??\COM1
\Device\NamedPipe\Spooler\LPT1
BOOL DefineDosDevice(
DWORD dwFlags, // options
LPCTSTR lpDeviceName, // device name
LPCTSTR lpTargetPath // path string
);
DWORD QueryDosDevice(
LPCTSTR lpDeviceName, // MS-DOS device name string
LPTSTR lpTargetPath, // query results buffer
DWORD ucchMax // maximum size of buffer
);
*/
/*****************************************************************************\
** C O N S O L E H E L P E R S **
\*****************************************************************************/
int ShowConsoleStatus(VOID)
{
HANDLE hConOut = GetStdHandle(STD_OUTPUT_HANDLE);
CONSOLE_SCREEN_BUFFER_INFO csbi;
DWORD dwKbdDelay, dwKbdSpeed;
ConPuts(StdOut, L"\n");
UnderlinedResPrintf(StdOut, IDS_DEVICE_STATUS_HEADER, L"CON");
ConPuts(StdOut, L"\n");
if (GetConsoleScreenBufferInfo(hConOut, &csbi))
{
ConResPrintf(StdOut, IDS_CONSOLE_STATUS_LINES, csbi.dwSize.Y);
ConResPrintf(StdOut, IDS_CONSOLE_STATUS_COLS , csbi.dwSize.X);
}
if (SystemParametersInfoW(SPI_GETKEYBOARDSPEED, 0, &dwKbdSpeed, 0))
{
ConResPrintf(StdOut, IDS_CONSOLE_KBD_RATE, dwKbdSpeed);
}
if (SystemParametersInfoW(SPI_GETKEYBOARDDELAY, 0, &dwKbdDelay, 0))
{
ConResPrintf(StdOut, IDS_CONSOLE_KBD_DELAY, dwKbdDelay);
}
ConResPrintf(StdOut, IDS_CONSOLE_CODEPAGE, GetConsoleOutputCP());
ConPuts(StdOut, L"\n");
return 0;
}
int ShowConsoleCPStatus(VOID)
{
ConPuts(StdOut, L"\n");
UnderlinedResPrintf(StdOut, IDS_DEVICE_STATUS_HEADER, L"CON");
ConPuts(StdOut, L"\n");
ConResPrintf(StdOut, IDS_CONSOLE_CODEPAGE, GetConsoleOutputCP());
ConPuts(StdOut, L"\n");
return 0;
}
static VOID
ClearScreen(
IN HANDLE hConOut,
IN PCONSOLE_SCREEN_BUFFER_INFO pcsbi)
{
COORD coPos;
DWORD dwWritten;
coPos.X = 0;
coPos.Y = 0;
FillConsoleOutputAttribute(hConOut, pcsbi->wAttributes,
pcsbi->dwSize.X * pcsbi->dwSize.Y,
coPos, &dwWritten);
FillConsoleOutputCharacterW(hConOut, L' ',
pcsbi->dwSize.X * pcsbi->dwSize.Y,
coPos, &dwWritten);
SetConsoleCursorPosition(hConOut, coPos);
}
/*
* See, or adjust if needed, subsystems/mvdm/ntvdm/console/video.c!ResizeTextConsole()
* for more information.
*/
static BOOL
ResizeTextConsole(
IN HANDLE hConOut,
IN OUT PCONSOLE_SCREEN_BUFFER_INFO pcsbi,
IN COORD Resolution)
{
BOOL Success;
SHORT Width, Height;
SMALL_RECT ConRect;
/*
* Use this trick to effectively resize the console buffer and window,
* because:
* - SetConsoleScreenBufferSize fails if the new console screen buffer size
* is smaller than the current console window size, and:
* - SetConsoleWindowInfo fails if the new console window size is larger
* than the current console screen buffer size.
*/
/* Resize the screen buffer only if needed */
if (Resolution.X != pcsbi->dwSize.X || Resolution.Y != pcsbi->dwSize.Y)
{
Width = pcsbi->srWindow.Right - pcsbi->srWindow.Left + 1;
Height = pcsbi->srWindow.Bottom - pcsbi->srWindow.Top + 1;
/*
* If the current console window is too large for
* the new screen buffer, resize it first.
*/
if (Width > Resolution.X || Height > Resolution.Y)
{
/*
* NOTE: This is not a problem if we move the window back to (0,0)
* because when we resize the screen buffer, the window will move back
* to where the cursor is. Or, if the screen buffer is not resized,
* when we readjust again the window, we will move back to a correct
* position. This is what we wanted after all...
*/
ConRect.Left = ConRect.Top = 0;
ConRect.Right = ConRect.Left + min(Width , Resolution.X) - 1;
ConRect.Bottom = ConRect.Top + min(Height, Resolution.Y) - 1;
Success = SetConsoleWindowInfo(hConOut, TRUE, &ConRect);
if (!Success) return FALSE;
}
/*
* Now resize the screen buffer.
*
* SetConsoleScreenBufferSize automatically takes into account the current
* cursor position when it computes starting which row it should copy text
* when resizing the screen buffer, and scrolls the console window such that
* the cursor is placed in it again. We therefore do not need to care about
* the cursor position and do the maths ourselves.
*/
Success = SetConsoleScreenBufferSize(hConOut, Resolution);
if (!Success) return FALSE;
/*
* Setting a new screen buffer size can change other information,
* so update the console screen buffer information.
*/
GetConsoleScreenBufferInfo(hConOut, pcsbi);
}
/* Always resize the console window within the permitted maximum size */
Width = min(Resolution.X, pcsbi->dwMaximumWindowSize.X);
Height = min(Resolution.Y, pcsbi->dwMaximumWindowSize.Y);
ConRect.Left = 0;
ConRect.Right = ConRect.Left + Width - 1;
ConRect.Bottom = max(pcsbi->dwCursorPosition.Y, Height - 1);
ConRect.Top = ConRect.Bottom - Height + 1;
SetConsoleWindowInfo(hConOut, TRUE, &ConRect);
/* Update the console screen buffer information */
GetConsoleScreenBufferInfo(hConOut, pcsbi);
return TRUE;
}
int SetConsoleStateOld(IN PCWSTR ArgStr)
{
PCWSTR argStr = ArgStr;
HANDLE hConOut = GetStdHandle(STD_OUTPUT_HANDLE);
CONSOLE_SCREEN_BUFFER_INFO csbi;
COORD Resolution;
DWORD value;
if (!GetConsoleScreenBufferInfo(hConOut, &csbi))
{
// TODO: Error message?
return 0;
}
Resolution = csbi.dwSize;
/* Parse the column number (only MANDATORY argument) */
value = 0;
argStr = ParseNumber(argStr, &value);
if (!argStr) goto invalid_parameter;
Resolution.X = (SHORT)value;
/* Parse the line number (OPTIONAL argument) */
while (*argStr == L' ') argStr++;
if (!*argStr) goto Quit;
if (*argStr++ != L',') goto invalid_parameter;
while (*argStr == L' ') argStr++;
value = 0;
argStr = ParseNumber(argStr, &value);
if (!argStr) goto invalid_parameter;
Resolution.Y = (SHORT)value;
/* This should be the end of the string */
while (*argStr == L' ') argStr++;
if (*argStr) goto invalid_parameter;
Quit:
ClearScreen(hConOut, &csbi);
if (!ResizeTextConsole(hConOut, &csbi, Resolution))
ConResPuts(StdErr, IDS_ERROR_SCREEN_LINES_COL);
return 0;
invalid_parameter:
ConResPrintf(StdErr, IDS_ERROR_INVALID_PARAMETER, ArgStr);
return 1;
}
int SetConsoleState(IN PCWSTR ArgStr)
{
PCWSTR argStr = ArgStr;
BOOL dispMode = FALSE, kbdMode = FALSE;
HANDLE hConOut = GetStdHandle(STD_OUTPUT_HANDLE);
CONSOLE_SCREEN_BUFFER_INFO csbi;
COORD Resolution;
DWORD dwKbdDelay, dwKbdSpeed;
DWORD value;
if (!GetConsoleScreenBufferInfo(hConOut, &csbi))
{
// TODO: Error message?
return 0;
}
if (!SystemParametersInfoW(SPI_GETKEYBOARDDELAY, 0, &dwKbdDelay, 0))
{
// TODO: Error message?
return 0;
}
if (!SystemParametersInfoW(SPI_GETKEYBOARDSPEED, 0, &dwKbdSpeed, 0))
{
// TODO: Error message?
return 0;
}
Resolution = csbi.dwSize;
while (argStr && *argStr)
{
while (*argStr == L' ') argStr++;
if (!*argStr) break;
if (!kbdMode && _wcsnicmp(argStr, L"COLS=", 5) == 0)
{
dispMode = TRUE;
value = 0;
argStr = ParseNumber(argStr+5, &value);
if (!argStr) goto invalid_parameter;
Resolution.X = (SHORT)value;
}
else if (!kbdMode && _wcsnicmp(argStr, L"LINES=", 6) == 0)
{
dispMode = TRUE;
value = 0;
argStr = ParseNumber(argStr+6, &value);
if (!argStr) goto invalid_parameter;
Resolution.Y = (SHORT)value;
}
else if (!dispMode && _wcsnicmp(argStr, L"RATE=", 5) == 0)
{
kbdMode = TRUE;
argStr = ParseNumber(argStr+5, &dwKbdSpeed);
if (!argStr) goto invalid_parameter;
}
else if (!dispMode && _wcsnicmp(argStr, L"DELAY=", 6) == 0)
{
kbdMode = TRUE;
argStr = ParseNumber(argStr+6, &dwKbdDelay);
if (!argStr) goto invalid_parameter;
}
else
{
invalid_parameter:
ConResPrintf(StdErr, IDS_ERROR_INVALID_PARAMETER, ArgStr);
return 1;
}
}
if (dispMode)
{
ClearScreen(hConOut, &csbi);
if (!ResizeTextConsole(hConOut, &csbi, Resolution))
ConResPuts(StdErr, IDS_ERROR_SCREEN_LINES_COL);
}
else if (kbdMode)
{
/*
* Set the new keyboard settings. If those values are greater than
* their allowed range, they are automatically corrected as follows:
* dwKbdSpeed = min(dwKbdSpeed, 31);
* dwKbdDelay = (dwKbdDelay % 4);
*/
SystemParametersInfoW(SPI_SETKEYBOARDDELAY, dwKbdDelay, NULL, 0);
// "Invalid keyboard delay."
SystemParametersInfoW(SPI_SETKEYBOARDSPEED, dwKbdSpeed, NULL, SPIF_UPDATEINIFILE | SPIF_SENDCHANGE);
// "Invalid keyboard rate."
}
return 0;
}
int SetConsoleCPState(IN PCWSTR ArgStr)
{
PCWSTR argStr = ArgStr;
DWORD value = 0;
UINT uOldCodePage, uNewCodePage;
if ( (_wcsnicmp(argStr, L"SELECT=", 7) == 0 && (argStr += 7)) ||
(_wcsnicmp(argStr, L"SEL=", 4) == 0 && (argStr += 4)) )
{
argStr = ParseNumber(argStr, &value);
if (!argStr) goto invalid_parameter;
/* This should be the end of the string */
while (*argStr == L' ') argStr++;
if (*argStr) goto invalid_parameter;
}
else
{
invalid_parameter:
ConResPrintf(StdErr, IDS_ERROR_INVALID_PARAMETER, ArgStr);
return 1;
}
uNewCodePage = value;
/**
** IMPORTANT NOTE: This code must be kept synchronized with CHCP.COM
**/
/*
* Save the original console code page to be restored
* in case SetConsoleCP() or SetConsoleOutputCP() fails.
*/
uOldCodePage = GetConsoleCP();
/*
* Try changing the console input and output code pages.
* If it succeeds, refresh the local code page information.
*/
if (SetConsoleCP(uNewCodePage))
{
if (SetConsoleOutputCP(uNewCodePage))
{
/* Success, reset the current thread UI language
* and update the streams cached code page. */
ConSetThreadUILanguage(0);
ConStdStreamsSetCacheCodePage(uNewCodePage, uNewCodePage);
/* Display the current console status */
ShowConsoleStatus();
return 0;
}
else
{
/* Failure, restore the original console code page */
SetConsoleCP(uOldCodePage);
}
}
/* An error happened, display an error and bail out */
ConResPuts(StdErr, IDS_ERROR_INVALID_CODEPAGE);
return 1;
}
/*****************************************************************************\
** S E R I A L P O R T H E L P E R S **
\*****************************************************************************/
static BOOL
SerialPortQuery(INT nPortNum, LPDCB pDCB, LPCOMMTIMEOUTS pCommTimeouts, BOOL bWrite)
{
BOOL Success;
HANDLE hPort;
WCHAR szPortName[MAX_PORTNAME_LEN];
ASSERT(pDCB);
ASSERT(pCommTimeouts);
swprintf(szPortName, L"COM%d", nPortNum);
hPort = CreateFileW(szPortName,
bWrite ? GENERIC_WRITE : GENERIC_READ,
0, // exclusive
NULL, // sec attr
OPEN_EXISTING,
0, // no attributes
NULL); // no template
if (hPort == INVALID_HANDLE_VALUE)
{
DWORD dwLastError = GetLastError();
if (dwLastError == ERROR_ACCESS_DENIED)
ConResPrintf(StdErr, IDS_ERROR_DEVICE_NOT_AVAILABLE, szPortName);
else
ConResPrintf(StdErr, IDS_ERROR_ILLEGAL_DEVICE_NAME, szPortName, dwLastError);
return FALSE;
}
Success = bWrite ? SetCommState(hPort, pDCB)
: GetCommState(hPort, pDCB);
if (!Success)
{
ConResPrintf(StdErr,
bWrite ? IDS_ERROR_STATUS_SET_DEVICE : IDS_ERROR_STATUS_GET_DEVICE,
szPortName);
goto Quit;
}
Success = bWrite ? SetCommTimeouts(hPort, pCommTimeouts)
: GetCommTimeouts(hPort, pCommTimeouts);
if (!Success)
{
ConResPrintf(StdErr,
bWrite ? IDS_ERROR_TIMEOUT_SET_DEVICE : IDS_ERROR_TIMEOUT_GET_DEVICE,
szPortName);
goto Quit;
}
Quit:
CloseHandle(hPort);
return Success;
}
int ShowSerialStatus(INT nPortNum)
{
static const LPCWSTR parity_strings[] =
{
L"None", // NOPARITY
L"Odd", // ODDPARITY
L"Even", // EVENPARITY
L"Mark", // MARKPARITY
L"Space" // SPACEPARITY
};
static const LPCWSTR control_strings[] = { L"OFF", L"ON", L"HANDSHAKE", L"TOGGLE" };
static const LPCWSTR stopbit_strings[] = { L"1", L"1.5", L"2" };
DCB dcb;
COMMTIMEOUTS CommTimeouts;
WCHAR szPortName[MAX_PORTNAME_LEN];
if (!SerialPortQuery(nPortNum, &dcb, &CommTimeouts, FALSE))
{
return 1;
}
if (dcb.Parity >= ARRAYSIZE(parity_strings))
{
ConResPrintf(StdErr, IDS_ERROR_INVALID_PARITY_BITS, dcb.Parity);
dcb.Parity = 0;
}
if (dcb.StopBits >= ARRAYSIZE(stopbit_strings))
{
ConResPrintf(StdErr, IDS_ERROR_INVALID_STOP_BITS, dcb.StopBits);
dcb.StopBits = 0;
}
swprintf(szPortName, L"COM%d", nPortNum);
ConPuts(StdOut, L"\n");
UnderlinedResPrintf(StdOut, IDS_DEVICE_STATUS_HEADER, szPortName);
ConPuts(StdOut, L"\n");
ConResPrintf(StdOut, IDS_COM_STATUS_BAUD, dcb.BaudRate);
ConResPrintf(StdOut, IDS_COM_STATUS_PARITY, parity_strings[dcb.Parity]);
ConResPrintf(StdOut, IDS_COM_STATUS_DATA_BITS, dcb.ByteSize);
ConResPrintf(StdOut, IDS_COM_STATUS_STOP_BITS, stopbit_strings[dcb.StopBits]);
ConResPrintf(StdOut, IDS_COM_STATUS_TIMEOUT,
control_strings[(CommTimeouts.ReadTotalTimeoutConstant != 0) ||
(CommTimeouts.WriteTotalTimeoutConstant != 0) ? 1 : 0]);
ConResPrintf(StdOut, IDS_COM_STATUS_XON_XOFF,
control_strings[dcb.fOutX ? 1 : 0]);
ConResPrintf(StdOut, IDS_COM_STATUS_CTS_HANDSHAKING,
control_strings[dcb.fOutxCtsFlow ? 1 : 0]);
ConResPrintf(StdOut, IDS_COM_STATUS_DSR_HANDSHAKING,
control_strings[dcb.fOutxDsrFlow ? 1 : 0]);
ConResPrintf(StdOut, IDS_COM_STATUS_DSR_SENSITIVITY,
control_strings[dcb.fDsrSensitivity ? 1 : 0]);
ConResPrintf(StdOut, IDS_COM_STATUS_DTR_CIRCUIT, control_strings[dcb.fDtrControl]);
ConResPrintf(StdOut, IDS_COM_STATUS_RTS_CIRCUIT, control_strings[dcb.fRtsControl]);
ConPuts(StdOut, L"\n");
return 0;
}
/*
* Those procedures are inspired from Wine's dll/win32/kernel32/wine/comm.c
* Copyright 1996 Erik Bos and Marcus Meissner.
*/
static PCWSTR
ParseModes(PCWSTR argStr, PBYTE Mode)
{
if (_wcsnicmp(argStr, L"OFF", 3) == 0)
{
argStr += 3;
*Mode = 0;
}
else if (_wcsnicmp(argStr, L"ON", 2) == 0)
{
argStr += 2;
*Mode = 1;
}
else if (_wcsnicmp(argStr, L"HS", 2) == 0)
{
argStr += 2;
*Mode = 2;
}
else if (_wcsnicmp(argStr, L"TG", 2) == 0)
{
argStr += 2;
*Mode = 3;
}
return NULL;
}
static PCWSTR
ParseBaudRate(PCWSTR argStr, PDWORD BaudRate)
{
argStr = ParseNumber(argStr, BaudRate);
if (!argStr) return NULL;
/*
* Check for Baud Rate abbreviations. This means that using
* those values as real baud rates is impossible using MODE.
*/
switch (*BaudRate)
{
/* BaudRate = 110, 150, 300, 600 */
case 11: case 15: case 30: case 60:
*BaudRate *= 10;
break;
/* BaudRate = 1200, 2400, 4800, 9600 */
case 12: case 24: case 48: case 96:
*BaudRate *= 100;
break;
case 19:
*BaudRate = 19200;
break;
}
return argStr;
}
static PCWSTR
ParseParity(PCWSTR argStr, PBYTE Parity)
{
switch (towupper(*argStr++))
{
case L'N':
*Parity = NOPARITY;
break;
case L'O':
*Parity = ODDPARITY;
break;
case L'E':
*Parity = EVENPARITY;
break;
case L'M':
*Parity = MARKPARITY;
break;
case L'S':
*Parity = SPACEPARITY;
break;
default:
return NULL;
}
return argStr;
}
static PCWSTR
ParseByteSize(PCWSTR argStr, PBYTE ByteSize)
{
DWORD value = 0;
argStr = ParseNumber(argStr, &value);
if (!argStr) return NULL;
*ByteSize = (BYTE)value;
if (*ByteSize < 5 || *ByteSize > 8)
return NULL;
return argStr;
}
static PCWSTR
ParseStopBits(PCWSTR argStr, PBYTE StopBits)
{
if (_wcsnicmp(argStr, L"1.5", 3) == 0)
{
argStr += 3;
*StopBits = ONE5STOPBITS;
}
else
{
if (*argStr == L'1')
*StopBits = ONESTOPBIT;
else if (*argStr == L'2')
*StopBits = TWOSTOPBITS;
else
return NULL;
argStr++;
}
return argStr;
}
/*
* Build a DCB using the old style settings string eg: "96,n,8,1"
*
* See dll/win32/kernel32/wine/comm.c!COMM_BuildOldCommDCB()
* for more information.
*/
static BOOL
BuildOldCommDCB(
OUT LPDCB pDCB,
IN PCWSTR ArgStr)
{
PCWSTR argStr = ArgStr;
BOOL stop = FALSE;
/*
* Parse the baud rate (only MANDATORY argument)
*/
argStr = ParseBaudRate(argStr, &pDCB->BaudRate);
if (!argStr) return FALSE;
/*
* Now parse the rest (OPTIONAL arguments)
*/
while (*argStr == L' ') argStr++;
if (!*argStr) goto Quit;
if (*argStr++ != L',') return FALSE;
while (*argStr == L' ') argStr++;
if (!*argStr) goto Quit;
/* Parse the parity */
// Default: EVENPARITY
pDCB->Parity = EVENPARITY;
if (*argStr != L',')
{
argStr = ParseParity(argStr, &pDCB->Parity);
if (!argStr) return FALSE;
}
while (*argStr == L' ') argStr++;
if (!*argStr) goto Quit;
if (*argStr++ != L',') return FALSE;
while (*argStr == L' ') argStr++;
if (!*argStr) goto Quit;
/* Parse the data bits */
// Default: 7
pDCB->ByteSize = 7;
if (*argStr != L',')
{
argStr = ParseByteSize(argStr, &pDCB->ByteSize);
if (!argStr) return FALSE;
}
while (*argStr == L' ') argStr++;
if (!*argStr) goto Quit;
if (*argStr++ != L',') return FALSE;
while (*argStr == L' ') argStr++;
if (!*argStr) goto Quit;
/* Parse the stop bits */
// Default: 1, or 2 for BAUD=110
// pDCB->StopBits = ONESTOPBIT;
if (*argStr != L',')
{
stop = TRUE;
argStr = ParseStopBits(argStr, &pDCB->StopBits);
if (!argStr) return FALSE;
}
/* The last parameter (flow control "retry") is really optional */
while (*argStr == L' ') argStr++;
if (!*argStr) goto Quit;
if (*argStr++ != L',') return FALSE;
while (*argStr == L' ') argStr++;
if (!*argStr) goto Quit;
Quit:
switch (towupper(*argStr))
{
case L'\0':
pDCB->fInX = FALSE;
pDCB->fOutX = FALSE;
pDCB->fOutxCtsFlow = FALSE;
pDCB->fOutxDsrFlow = FALSE;
pDCB->fDtrControl = DTR_CONTROL_ENABLE;
pDCB->fRtsControl = RTS_CONTROL_ENABLE;
break;
case L'X':
pDCB->fInX = TRUE;
pDCB->fOutX = TRUE;
pDCB->fOutxCtsFlow = FALSE;
pDCB->fOutxDsrFlow = FALSE;
pDCB->fDtrControl = DTR_CONTROL_ENABLE;
pDCB->fRtsControl = RTS_CONTROL_ENABLE;
break;
case L'P':
pDCB->fInX = FALSE;
pDCB->fOutX = FALSE;
pDCB->fOutxCtsFlow = TRUE;
pDCB->fOutxDsrFlow = TRUE;
pDCB->fDtrControl = DTR_CONTROL_HANDSHAKE;
pDCB->fRtsControl = RTS_CONTROL_HANDSHAKE;
break;
default:
/* Unsupported */
return FALSE;
}
if (*argStr) argStr++;
/* This should be the end of the string */
while (*argStr == L' ') argStr++;
if (*argStr) return FALSE;
/* If stop bits were not specified, a default is always supplied */
if (!stop)
{
if (pDCB->BaudRate == 110)
pDCB->StopBits = TWOSTOPBITS;
else
pDCB->StopBits = ONESTOPBIT;
}
return TRUE;
}
/*
* Build a DCB using the new style settings string.
* eg: "baud=9600 parity=n data=8 stop=1 xon=on to=on"
*
* See dll/win32/kernel32/wine/comm.c!COMM_BuildNewCommDCB()
* for more information.
*/
static BOOL
BuildNewCommDCB(
OUT LPDCB pDCB,
OUT LPCOMMTIMEOUTS pCommTimeouts,
IN PCWSTR ArgStr)
{
PCWSTR argStr = ArgStr;
BOOL baud = FALSE, stop = FALSE;
BYTE value;
while (argStr && *argStr)
{
while (*argStr == L' ') argStr++;
if (!*argStr) break;
if (_wcsnicmp(argStr, L"BAUD=", 5) == 0)
{
baud = TRUE;
argStr = ParseBaudRate(argStr+5, &pDCB->BaudRate);
if (!argStr) return FALSE;
}
else if (_wcsnicmp(argStr, L"PARITY=", 7) == 0)
{
// Default: EVENPARITY
argStr = ParseParity(argStr+7, &pDCB->Parity);
if (!argStr) return FALSE;
}
else if (_wcsnicmp(argStr, L"DATA=", 5) == 0)
{
// Default: 7
argStr = ParseByteSize(argStr+5, &pDCB->ByteSize);
if (!argStr) return FALSE;
}
else if (_wcsnicmp(argStr, L"STOP=", 5) == 0)
{
// Default: 1, or 2 for BAUD=110
stop = TRUE;
argStr = ParseStopBits(argStr+5, &pDCB->StopBits);
if (!argStr) return FALSE;
}
else if (_wcsnicmp(argStr, L"TO=", 3) == 0) // TO=ON|OFF
{
/* Only total time-outs are get/set by Windows' MODE.COM */
argStr = ParseModes(argStr+3, &value);
if (!argStr) return FALSE;
if (value == 0) // OFF
{
pCommTimeouts->ReadTotalTimeoutConstant = 0;
pCommTimeouts->WriteTotalTimeoutConstant = 0;
}
else if (value == 1) // ON
{
pCommTimeouts->ReadTotalTimeoutConstant = 60000;
pCommTimeouts->WriteTotalTimeoutConstant = 60000;
}
else
{
return FALSE;
}
}
else if (_wcsnicmp(argStr, L"XON=", 4) == 0) // XON=ON|OFF
{
argStr = ParseModes(argStr+4, &value);
if (!argStr) return FALSE;
if ((value == 0) || (value == 1))
{
pDCB->fOutX = value;
pDCB->fInX = value;
}
else
{
return FALSE;
}
}
else if (_wcsnicmp(argStr, L"ODSR=", 5) == 0) // ODSR=ON|OFF
{
value = 0;
argStr = ParseModes(argStr+5, &value);
if (!argStr) return FALSE;
if ((value == 0) || (value == 1))
pDCB->fOutxDsrFlow = value;
else
return FALSE;
}
else if (_wcsnicmp(argStr, L"OCTS=", 5) == 0) // OCTS=ON|OFF
{
value = 0;
argStr = ParseModes(argStr+5, &value);
if (!argStr) return FALSE;
if ((value == 0) || (value == 1))
pDCB->fOutxCtsFlow = value;
else
return FALSE;
}
else if (_wcsnicmp(argStr, L"DTR=", 4) == 0) // DTR=ON|OFF|HS
{
value = 0;
argStr = ParseModes(argStr+4, &value);
if (!argStr) return FALSE;
if ((value == 0) || (value == 1) || (value == 2))
pDCB->fDtrControl = value;
else
return FALSE;
}
else if (_wcsnicmp(argStr, L"RTS=", 4) == 0) // RTS=ON|OFF|HS|TG
{
value = 0;
argStr = ParseModes(argStr+4, &value);
if (!argStr) return FALSE;
if ((value == 0) || (value == 1) || (value == 2) || (value == 3))
pDCB->fRtsControl = value;
else
return FALSE;
}
else if (_wcsnicmp(argStr, L"IDSR=", 5) == 0) // IDSR=ON|OFF
{
value = 0;
argStr = ParseModes(argStr+5, &value);
if (!argStr) return FALSE;
if ((value == 0) || (value == 1))
pDCB->fDsrSensitivity = value;
else
return FALSE;
}
else
{
return FALSE;
}
}
/* If stop bits were not specified, a default is always supplied */
if (!stop)
{
if (baud && pDCB->BaudRate == 110)
pDCB->StopBits = TWOSTOPBITS;
else
pDCB->StopBits = ONESTOPBIT;
}
return TRUE;
}
int SetSerialState(INT nPortNum, IN PCWSTR ArgStr)
{
BOOL Success;
DCB dcb;
COMMTIMEOUTS CommTimeouts;
if (!SerialPortQuery(nPortNum, &dcb, &CommTimeouts, FALSE))
{
// TODO: Error message?
return 0;
}
/*
* Check whether we should use the old or the new MODE syntax:
* in the old syntax, the separators are both spaces and commas.
*/
if (wcschr(ArgStr, L','))
Success = BuildOldCommDCB(&dcb, ArgStr);
else
Success = BuildNewCommDCB(&dcb, &CommTimeouts, ArgStr);
if (!Success)
{
ConResPrintf(StdErr, IDS_ERROR_INVALID_PARAMETER, ArgStr);
return 1;
}
SerialPortQuery(nPortNum, &dcb, &CommTimeouts, TRUE);
ShowSerialStatus(nPortNum);
return 0;
}
/*****************************************************************************\
** E N T R Y P O I N T **
\*****************************************************************************/
static PCWSTR
FindPortNum(PCWSTR argStr, PINT PortNum)
{
PWSTR endptr = NULL;
*PortNum = wcstol(argStr, &endptr, 10);
if (endptr == argStr)
{
*PortNum = -1;
return NULL;
}
return endptr;
}
int EnumerateDevices(VOID)
{
PWSTR Buffer, ptr;
PCWSTR argStr;
DWORD dwLen = MAX_PATH;
INT nPortNum;
/* Pre-allocate a buffer for QueryDosDeviceW() */
Buffer = HeapAlloc(GetProcessHeap(), 0, dwLen * sizeof(WCHAR));
if (Buffer == NULL)
{
/* We failed, bail out */
ConPuts(StdErr, L"ERROR: Not enough memory\n");
return 0;
}
for (;;)
{
*Buffer = UNICODE_NULL;
if (QueryDosDeviceW(NULL, Buffer, dwLen))
break;
if (GetLastError() != ERROR_INSUFFICIENT_BUFFER)
{
/* We failed, bail out */
ConPrintf(StdErr, L"ERROR: QueryDosDeviceW(...) failed: 0x%lx\n", GetLastError());
HeapFree(GetProcessHeap(), 0, Buffer);
return 0;
}
/* The buffer was too small, try to re-allocate it */
dwLen *= 2;
ptr = HeapReAlloc(GetProcessHeap(), 0, Buffer, dwLen * sizeof(WCHAR));
if (ptr == NULL)
{
/* We failed, bail out */
ConPuts(StdErr, L"ERROR: Not enough memory\n");
HeapFree(GetProcessHeap(), 0, Buffer);
return 0;
}
Buffer = ptr;
}
for (ptr = Buffer; *ptr != UNICODE_NULL; ptr += wcslen(ptr) + 1)
{
if (_wcsnicmp(ptr, L"COM", 3) == 0)
{
argStr = FindPortNum(ptr+3, &nPortNum);
if (!argStr || *argStr || nPortNum == -1)
continue;
// ConResPrintf(StdOut, IDS_QUERY_SERIAL_FOUND, ptr);
ShowSerialStatus(nPortNum);
}
else if (_wcsicmp(ptr, L"PRN") == 0)
{
ConResPrintf(StdOut, IDS_QUERY_PRINTER_FOUND, ptr);
}
else if (_wcsnicmp(ptr, L"LPT", 3) == 0)
{
argStr = FindPortNum(ptr+3, &nPortNum);
if (!argStr || *argStr || nPortNum == -1)
continue;
// ConResPrintf(StdOut, IDS_QUERY_PARALLEL_FOUND, ptr);
ShowParallelStatus(nPortNum);
}
else if (_wcsicmp(ptr, L"AUX") == 0 || _wcsicmp(ptr, L"NUL") == 0)
{
ConResPrintf(StdOut, IDS_QUERY_DOSDEV_FOUND, ptr);
}
else
{
// ConResPrintf(StdOut, IDS_QUERY_MISC_FOUND, ptr);
}
}
ShowConsoleStatus();
/* Free the buffer and return success */
HeapFree(GetProcessHeap(), 0, Buffer);
return 1;
}
int wmain(int argc, WCHAR* argv[])
{
int ret = 0;
int arg;
SIZE_T ArgStrSize;
PCWSTR ArgStr, argStr;
INT nPortNum;
/* Initialize the Console Standard Streams */
ConInitStdStreams();
/*
* MODE.COM has a very peculiar way of parsing its arguments,
* as they can be even not separated by any space. This extreme
* behaviour certainly is present for backwards compatibility
* with the oldest versions of the utility present on MS-DOS.
*
* For example, such a command:
* "MODE.COM COM1baud=9600parity=ndata=8stop=1xon=onto=on"
* will be correctly understood as:
* "MODE.COM COM1 baud=9600 parity=n data=8 stop=1 xon=on to=on"
*
* Note also that the "/STATUS" switch is actually really "/STA".
*
* However we will not use GetCommandLine() because we do not want
* to deal with the prepended application path and try to find
* where the arguments start. Our approach here will consist in
* flattening the arguments vector.
*/
ArgStrSize = 0;
/* Compute the space needed for the new string, and allocate it */
for (arg = 1; arg < argc; arg++)
{
ArgStrSize += wcslen(argv[arg]) + 1; // 1 for space
}
ArgStr = HeapAlloc(GetProcessHeap(), 0, (ArgStrSize + 1) * sizeof(WCHAR));
if (ArgStr == NULL)
{
ConPuts(StdErr, L"ERROR: Not enough memory\n");
return 1;
}
/* Copy the contents and NULL-terminate the string */
argStr = ArgStr;
for (arg = 1; arg < argc; arg++)
{
wcscpy((PWSTR)argStr, argv[arg]);
argStr += wcslen(argv[arg]);
*(PWSTR)argStr++ = L' ';
}
*(PWSTR)argStr = L'\0';
/* Parse the command line */
argStr = ArgStr;
while (*argStr == L' ') argStr++;
if (!*argStr) goto show_status;
if (wcsstr(argStr, L"/?") || wcsstr(argStr, L"-?"))
{
ConResPuts(StdOut, IDS_USAGE);
goto Quit;
}
else if (_wcsnicmp(argStr, L"/STA", 4) == 0)
{
/* Skip this parameter */
while (*argStr != L' ') argStr++;
/* Skip any delimiter */
while (*argStr == L' ') argStr++;
/* The presence of any other parameter is invalid */
if (*argStr)
goto invalid_parameter;
goto show_status;
}
else if (_wcsnicmp(argStr, L"LPT", 3) == 0)
{
argStr = FindPortNum(argStr+3, &nPortNum);
if (!argStr || nPortNum == -1)
goto invalid_parameter;
if (*argStr == L':') argStr++;
while (*argStr == L' ') argStr++;
if (!*argStr || _wcsnicmp(argStr, L"/STA", 4) == 0)
ret = ShowParallelStatus(nPortNum);
else
ConPuts(StdErr, L"ERROR: LPT port redirection is not implemented!\n");
// TODO: Implement setting LPT port redirection using SetParallelState().
goto Quit;
}
else if (_wcsnicmp(argStr, L"COM", 3) == 0)
{
argStr = FindPortNum(argStr+3, &nPortNum);
if (!argStr || nPortNum == -1)
goto invalid_parameter;
if (*argStr == L':') argStr++;
while (*argStr == L' ') argStr++;
if (!*argStr || _wcsnicmp(argStr, L"/STA", 4) == 0)
ret = ShowSerialStatus(nPortNum);
else
ret = SetSerialState(nPortNum, argStr);
goto Quit;
}
else if (_wcsnicmp(argStr, L"CON", 3) == 0)
{
argStr += 3;
if (*argStr == L':') argStr++;
while (*argStr == L' ') argStr++;
if (!*argStr || _wcsnicmp(argStr, L"/STA", 4) == 0)
{
ret = ShowConsoleStatus();
}
else if ( (_wcsnicmp(argStr, L"CP", 2) == 0 && (argStr += 2)) ||
(_wcsnicmp(argStr, L"CODEPAGE", 8) == 0 && (argStr += 8)) )
{
while (*argStr == L' ') argStr++;
if (!*argStr || _wcsnicmp(argStr, L"/STA", 4) == 0)
ret = ShowConsoleCPStatus();
else
ret = SetConsoleCPState(argStr);
}
else
{
ret = SetConsoleState(argStr);
}
goto Quit;
}
// else if (wcschr(argStr, L','))
else
{
/* Old syntax: MODE [COLS],[LINES] */
ret = SetConsoleStateOld(argStr);
goto Quit;
}
show_status:
EnumerateDevices();
goto Quit;
invalid_parameter:
ConResPrintf(StdErr, IDS_ERROR_INVALID_PARAMETER, ArgStr);
goto Quit;
Quit:
/* Free the string and quit */
HeapFree(GetProcessHeap(), 0, (PWSTR)ArgStr);
return ret;
}