/* * 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 "". * * 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 ) * added config.h include * * 03-Dec-1998 (Eric Kohl ) * Replaced DOS calls by Win32 calls. * * 08-Dec-1998 (Eric Kohl ) * Added help texts ("/?"). * * 18-Dec-1998 (Eric Kohl ) * Added support for quoted arguments (cd "program files"). * * 07-Jan-1999 (Eric Kohl ) * Clean up. * * 26-Jan-1999 (Eric Kohl ) * Replaced remaining CRT io functions by Win32 io functions. * Unicode safe! * * 30-Jan-1999 (Eric Kohl ) * Added "cd -" feature. Changes to the previous directory. * * 15-Mar-1999 (Eric Kohl ) * Fixed bug in "cd -" feature. If the previous directory was a root * directory, it was ignored. * * 23-Feb-2001 (Carl Nettelblad ) * Improved chdir/cd command. */ #include "config.h" #include #include #include #include #include #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 */