[CMD] GOTO: Fix label parsing.

We note two things, when CMD searches for the corresponding label in the
batch file:
- the first character of the line is always ignored, unless it's a colon;
- the escape caret ^ is supported and interpreted.

Fixes some cmd_winetests.
This commit is contained in:
Hermès Bélusca-Maïto 2020-07-22 01:16:53 +02:00
parent 71cd64d66a
commit 2e4c8c019e
No known key found for this signature in database
GPG key ID: 3B2539C65E7B93D0
4 changed files with 74 additions and 34 deletions

View file

@ -131,7 +131,7 @@ static LPTSTR BatchParams(LPTSTR s1, LPTSTR s2)
BOOL inquotes = FALSE;
/* Find next parameter */
while (_istspace(*s2) || (*s2 && _tcschr(_T(",;="), *s2)))
while (_istspace(*s2) || (*s2 && _tcschr(STANDARD_SEPS, *s2)))
s2++;
if (!*s2)
break;
@ -139,7 +139,7 @@ static LPTSTR BatchParams(LPTSTR s1, LPTSTR s2)
/* Copy it */
do
{
if (!inquotes && (_istspace(*s2) || _tcschr(_T(",;="), *s2)))
if (!inquotes && (_istspace(*s2) || _tcschr(STANDARD_SEPS, *s2)))
break;
inquotes ^= (*s2 == _T('"'));
*s1++ = *s2++;

View file

@ -301,6 +301,10 @@ INT cmd_move (LPTSTR);
INT CommandMsgbox (LPTSTR);
/* Prototypes from PARSER.C */
/* These three characters act like spaces to the parser in most contexts */
#define STANDARD_SEPS _T(",;=")
enum { C_COMMAND, C_QUIET, C_BLOCK, C_MULTI, C_OR, C_AND, C_PIPE, C_IF, C_FOR };
typedef struct _PARSED_COMMAND
{

View file

@ -35,7 +35,7 @@
*/
INT cmd_goto(LPTSTR param)
{
LPTSTR tmp, tmp2;
LPTSTR label, tmp;
DWORD dwCurrPos;
BOOL bRetry;
@ -59,11 +59,9 @@ INT cmd_goto(LPTSTR param)
return 1;
}
/* Terminate label at first space char */
tmp = param + 1;
while (!_istcntrl(*tmp) && !_istspace(*tmp) && (*tmp != _T(':')))
++tmp;
*tmp = _T('\0');
/* Strip leading whitespace */
while (_istspace(*param))
++param;
/* Support jumping to the end of the file, only if extensions are enabled */
if (bEnableExtensions &&
@ -78,6 +76,23 @@ INT cmd_goto(LPTSTR param)
return 0;
}
/* Skip the first colon or plus sign */
if (*param == _T(':') || *param == _T('+'))
++param;
/* Terminate the label at the first delimiter character */
tmp = param;
while (!_istcntrl(*tmp) && !_istspace(*tmp) &&
!_tcschr(_T(":+"), *tmp) && !_tcschr(STANDARD_SEPS, *tmp) /* &&
!_tcschr(_T("&|<>"), *tmp) */)
{
++tmp;
}
*tmp = _T('\0');
/* If we don't have any label, bail out */
if (!*param)
goto NotFound;
/*
* Search the next label starting our position, until the end of the file.
* If none has been found, restart at the beginning of the file, and continue
@ -89,36 +104,59 @@ INT cmd_goto(LPTSTR param)
retry:
while (BatchGetString(textline, ARRAYSIZE(textline)))
{
INT pos;
INT_PTR size;
if (bRetry && (bc->mempos >= dwCurrPos))
break;
/* Strip out any trailing spaces or control chars */
tmp = textline + _tcslen(textline) - 1;
while (tmp > textline && (_istcntrl(*tmp) || _istspace(*tmp) || (*tmp == _T(':'))))
--tmp;
*(tmp + 1) = _T('\0');
#if 0
/* If this is not a label, continue searching */
if (!_tcschr(textline, _T(':')))
continue;
#endif
/* Then leading spaces... */
tmp = textline;
while (_istspace(*tmp))
++tmp;
label = textline;
/* All space after leading space terminate the string */
size = _tcslen(tmp) - 1;
pos = 0;
while (tmp + pos < tmp + size)
/* A bug in Windows' CMD makes it always ignore the
* first character of the line, unless it's a colon. */
if (*label != _T(':'))
++label;
/* Strip any leading whitespace */
while (_istspace(*label))
++label;
/* If this is not a label, continue searching */
if (*label != _T(':'))
continue;
/* Skip the first colon or plus sign */
#if 0
if (*label == _T(':') || *label == _T('+'))
++label;
#endif
++label;
/* Strip any whitespace between the colon and the label */
while (_istspace(*label))
++label;
/* Terminate the label at the first delimiter character */
tmp = label;
while (!_istcntrl(*tmp) && !_istspace(*tmp) &&
!_tcschr(_T(":+"), *tmp) && !_tcschr(STANDARD_SEPS, *tmp) &&
!_tcschr(_T("&|<>"), *tmp))
{
if (_istspace(tmp[pos]))
tmp[pos]=_T('\0');
++pos;
}
/* Support the escape caret */
if (*tmp == _T('^'))
{
/* Move the buffer back one character */
memmove(tmp, tmp + 1, (_tcslen(tmp + 1) + 1) * sizeof(TCHAR));
/* We will ignore the new character */
}
tmp2 = param;
/* Use whole label name */
if ((*tmp == _T(':')) && ((_tcsicmp(++tmp, param) == 0) || (_tcsicmp(tmp, ++tmp2) == 0)))
++tmp;
}
*tmp = _T('\0');
/* Jump if the labels are identical */
if (_tcsicmp(label, param) == 0)
{
/* Do not process any more parts of a compound command */
bc->current = NULL;
@ -132,6 +170,7 @@ retry:
goto retry;
}
NotFound:
ConErrResPrintf(STRING_GOTO_ERROR2, param);
ExitBatch();
return 1;

View file

@ -39,9 +39,6 @@ static const TCHAR *const IfOperatorString[] =
#define IF_MAX_COMPARISON IF_NEQ
};
/* These three characters act like spaces to the parser in most contexts */
#define STANDARD_SEPS _T(",;=")
static BOOL IsSeparator(TCHAR Char)
{
return _istspace(Char) || (Char && _tcschr(STANDARD_SEPS, Char));