2009-03-16 18:18:26 +00:00
|
|
|
/*
|
2020-07-26 18:29:34 +00:00
|
|
|
* ASSOC.C - assoc internal command.
|
2009-03-16 18:18:26 +00:00
|
|
|
*
|
|
|
|
*
|
|
|
|
* History:
|
|
|
|
*
|
|
|
|
* 14-Mar-2009 Lee C. Baker
|
2020-07-26 18:29:34 +00:00
|
|
|
* - initial implementation.
|
2009-03-16 18:18:26 +00:00
|
|
|
*
|
|
|
|
* 15-Mar-2009 Lee C. Baker
|
2020-07-26 18:29:34 +00:00
|
|
|
* - Don't write to (or use) HKEY_CLASSES_ROOT directly.
|
|
|
|
* - Externalize strings.
|
2009-03-16 18:18:26 +00:00
|
|
|
*
|
|
|
|
* TODO:
|
2020-07-26 18:29:34 +00:00
|
|
|
* - PrintAllAssociations could be optimized to not fetch all registry subkeys under 'Classes', just the ones that start with '.'
|
2009-03-16 18:18:26 +00:00
|
|
|
*/
|
|
|
|
|
2013-01-24 23:00:42 +00:00
|
|
|
#include "precomp.h"
|
2009-03-16 18:18:26 +00:00
|
|
|
|
|
|
|
#ifdef INCLUDE_CMD_ASSOC
|
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
static LONG
|
|
|
|
PrintAssociationEx(
|
|
|
|
IN HKEY hKeyClasses,
|
|
|
|
IN PCTSTR pszExtension)
|
2009-03-16 18:18:26 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
LONG lRet;
|
|
|
|
HKEY hKey;
|
|
|
|
DWORD dwFileTypeLen = 0;
|
|
|
|
PTSTR pszFileType;
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
lRet = RegOpenKeyEx(hKeyClasses, pszExtension, 0, KEY_QUERY_VALUE, &hKey);
|
2020-07-26 18:29:34 +00:00
|
|
|
if (lRet != ERROR_SUCCESS)
|
2020-07-27 23:10:58 +00:00
|
|
|
{
|
|
|
|
if (lRet != ERROR_FILE_NOT_FOUND)
|
|
|
|
ErrorMessage(lRet, NULL);
|
|
|
|
return lRet;
|
|
|
|
}
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
/* Obtain the string length */
|
|
|
|
lRet = RegQueryValueEx(hKey, NULL, NULL, NULL, NULL, &dwFileTypeLen);
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
/* If there is no default value, don't display it */
|
2020-07-26 18:29:34 +00:00
|
|
|
if (lRet == ERROR_FILE_NOT_FOUND)
|
2013-06-30 00:08:43 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
RegCloseKey(hKey);
|
|
|
|
return lRet;
|
2013-06-30 00:08:43 +00:00
|
|
|
}
|
2020-07-26 18:29:34 +00:00
|
|
|
if (lRet != ERROR_SUCCESS)
|
2013-06-30 00:08:43 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
ErrorMessage(lRet, NULL);
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
return lRet;
|
2013-06-30 00:08:43 +00:00
|
|
|
}
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
++dwFileTypeLen;
|
|
|
|
pszFileType = cmd_alloc(dwFileTypeLen * sizeof(TCHAR));
|
|
|
|
if (!pszFileType)
|
2017-12-03 17:49:41 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
WARN("Cannot allocate memory for pszFileType!\n");
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
return ERROR_NOT_ENOUGH_MEMORY;
|
2017-12-03 17:49:41 +00:00
|
|
|
}
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
/* Obtain the actual file type */
|
|
|
|
lRet = RegQueryValueEx(hKey, NULL, NULL, NULL, (LPBYTE)pszFileType, &dwFileTypeLen);
|
|
|
|
RegCloseKey(hKey);
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-26 18:29:34 +00:00
|
|
|
if (lRet != ERROR_SUCCESS)
|
2013-06-30 00:08:43 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
ErrorMessage(lRet, NULL);
|
|
|
|
cmd_free(pszFileType);
|
|
|
|
return lRet;
|
2013-06-30 00:08:43 +00:00
|
|
|
}
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
/* If there is a default key, display the relevant information */
|
|
|
|
if (dwFileTypeLen != 0)
|
2013-06-30 00:08:43 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
ConOutPrintf(_T("%s=%s\n"), pszExtension, pszFileType);
|
2013-06-30 00:08:43 +00:00
|
|
|
}
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
cmd_free(pszFileType);
|
|
|
|
return ERROR_SUCCESS;
|
2009-03-16 18:18:26 +00:00
|
|
|
}
|
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
static LONG
|
|
|
|
PrintAssociation(
|
|
|
|
IN PCTSTR pszExtension)
|
2009-03-16 18:18:26 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
LONG lRet;
|
|
|
|
HKEY hKeyClasses;
|
|
|
|
|
|
|
|
lRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE, _T("SOFTWARE\\Classes"), 0,
|
|
|
|
KEY_ENUMERATE_SUB_KEYS, &hKeyClasses);
|
|
|
|
if (lRet != ERROR_SUCCESS)
|
|
|
|
{
|
|
|
|
ErrorMessage(lRet, NULL);
|
|
|
|
return lRet;
|
|
|
|
}
|
|
|
|
|
|
|
|
lRet = PrintAssociationEx(hKeyClasses, pszExtension);
|
2013-06-30 00:08:43 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
RegCloseKey(hKeyClasses);
|
|
|
|
return lRet;
|
|
|
|
}
|
2013-06-30 00:08:43 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
static LONG
|
|
|
|
PrintAllAssociations(VOID)
|
|
|
|
{
|
|
|
|
LONG lRet;
|
|
|
|
HKEY hKeyClasses;
|
|
|
|
DWORD dwKeyCtr;
|
|
|
|
DWORD dwNumKeys = 0;
|
|
|
|
DWORD dwExtLen = 0;
|
|
|
|
PTSTR pszExtName;
|
|
|
|
|
|
|
|
lRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE, _T("SOFTWARE\\Classes"), 0,
|
|
|
|
KEY_QUERY_VALUE | KEY_ENUMERATE_SUB_KEYS, &hKeyClasses);
|
2020-07-26 18:29:34 +00:00
|
|
|
if (lRet != ERROR_SUCCESS)
|
2020-07-27 23:10:58 +00:00
|
|
|
{
|
|
|
|
ErrorMessage(lRet, NULL);
|
|
|
|
return lRet;
|
|
|
|
}
|
2013-06-30 00:08:43 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
lRet = RegQueryInfoKey(hKeyClasses, NULL, NULL, NULL, &dwNumKeys, &dwExtLen,
|
2020-07-26 18:29:34 +00:00
|
|
|
NULL, NULL, NULL, NULL, NULL, NULL);
|
|
|
|
if (lRet != ERROR_SUCCESS)
|
2013-06-30 00:08:43 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
ErrorMessage(lRet, NULL);
|
|
|
|
RegCloseKey(hKeyClasses);
|
|
|
|
return lRet;
|
2013-06-30 00:08:43 +00:00
|
|
|
}
|
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
++dwExtLen;
|
|
|
|
pszExtName = cmd_alloc(dwExtLen * sizeof(TCHAR));
|
|
|
|
if (!pszExtName)
|
2017-12-03 17:49:41 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
WARN("Cannot allocate memory for pszExtName!\n");
|
|
|
|
RegCloseKey(hKeyClasses);
|
|
|
|
return ERROR_NOT_ENOUGH_MEMORY;
|
2017-12-03 17:49:41 +00:00
|
|
|
}
|
2013-06-30 00:08:43 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
for (dwKeyCtr = 0; dwKeyCtr < dwNumKeys; ++dwKeyCtr)
|
2013-06-30 00:08:43 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
DWORD dwBufSize = dwExtLen;
|
|
|
|
lRet = RegEnumKeyEx(hKeyClasses, dwKeyCtr, pszExtName, &dwBufSize,
|
2020-07-26 18:29:34 +00:00
|
|
|
NULL, NULL, NULL, NULL);
|
2013-06-30 00:08:43 +00:00
|
|
|
|
2020-07-26 18:29:34 +00:00
|
|
|
if (lRet == ERROR_SUCCESS || lRet == ERROR_MORE_DATA)
|
2013-06-30 00:08:43 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
/* Name starts with '.': this is an extension */
|
|
|
|
if (*pszExtName == _T('.'))
|
|
|
|
PrintAssociationEx(hKeyClasses, pszExtName);
|
2013-06-30 00:08:43 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
ErrorMessage(lRet, NULL);
|
|
|
|
cmd_free(pszExtName);
|
|
|
|
RegCloseKey(hKeyClasses);
|
|
|
|
return lRet;
|
2013-06-30 00:08:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
RegCloseKey(hKeyClasses);
|
2013-06-30 00:08:43 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
cmd_free(pszExtName);
|
|
|
|
return ERROR_SUCCESS;
|
2009-03-16 18:18:26 +00:00
|
|
|
}
|
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
static LONG
|
2020-07-26 18:29:34 +00:00
|
|
|
AddAssociation(
|
2020-07-27 23:10:58 +00:00
|
|
|
IN PCTSTR pszExtension,
|
|
|
|
IN PCTSTR pszType)
|
2009-03-16 18:18:26 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
LONG lRet;
|
|
|
|
HKEY hKeyClasses, hKey;
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
lRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE, _T("SOFTWARE\\Classes"), 0,
|
|
|
|
KEY_CREATE_SUB_KEY, &hKeyClasses);
|
2020-07-26 18:29:34 +00:00
|
|
|
if (lRet != ERROR_SUCCESS)
|
2020-07-27 23:10:58 +00:00
|
|
|
{
|
|
|
|
ErrorMessage(lRet, NULL);
|
|
|
|
return lRet;
|
|
|
|
}
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
lRet = RegCreateKeyEx(hKeyClasses, pszExtension, 0, NULL, REG_OPTION_NON_VOLATILE,
|
|
|
|
KEY_SET_VALUE, NULL, &hKey, NULL);
|
|
|
|
RegCloseKey(hKeyClasses);
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-26 18:29:34 +00:00
|
|
|
if (lRet != ERROR_SUCCESS)
|
2020-07-27 23:10:58 +00:00
|
|
|
{
|
|
|
|
ErrorMessage(lRet, NULL);
|
|
|
|
return lRet;
|
|
|
|
}
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
lRet = RegSetValueEx(hKey, NULL, 0, REG_SZ,
|
|
|
|
(LPBYTE)pszType, (DWORD)(_tcslen(pszType) + 1) * sizeof(TCHAR));
|
|
|
|
RegCloseKey(hKey);
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-26 18:29:34 +00:00
|
|
|
if (lRet != ERROR_SUCCESS)
|
2020-07-27 23:10:58 +00:00
|
|
|
{
|
|
|
|
ErrorMessage(lRet, NULL);
|
|
|
|
return lRet;
|
|
|
|
}
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
return ERROR_SUCCESS;
|
2009-03-16 18:18:26 +00:00
|
|
|
}
|
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
static LONG
|
2020-07-26 18:29:34 +00:00
|
|
|
RemoveAssociation(
|
2020-07-27 23:10:58 +00:00
|
|
|
IN PCTSTR pszExtension)
|
2009-03-16 18:18:26 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
LONG lRet;
|
|
|
|
HKEY hKeyClasses;
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
lRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE, _T("SOFTWARE\\Classes"), 0,
|
|
|
|
KEY_QUERY_VALUE, &hKeyClasses);
|
2020-07-26 18:29:34 +00:00
|
|
|
if (lRet != ERROR_SUCCESS)
|
2020-07-27 23:10:58 +00:00
|
|
|
{
|
|
|
|
ErrorMessage(lRet, NULL);
|
|
|
|
return lRet;
|
|
|
|
}
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
lRet = RegDeleteKey(hKeyClasses, pszExtension);
|
|
|
|
RegCloseKey(hKeyClasses);
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-26 18:29:34 +00:00
|
|
|
if (lRet != ERROR_SUCCESS)
|
2020-07-27 23:10:58 +00:00
|
|
|
{
|
|
|
|
if (lRet != ERROR_FILE_NOT_FOUND)
|
|
|
|
ErrorMessage(lRet, NULL);
|
|
|
|
return lRet;
|
|
|
|
}
|
2009-03-16 18:18:26 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
return ERROR_SUCCESS;
|
2009-03-16 18:18:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-07-26 18:29:34 +00:00
|
|
|
INT CommandAssoc(LPTSTR param)
|
2009-03-16 18:18:26 +00:00
|
|
|
{
|
[CMD] Change ERRORLEVEL behaviour for commands ASSOC, PATH, PROMPT and SET.
Commands APPEND/DPATH and FTYPE are also concerned by this; however
we do not implement them in our CMD.EXE yet.
These commands set the ERRORLEVEL differently, whether or not they are
run manually from the command-line/from a .BAT file, or from a .CMD file:
- From command-line/.BAT file, these commands set the ERRORLEVEL only if
an error occurs. So, if two commands are run consecutively and the first
one fails, the ERRORLEVEL will remain set even if the second command
succeeds.
- However, when being run from a .CMD file, these command will always
set the ERRORLEVEL. In the example case described above, the second
command that succeeds will reset the ERRORLEVEL to 0.
This behaviour is determined from the top-level batch/script file being
run. This means that, if a .BAT file is first started, then starts a
.CMD file, the commands will still behave the .BAT way; on the opposite,
if a .CMD file is first started, then starts a .BAT file, these commands
will still behave the .CMD way.
To implement this we introduce one global BATCH_TYPE enum variable that
is initialized to the corresponding batch/script file type when the
top-level script is loaded. It is reset to "none" when that script
terminates.
See https://ss64.com/nt/errorlevel.html for more details,
section "Old style .bat Batch files vs .cmd Batch scripts",
and https://groups.google.com/forum/#!msg/microsoft.public.win2000.cmdprompt.admin/XHeUq8oe2wk/LIEViGNmkK0J
(comment by Mark Zbikowski).
2020-07-01 00:15:52 +00:00
|
|
|
INT retval = 0;
|
2020-07-27 23:10:58 +00:00
|
|
|
PTCHAR pEqualSign;
|
2020-07-26 18:29:34 +00:00
|
|
|
|
|
|
|
/* Print help */
|
|
|
|
if (!_tcsncmp(param, _T("/?"), 2))
|
2013-06-30 00:08:43 +00:00
|
|
|
{
|
2020-07-26 18:29:34 +00:00
|
|
|
ConOutResPaging(TRUE, STRING_ASSOC_HELP);
|
2013-06-30 00:08:43 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
/* Print all associations if no parameter has been specified */
|
|
|
|
if (!*param)
|
2017-12-03 17:49:41 +00:00
|
|
|
{
|
2013-06-30 00:08:43 +00:00
|
|
|
PrintAllAssociations();
|
[CMD] Change ERRORLEVEL behaviour for commands ASSOC, PATH, PROMPT and SET.
Commands APPEND/DPATH and FTYPE are also concerned by this; however
we do not implement them in our CMD.EXE yet.
These commands set the ERRORLEVEL differently, whether or not they are
run manually from the command-line/from a .BAT file, or from a .CMD file:
- From command-line/.BAT file, these commands set the ERRORLEVEL only if
an error occurs. So, if two commands are run consecutively and the first
one fails, the ERRORLEVEL will remain set even if the second command
succeeds.
- However, when being run from a .CMD file, these command will always
set the ERRORLEVEL. In the example case described above, the second
command that succeeds will reset the ERRORLEVEL to 0.
This behaviour is determined from the top-level batch/script file being
run. This means that, if a .BAT file is first started, then starts a
.CMD file, the commands will still behave the .BAT way; on the opposite,
if a .CMD file is first started, then starts a .BAT file, these commands
will still behave the .CMD way.
To implement this we introduce one global BATCH_TYPE enum variable that
is initialized to the corresponding batch/script file type when the
top-level script is loaded. It is reset to "none" when that script
terminates.
See https://ss64.com/nt/errorlevel.html for more details,
section "Old style .bat Batch files vs .cmd Batch scripts",
and https://groups.google.com/forum/#!msg/microsoft.public.win2000.cmdprompt.admin/XHeUq8oe2wk/LIEViGNmkK0J
(comment by Mark Zbikowski).
2020-07-01 00:15:52 +00:00
|
|
|
goto Quit;
|
2017-12-03 17:49:41 +00:00
|
|
|
}
|
2020-07-26 18:29:34 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
pEqualSign = _tcschr(param, _T('='));
|
|
|
|
if (pEqualSign != NULL)
|
2013-06-30 00:08:43 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
PTSTR pszFileType = pEqualSign + 1;
|
2020-07-26 18:29:34 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
/* NULL-terminate at the equals sign */
|
|
|
|
*pEqualSign = 0;
|
2020-07-26 18:29:34 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
/* If the equals sign is the last character
|
|
|
|
* in the string, delete the association. */
|
|
|
|
if (*pszFileType == 0)
|
2013-06-30 00:08:43 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
retval = RemoveAssociation(param);
|
2013-06-30 00:08:43 +00:00
|
|
|
}
|
|
|
|
else
|
2020-07-27 23:10:58 +00:00
|
|
|
/* Otherwise, add the association and print it out */
|
2013-06-30 00:08:43 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
retval = AddAssociation(param, pszFileType);
|
|
|
|
PrintAssociation(param);
|
2013-06-30 00:08:43 +00:00
|
|
|
}
|
2020-07-26 18:29:34 +00:00
|
|
|
|
2020-07-27 23:10:58 +00:00
|
|
|
if (retval != ERROR_SUCCESS)
|
|
|
|
{
|
|
|
|
if (retval != ERROR_FILE_NOT_FOUND)
|
|
|
|
{
|
2024-07-11 00:12:12 +00:00
|
|
|
ConErrResPrintf(STRING_ERROR_WHILE_PROCESSING, param);
|
2020-07-27 23:10:58 +00:00
|
|
|
}
|
|
|
|
// retval = 1; /* Fixup the error value */
|
|
|
|
}
|
2020-07-26 18:29:34 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
/* No equals sign, print the association */
|
[CMD] Change ERRORLEVEL behaviour for commands ASSOC, PATH, PROMPT and SET.
Commands APPEND/DPATH and FTYPE are also concerned by this; however
we do not implement them in our CMD.EXE yet.
These commands set the ERRORLEVEL differently, whether or not they are
run manually from the command-line/from a .BAT file, or from a .CMD file:
- From command-line/.BAT file, these commands set the ERRORLEVEL only if
an error occurs. So, if two commands are run consecutively and the first
one fails, the ERRORLEVEL will remain set even if the second command
succeeds.
- However, when being run from a .CMD file, these command will always
set the ERRORLEVEL. In the example case described above, the second
command that succeeds will reset the ERRORLEVEL to 0.
This behaviour is determined from the top-level batch/script file being
run. This means that, if a .BAT file is first started, then starts a
.CMD file, the commands will still behave the .BAT way; on the opposite,
if a .CMD file is first started, then starts a .BAT file, these commands
will still behave the .CMD way.
To implement this we introduce one global BATCH_TYPE enum variable that
is initialized to the corresponding batch/script file type when the
top-level script is loaded. It is reset to "none" when that script
terminates.
See https://ss64.com/nt/errorlevel.html for more details,
section "Old style .bat Batch files vs .cmd Batch scripts",
and https://groups.google.com/forum/#!msg/microsoft.public.win2000.cmdprompt.admin/XHeUq8oe2wk/LIEViGNmkK0J
(comment by Mark Zbikowski).
2020-07-01 00:15:52 +00:00
|
|
|
retval = PrintAssociation(param);
|
2020-07-27 23:10:58 +00:00
|
|
|
if (retval != ERROR_SUCCESS)
|
[CMD] Change ERRORLEVEL behaviour for commands ASSOC, PATH, PROMPT and SET.
Commands APPEND/DPATH and FTYPE are also concerned by this; however
we do not implement them in our CMD.EXE yet.
These commands set the ERRORLEVEL differently, whether or not they are
run manually from the command-line/from a .BAT file, or from a .CMD file:
- From command-line/.BAT file, these commands set the ERRORLEVEL only if
an error occurs. So, if two commands are run consecutively and the first
one fails, the ERRORLEVEL will remain set even if the second command
succeeds.
- However, when being run from a .CMD file, these command will always
set the ERRORLEVEL. In the example case described above, the second
command that succeeds will reset the ERRORLEVEL to 0.
This behaviour is determined from the top-level batch/script file being
run. This means that, if a .BAT file is first started, then starts a
.CMD file, the commands will still behave the .BAT way; on the opposite,
if a .CMD file is first started, then starts a .BAT file, these commands
will still behave the .CMD way.
To implement this we introduce one global BATCH_TYPE enum variable that
is initialized to the corresponding batch/script file type when the
top-level script is loaded. It is reset to "none" when that script
terminates.
See https://ss64.com/nt/errorlevel.html for more details,
section "Old style .bat Batch files vs .cmd Batch scripts",
and https://groups.google.com/forum/#!msg/microsoft.public.win2000.cmdprompt.admin/XHeUq8oe2wk/LIEViGNmkK0J
(comment by Mark Zbikowski).
2020-07-01 00:15:52 +00:00
|
|
|
{
|
2020-07-27 23:10:58 +00:00
|
|
|
ConErrResPrintf(STRING_ASSOC_ERROR, param);
|
[CMD] Change ERRORLEVEL behaviour for commands ASSOC, PATH, PROMPT and SET.
Commands APPEND/DPATH and FTYPE are also concerned by this; however
we do not implement them in our CMD.EXE yet.
These commands set the ERRORLEVEL differently, whether or not they are
run manually from the command-line/from a .BAT file, or from a .CMD file:
- From command-line/.BAT file, these commands set the ERRORLEVEL only if
an error occurs. So, if two commands are run consecutively and the first
one fails, the ERRORLEVEL will remain set even if the second command
succeeds.
- However, when being run from a .CMD file, these command will always
set the ERRORLEVEL. In the example case described above, the second
command that succeeds will reset the ERRORLEVEL to 0.
This behaviour is determined from the top-level batch/script file being
run. This means that, if a .BAT file is first started, then starts a
.CMD file, the commands will still behave the .BAT way; on the opposite,
if a .CMD file is first started, then starts a .BAT file, these commands
will still behave the .CMD way.
To implement this we introduce one global BATCH_TYPE enum variable that
is initialized to the corresponding batch/script file type when the
top-level script is loaded. It is reset to "none" when that script
terminates.
See https://ss64.com/nt/errorlevel.html for more details,
section "Old style .bat Batch files vs .cmd Batch scripts",
and https://groups.google.com/forum/#!msg/microsoft.public.win2000.cmdprompt.admin/XHeUq8oe2wk/LIEViGNmkK0J
(comment by Mark Zbikowski).
2020-07-01 00:15:52 +00:00
|
|
|
retval = 1; /* Fixup the error value */
|
|
|
|
}
|
2013-06-30 00:08:43 +00:00
|
|
|
}
|
|
|
|
|
[CMD] Change ERRORLEVEL behaviour for commands ASSOC, PATH, PROMPT and SET.
Commands APPEND/DPATH and FTYPE are also concerned by this; however
we do not implement them in our CMD.EXE yet.
These commands set the ERRORLEVEL differently, whether or not they are
run manually from the command-line/from a .BAT file, or from a .CMD file:
- From command-line/.BAT file, these commands set the ERRORLEVEL only if
an error occurs. So, if two commands are run consecutively and the first
one fails, the ERRORLEVEL will remain set even if the second command
succeeds.
- However, when being run from a .CMD file, these command will always
set the ERRORLEVEL. In the example case described above, the second
command that succeeds will reset the ERRORLEVEL to 0.
This behaviour is determined from the top-level batch/script file being
run. This means that, if a .BAT file is first started, then starts a
.CMD file, the commands will still behave the .BAT way; on the opposite,
if a .CMD file is first started, then starts a .BAT file, these commands
will still behave the .CMD way.
To implement this we introduce one global BATCH_TYPE enum variable that
is initialized to the corresponding batch/script file type when the
top-level script is loaded. It is reset to "none" when that script
terminates.
See https://ss64.com/nt/errorlevel.html for more details,
section "Old style .bat Batch files vs .cmd Batch scripts",
and https://groups.google.com/forum/#!msg/microsoft.public.win2000.cmdprompt.admin/XHeUq8oe2wk/LIEViGNmkK0J
(comment by Mark Zbikowski).
2020-07-01 00:15:52 +00:00
|
|
|
Quit:
|
|
|
|
if (BatType != CMD_TYPE)
|
|
|
|
{
|
|
|
|
if (retval != 0)
|
|
|
|
nErrorLevel = retval;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nErrorLevel = retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
return retval;
|
2009-03-16 18:18:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* INCLUDE_CMD_ASSOC */
|