DragonGamer
DragonGamer

Reputation: 900

How to execute a simple command line with CreateProcess?

I'd like to execute a simple command line but without to make a window appear. Therefore I can't use System and have to use CreateProcess as far as I know. So I have the following code for example:

//.../

CreateProcess(NULL,input,NULL,NULL,false,NORMAL_PRIORITY_CLASS | 
 CREATE_NO_WINDOW,NULL,NULL,&startInf,&procInf);//)

//.../

If input is a line like "ping www.google.com -n 2" it seems to work. What I need is the deletion-function though. Therefore I tried a lot of variations like:

input = "rd /S /Q \"D:\\ALEX_DATEN\\PC\\C++\\bla\"";

and

  input = "rd /S /Q \"D:/DATEN/PC/C++/bla\"";

But nothing happens and the function returns failure :/ If I write it as a .bat file (without using the "\" escaping chars) the deleting works perfectly!

Does anyone know what I'm doing wrong?

P.s. no, I'm not writing a destructive virus.. if that would have been my target, I would have definitely have found simpler ways...

Upvotes: 5

Views: 23038

Answers (2)

BigBoss
BigBoss

Reputation: 6914

Some system command like rd, del and ... are not actual executable Images (e.g. .exe Files), so you can't execute/run them using CreateProcess they are built-in commands that known to cmd(command interpreter of windows) so you should create cmd and pass your command to it:

wchar_t cmd[ MAX_PATH ];
size_t nSize = _countof(cmd);
_wgetenv_s( &nSize, cmd, L"COMSPEC" );
BOOL b = CreateProcessW( cmd, input, NULL, NULL, FALSE,
    NORMAL_PRIORITY_CLASS | CREATE_NO_WINDOW, NULL, NULL, &startInf, &procInf );

Note: Please see arguments of cmd, you have to use /C to pass your command. So your command is as follow:

wchar_t input[] = L"some command";
wchar_t cmd[MAX_PATH] ;
// initialize cmd
wchar_t cmdline[ MAX_PATH + 50 ];
swprintf_s( cmdline, L"%s /c %s", cmd, input );
STARTUPINFOW startInf;
memset( &startInf, 0, sizeof startInf );
startInf.cb = sizeof(startInf);
// If you want to redirect result of command, set startInf.hStdOutput to a file
// or pipe handle that you can read it, otherwise we are done!
PROCESS_INFORMATION procInf;
memset( &procInf, 0, sizeof procInf );
BOOL b = CreateProcessW( NULL, cmdline, NULL, NULL, FALSE,
    NORMAL_PRIORITY_CLASS | CREATE_NO_WINDOW, NULL, NULL, &startInf, &procInf );
DWORD dwErr = 0;
if( b ) {
    // Wait till cmd do its job
    WaitForSingleObject( procInf.hProcess, INFINITE );
    // Check whether our command succeeded?
    GetExitCodeProcess( procInfo.hProcess, &dwErr );
    // Avoid memory leak by closing process handle
    CloseHandle( procInfo.hProcess );
} else {
    dwErr = GetLastError();
}
if( dwErr ) {
    // deal with error here
}

Upvotes: 4

Remy Lebeau
Remy Lebeau

Reputation: 595392

As others have stated, rd cannot be executed with CreateProcess() directly, you have to execute cmd.exe with /C rd ... as its command-line parameter. Rather than using CreateProcess() like this, you should use SHFileOperation() instead:

SHFILEOPSTRUCT FileOp = {0};
FilOp.wFunc = FO_DELETE;
FileOp.pFrom = "D:\\ALEX_DATEN\\PC\\C++\\bla\0"; 
FileOp.fFlags = FOF_SILENT | FOF_NOCONFIRMATION | FOF_NOERRORUI.

int ErrorCode = SHFileOperation(&FileOp);
if (ErrorCode == 0)
{
    if (FileOp.fAnyOperationsAborted)
        // not everything was deleted
    else
        // delete was successful
}
else
{
    // delete failed
    // note that ErrorCode might not be a Win32 error code,
    // so check the SHFileOperation() documentation for
    // possible alternatives
}

Upvotes: 0

Related Questions