Using execv (C language) to run commands from a li

2019-08-11 05:31发布

The only part I am confused on thus far is how to set up execv with the first parameter as the current working directory. I've tried both "." and "~", neither are executing anything to the screen; same for "/." and "/~". I'm confused on how to have execv run something like this:
$ ./prog ls -t -al

And have it execute the commands after the program execution (which are stored into argv) in the current directory, or the same directory as the file is in (which will vary based on who is using it.)

My code:

#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>

void main(int argc, char *argv[])
{
        int pid;
        int count = 0;
        char *argv2[argc+1];

        for(count = 0; count < argc-1; count++){
                argv2[count] = argv[count+1];
                printf("Argv2: %s\n", argv2[count]);  //just double checking
                argv2[argc-1] = NULL;
        }

        pid = fork();
        if(pid == 0){
                printf("Child's PID is %d. Parent's PID is %d\n", (int)getpid, (int)getppid());
                execv(".", argv2);       //<---- confused here
        }
        else{
        wait(pid);
        exit(0);
        }
}




Some sample output:
$ ./prog ls -t -al
Argv2: ls
Argv2: -t
Argv2: -al
Child's PID is 19194. Parent's PID is 19193

标签: c unix argv execv
6条回答
等我变得足够好
2楼-- · 2019-08-11 06:10

The first argument is the absolute path of the program you want to run; in other words, the first part of the command you type in a terminal. You can find where the program using the command whereis. Also, execv function can look up the path for you, which makes your code more portable.

查看更多
聊天终结者
3楼-- · 2019-08-11 06:11

The first arg to execv() is not the directory in which to run but rather the path to the file you want to exec. By default, your exec-ed program will run in the context of the caller's current directory.

查看更多
乱世女痞
4楼-- · 2019-08-11 06:17

Code sample for your reference:

#include <stdio.h>

int main() {

    int ret = fork();
    if(ret == 0)
    {
            char *params[4]  = {"/bin/ls", "-l",0}; //cmd params filled

            int res = execv( "/bin/ls" , params);  //parameters for cmd
            //int res = execv( "/bin/ls" , NULL);  //this is fail case (when child-exit with -1 status can be seen)
            printf("\n child exiting (%d) .. \n", res); //on successful execution of cmd, this exit never appears
        }
        else
        {
            waitpid(ret,1,0);
            printf("parent exiting\n");
    }

    return 1;
 }
查看更多
一纸荒年 Trace。
5楼-- · 2019-08-11 06:18
/* main() returns int */
int main(int argc, char *argv[])
{
        int pid;

        pid = fork();
        if(pid == 0){
                printf("Child's PID is %d. Parent's PID is %d\n"
                      , (int)getpid, (int)getppid());
                execv(argv[1], argv+1);
        }
        else{
        wait(NULL);
        exit(0);
        }

        return 0;
}

UPDATE: execv() needs the absolute path of the executable; for files in the current directory you would have to construct that path (eg via pwd()). If you want the executable to seached via the $PATH environment variable, you could use execvp(), which does all the seaching for you.

查看更多
乱世女痞
6楼-- · 2019-08-11 06:19

I guess execv is what is required to be used. execvp is a lot nicer since it will look for commands in your PATH setting.

execv(".", argv2);       //<---- confused here

...

#include <errno.h>
#include <string.h>
if ( execv(argv2[0],argv2) )
{
    printf("execv failed with error %d %s\n",errno,strerror(errno));
    return 254;  
}

wait(pid);

...

pid_t wait_status = wait(&pid);
查看更多
smile是对你的礼貌
7楼-- · 2019-08-11 06:30

The first argument, by convention, should point to the filename associated with the file being executed. You want to exec ls, so the first argument should be /bin/ls,that means the code is

execv("/bin/ls", argv2);

You could try it

查看更多
登录 后发表回答