execvp和读命令参数

时间:2014-02-14 21:02:39

标签: c linux shell

我正在开发一个项目,我必须在C中为linux编写一个命令shell。到目前为止,它正在为没有输入的命令工作(即,shell将运行'date'命令和'ls'命令很好。)

但是,需要一些输入的命令,似乎它将每个输入读作一个单独的命令。例如,对于命令:

% gcc -o testFile testFile.c

好像shell正在运行gcc作为自己的命令,然后是-o,然后是testFile和testfile.c,它应该将gcc作为命令,其他三个条目作为输入。

我不明白代码中发生了什么 - 它可能来自于不完全理解execvp函数(我从几个来源读到它并且我仍然认为我不理解它 - 我以为我做到了! )。

execvp调用在函数execute。

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include "commandStorage.c"

#define MAX_ARGUMENTS 10


void parseLine(char *command, char **args) {

const char split = '\0';
int i;
char *ptrToken;

while(*command != '\0') {
    while ((*command == '\n') || (*command == '\t')
        || (*command == ' ')) {
        *(command++) = '\0';
        } // end 'while'
    *args++ = command;
    printf("%s\n", command);
    while ((*command != '\n') && (*command != '\t')
        && (*command != '\0') && (*command != ' ')) {
        command++;
        } // end 'while'

    } // end 'while'
*args = '\0';
} // end parseLine

void execute(char **arrayOfPtrs) {
/*
CURRENT BUG:
The function is taking the array of pointers, and executing each
input from the array seperately.

The execvp function is being misused.
*/
pid_t pid;
int waitStatus;

switch (pid = fork()) {
    case 0:  // Child process
        if (execvp(arrayOfPtrs[0], arrayOfPtrs) < 0) {
            perror("THE COMMAND FAILED TO EXECUTE!");
            break;

            } // end 'if

    case -1:  // Fork failed
        perror("THE PROCESS FAILED TO FORK");
        break;

    default: // Parent process
        while ((wait(&waitStatus) != pid)) {};
        break;
} // end 'switch'
return;
} // end 'execute

void clearPointerArray(char **args){

while (*args != NULL) {
    *(args++) = NULL;
}
}


int main() {

int i;
char command[MAX_STRING_LENGTH]; // unparsed command
pid_t pid;
char *args[MAX_ARGUMENTS]; // Argument vector.


while(1) {
    clearPointerArray(args);
    printf("%s", "TimsShell--> ");
    scanf("%s", command);

    parseLine(command, args);

    if ((strcmp(args[0], "exit") == 0) || (strcmp(args[0], "quit") == 0)) {
        printf("%s\n", "Goodbye!");
        return 0;
    }

    execute(args);

    }// while loop
return 0;

} // main

以下是命令行的一些输出:

% gcc -o mainShell mainShell.c
% ./mainShell
TimsShell--> date
date
Fri Feb 14 15:50:28 EST 2014
TimsShell--> ls
ls
change.log      doLocalConf.xml   license.txt   notepad++.exe  session.xml               testFile.c
commandStorage.c  functionList.xml  localization  plugins     shortcuts.xml         themes
config.model.xml  langs.model.xml   mainShell      readme.txt     stylers.model.xml    updater
config.xml      langs.xml        mainShell.c   SciLexer.dll     stylers.xml          user.manual
TimsShell--> gcc -o testFile testFile.c
gcc
gcc: fatal error: no input files
compilation terminated.
TimsShell--> -o
THE COMMAND FAILED TO EXECUTE!: No such file or directory
TimsShell--> testFile
THE COMMAND FAILED TO EXECUTE!: No such file or directory
TimsShell--> testFile.c
: not found 2: testFile.c: 
testFile.c: 3: testFile.c: Syntax error: "(" unexpected
TimsShell--> ^C

1 个答案:

答案 0 :(得分:1)

有关发生这种情况的原因,请参阅scanf的手册页。问题在于格式字符串:

  

%S

     

匹配一系列非空白字符;下一个指针   必须是一个指向字符数组的指针,该数组足够长以容纳   输入序列和添加的终止空字节('\ 0')   自动。输入字符串在空白处或最大处停止   字段宽度,以先到者为准。

请改为尝试:

scanf("%[^\n]s", command);
getchar();

scanf(" %[^\n]s", command);