mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
3ade7829d0
* Only expand arguments actually containing wildcard characters (to prevent expanding directory names to all files within that directory) svn path=/trunk/; revision=4018
468 lines
9.8 KiB
C
468 lines
9.8 KiB
C
/*
|
|
* INTERNAL.C - command.com internal commands.
|
|
*
|
|
*
|
|
* History:
|
|
*
|
|
* 17/08/94 (Tim Norman)
|
|
* started.
|
|
*
|
|
* 08/08/95 (Matt Rains)
|
|
* i have cleaned up the source code. changes now bring this source into
|
|
* guidelines for recommended programming practice.
|
|
*
|
|
* cd()
|
|
* started.
|
|
*
|
|
* dir()
|
|
* i have added support for file attributes to the DIR() function. the
|
|
* routine adds "d" (directory) and "r" (read only) output. files with the
|
|
* system attribute have the filename converted to lowercase. files with
|
|
* the hidden attribute are not displayed.
|
|
*
|
|
* i have added support for directorys. now if the directory attribute is
|
|
* detected the file size if replaced with the string "<dir>".
|
|
*
|
|
* ver()
|
|
* started.
|
|
*
|
|
* md()
|
|
* started.
|
|
*
|
|
* rd()
|
|
* started.
|
|
*
|
|
* del()
|
|
* started.
|
|
*
|
|
* does not support wildcard selection.
|
|
*
|
|
* todo: add delete directory support.
|
|
* add recursive directory delete support.
|
|
*
|
|
* ren()
|
|
* started.
|
|
*
|
|
* does not support wildcard selection.
|
|
*
|
|
* todo: add rename directory support.
|
|
*
|
|
* a general structure has been used for the cd, rd and md commands. this
|
|
* will be better in the long run. it is too hard to maintain such diverse
|
|
* functions when you are involved in a group project like this.
|
|
*
|
|
* 12/14/95 (Tim Norman)
|
|
* fixed DIR so that it will stick \*.* if a directory is specified and
|
|
* that it will stick on .* if a file with no extension is specified or
|
|
* *.* if it ends in a \
|
|
*
|
|
* 1/6/96 (Tim Norman)
|
|
* added an isatty call to DIR so it won't prompt for keypresses unless
|
|
* stdin and stdout are the console.
|
|
*
|
|
* changed parameters to be mutually consistent to make calling the
|
|
* functions easier
|
|
*
|
|
* rem()
|
|
* started.
|
|
*
|
|
* doskey()
|
|
* started.
|
|
*
|
|
* 01/22/96 (Oliver Mueller)
|
|
* error messages are now handled by perror.
|
|
*
|
|
* 02/05/96 (Tim Norman)
|
|
* converted all functions to accept first/rest parameters
|
|
*
|
|
* 07/26/96 (Tim Norman)
|
|
* changed return values to int instead of void
|
|
*
|
|
* path() started.
|
|
*
|
|
* 12/23/96 (Aaron Kaufman)
|
|
* rewrote dir() to mimic MS-DOS's dir
|
|
*
|
|
* 01/28/97 (Tim Norman)
|
|
* cleaned up Aaron's DIR code
|
|
*
|
|
* 06/13/97 (Tim Norman)
|
|
* moved DIR code to dir.c
|
|
* re-implemented Aaron's DIR code
|
|
*
|
|
* 06/14/97 (Steffan Kaiser)
|
|
* ctrl-break handling
|
|
* bug fixes
|
|
*
|
|
* 27-Jul-1998 (John P Price <linux-guru@gcfl.net>)
|
|
* added config.h include
|
|
*
|
|
* 03-Dec-1998 (Eric Kohl <ekohl@abo.rhein-zeitung.de>)
|
|
* Replaced DOS calls by Win32 calls.
|
|
*
|
|
* 08-Dec-1998 (Eric Kohl <ekohl@abo.rhein-zeitung.de>)
|
|
* Added help texts ("/?").
|
|
*
|
|
* 18-Dec-1998 (Eric Kohl <ekohl@abo.rhein-zeitung.de>)
|
|
* Added support for quoted arguments (cd "program files").
|
|
*
|
|
* 07-Jan-1999 (Eric Kohl <ekohl@abo.rhein-zeitung.de>)
|
|
* Clean up.
|
|
*
|
|
* 26-Jan-1999 (Eric Kohl <ekohl@abo.rhein-zeitung.de>)
|
|
* Replaced remaining CRT io functions by Win32 io functions.
|
|
* Unicode safe!
|
|
*
|
|
* 30-Jan-1999 (Eric Kohl <ekohl@abo.rhein-zeitung.de>)
|
|
* Added "cd -" feature. Changes to the previous directory.
|
|
*
|
|
* 15-Mar-1999 (Eric Kohl <ekohl@abo.rhein-zeitung.de>)
|
|
* Fixed bug in "cd -" feature. If the previous directory was a root
|
|
* directory, it was ignored.
|
|
*
|
|
* 23-Feb-2001 (Carl Nettelblad <cnettel@hem.passagen.se>)
|
|
* Improved chdir/cd command.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <windows.h>
|
|
#include <tchar.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <ctype.h>
|
|
|
|
#include "cmd.h"
|
|
|
|
|
|
#ifdef INCLUDE_CMD_CHDIR
|
|
|
|
static LPTSTR lpLastPath;
|
|
|
|
|
|
VOID InitLastPath (VOID)
|
|
{
|
|
lpLastPath = NULL;
|
|
}
|
|
|
|
|
|
VOID FreeLastPath (VOID)
|
|
{
|
|
if (lpLastPath)
|
|
free (lpLastPath);
|
|
}
|
|
|
|
/*
|
|
* CD / CHDIR
|
|
*
|
|
*/
|
|
INT cmd_chdir (LPTSTR cmd, LPTSTR param)
|
|
{
|
|
LPTSTR dir; /* pointer to the directory to change to */
|
|
LPTSTR lpOldPath;
|
|
LPTSTR endofstring; /* pointer to the null character in the directory to change to */
|
|
LPTSTR lastquote; /* pointer to the last quotation mark in the directory to change to */
|
|
|
|
/*Should we better declare a variable containing _tsclen(dir) ? It's used a few times,
|
|
but on the other hand paths are generally not very long*/
|
|
|
|
if (!_tcsncmp (param, _T("/?"), 2))
|
|
{
|
|
ConOutPuts (_T("Changes the current directory or displays it's name\n\n"
|
|
"CHDIR [drive:][path]\n"
|
|
"CHDIR[..|-]\n"
|
|
"CD [drive:][path]\n"
|
|
"CD[..|-]\n\n"
|
|
" .. parent directory\n"
|
|
" - previous directory\n\n"
|
|
"Type CD drive: to display the current directory on the specified drive.\n"
|
|
"Type CD without a parameter to display the current drive and directory."));
|
|
return 0;
|
|
}
|
|
|
|
/* The whole param string is our parameter these days. The only thing we do is eliminating every quotation mark */
|
|
/* Is it safe to change the characters param is pointing to? I presume it is, as there doesn't seem to be any
|
|
post-processing of it after the function call (what would that accomplish?) */
|
|
|
|
dir=param;
|
|
endofstring=dir+_tcslen(dir);
|
|
|
|
while ((lastquote = _tcsrchr(dir,'\"')))
|
|
{
|
|
endofstring--;
|
|
memmove(lastquote,lastquote+1,endofstring-lastquote);
|
|
*endofstring=_T('\0');
|
|
}
|
|
|
|
/* if doing a CD and no parameters given, print out current directory */
|
|
if (!dir || !dir[0])
|
|
{
|
|
TCHAR szPath[MAX_PATH];
|
|
|
|
GetCurrentDirectory (MAX_PATH, szPath);
|
|
|
|
ConOutPuts (szPath);
|
|
|
|
|
|
return 0;
|
|
}
|
|
|
|
if (dir && _tcslen (dir) == 1 && *dir == _T('-'))
|
|
{
|
|
if (lpLastPath)
|
|
dir = lpLastPath;
|
|
else
|
|
return 0;
|
|
}
|
|
else if (dir && _tcslen (dir)==2 && dir[1] == _T(':'))
|
|
{
|
|
TCHAR szRoot[3] = _T("A:");
|
|
TCHAR szPath[MAX_PATH];
|
|
|
|
szRoot[0] = _totupper (dir[0]);
|
|
GetFullPathName (szRoot, MAX_PATH, szPath, NULL);
|
|
|
|
/* PathRemoveBackslash */
|
|
if (_tcslen (szPath) > 3)
|
|
{
|
|
LPTSTR p = _tcsrchr (szPath, _T('\\'));
|
|
*p = _T('\0');
|
|
}
|
|
|
|
ConOutPuts (szPath);
|
|
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* remove trailing \ if any, but ONLY if dir is not the root dir */
|
|
if (_tcslen (dir) > 3 && dir[_tcslen (dir) - 1] == _T('\\'))
|
|
dir[_tcslen(dir) - 1] = _T('\0');
|
|
|
|
|
|
/* store current directory */
|
|
lpOldPath = (LPTSTR)malloc (MAX_PATH * sizeof(TCHAR));
|
|
GetCurrentDirectory (MAX_PATH, lpOldPath);
|
|
|
|
if (!SetCurrentDirectory (dir))
|
|
{
|
|
ErrorMessage (GetLastError(), _T("CD"));
|
|
|
|
/* throw away current directory */
|
|
free (lpOldPath);
|
|
lpOldPath = NULL;
|
|
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
GetCurrentDirectory(MAX_PATH, dir);
|
|
if (dir[0]!=lpOldPath[0])
|
|
{
|
|
SetCurrentDirectory(lpOldPath);
|
|
free(lpOldPath);
|
|
}
|
|
else
|
|
{
|
|
if (lpLastPath)
|
|
free (lpLastPath);
|
|
lpLastPath = lpOldPath;
|
|
}
|
|
}
|
|
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef INCLUDE_CMD_MKDIR
|
|
/*
|
|
* MD / MKDIR
|
|
*
|
|
*/
|
|
INT cmd_mkdir (LPTSTR cmd, LPTSTR param)
|
|
{
|
|
LPTSTR dir; /* pointer to the directory to change to */
|
|
LPTSTR place; /* used to search for the \ when no space is used */
|
|
LPTSTR *p = NULL;
|
|
INT argc;
|
|
|
|
|
|
if (!_tcsncmp (param, _T("/?"), 2))
|
|
{
|
|
ConOutPuts (_T("Creates a directory.\n\n"
|
|
"MKDIR [drive:]path\nMD [drive:]path"));
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* check if there is no space between the command and the path */
|
|
if (param[0] == _T('\0'))
|
|
{
|
|
/* search for the \ or . so that both short & long names will work */
|
|
for (place = cmd; *place; place++)
|
|
if (*place == _T('.') || *place == _T('\\'))
|
|
break;
|
|
|
|
if (*place)
|
|
dir = place;
|
|
else
|
|
/* signal that there are no parameters */
|
|
dir = NULL;
|
|
}
|
|
else
|
|
{
|
|
p = split (param, &argc, FALSE);
|
|
if (argc > 1)
|
|
{
|
|
/*JPP 20-Jul-1998 use standard error message */
|
|
error_too_many_parameters (param);
|
|
freep (p);
|
|
return 1;
|
|
}
|
|
else
|
|
dir = p[0];
|
|
}
|
|
|
|
if (!dir)
|
|
{
|
|
ConErrPrintf (_T("Required parameter missing\n"));
|
|
return 1;
|
|
}
|
|
|
|
/* remove trailing \ if any, but ONLY if dir is not the root dir */
|
|
if (_tcslen (dir) >= 2 && dir[_tcslen (dir) - 1] == _T('\\'))
|
|
dir[_tcslen(dir) - 1] = _T('\0');
|
|
|
|
if (!CreateDirectory (dir, NULL))
|
|
{
|
|
ErrorMessage (GetLastError(), _T("MD"));
|
|
|
|
freep (p);
|
|
return 1;
|
|
}
|
|
|
|
freep (p);
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef INCLUDE_CMD_RMDIR
|
|
/*
|
|
* RD / RMDIR
|
|
*
|
|
*/
|
|
INT cmd_rmdir (LPTSTR cmd, LPTSTR param)
|
|
{
|
|
LPTSTR dir; /* pointer to the directory to change to */
|
|
LPTSTR place; /* used to search for the \ when no space is used */
|
|
|
|
LPTSTR *p = NULL;
|
|
INT argc;
|
|
|
|
if (!_tcsncmp (param, _T("/?"), 2))
|
|
{
|
|
ConOutPuts (_T("Removes a directory.\n\n"
|
|
"RMDIR [drive:]path\nRD [drive:]path"));
|
|
return 0;
|
|
}
|
|
|
|
/* check if there is no space between the command and the path */
|
|
if (param[0] == _T('\0'))
|
|
{
|
|
/* search for the \ or . so that both short & long names will work */
|
|
for (place = cmd; *place; place++)
|
|
if (*place == _T('.') || *place == _T('\\'))
|
|
break;
|
|
|
|
if (*place)
|
|
dir = place;
|
|
else
|
|
/* signal that there are no parameters */
|
|
dir = NULL;
|
|
}
|
|
else
|
|
{
|
|
p = split (param, &argc, FALSE);
|
|
if (argc > 1)
|
|
{
|
|
/*JPP 20-Jul-1998 use standard error message */
|
|
error_too_many_parameters (param);
|
|
freep (p);
|
|
return 1;
|
|
}
|
|
else
|
|
dir = p[0];
|
|
}
|
|
|
|
if (!dir)
|
|
{
|
|
ConErrPrintf (_T("Required parameter missing\n"));
|
|
return 1;
|
|
}
|
|
|
|
/* remove trailing \ if any, but ONLY if dir is not the root dir */
|
|
if (_tcslen (dir) >= 2 && dir[_tcslen (dir) - 1] == _T('\\'))
|
|
dir[_tcslen(dir) - 1] = _T('\0');
|
|
|
|
if (!RemoveDirectory (dir))
|
|
{
|
|
ErrorMessage (GetLastError(), _T("RD"));
|
|
freep (p);
|
|
|
|
return 1;
|
|
}
|
|
|
|
freep (p);
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
|
|
/*
|
|
* set the exitflag to true
|
|
*
|
|
*/
|
|
INT CommandExit (LPTSTR cmd, LPTSTR param)
|
|
{
|
|
if (!_tcsncmp (param, _T("/?"), 2))
|
|
{
|
|
ConOutPuts (_T("Exits the command line interpreter.\n\nEXIT"));
|
|
return 0;
|
|
}
|
|
|
|
bExit = TRUE;
|
|
return 0;
|
|
}
|
|
|
|
|
|
#ifdef INCLUDE_CMD_REM
|
|
/*
|
|
* does nothing
|
|
*
|
|
*/
|
|
INT CommandRem (LPTSTR cmd, LPTSTR param)
|
|
{
|
|
if (!_tcsncmp (param, _T("/?"), 2))
|
|
{
|
|
ConOutPuts (_T("Starts a comment line in a batch file.\n\n"
|
|
"REM [Comment]"));
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
#endif /* INCLUDE_CMD_REM */
|
|
|
|
|
|
INT CommandShowCommands (LPTSTR cmd, LPTSTR param)
|
|
{
|
|
PrintCommandList ();
|
|
return 0;
|
|
}
|
|
|
|
/* EOF */
|