Команда синтаксического анализа будет работать только для команд из одного слова - PullRequest
0 голосов
/ 22 октября 2018
#include <stdlib.h>
#include <stdio.h>
#include <sys/types.h>
#include <string.h>
#include <sys/wait.h>
#include <unistd.h>
#include <readline/readline.h>
#include <readline/history.h>
HIST_ENTRY;
void printPrompt()
{
    printf("SB> ");
}
char *readCommandLine()
{
    char* myComm = (char*)malloc(1024);
    scanf("%s", myComm);
    return myComm;
}
char *parseCmd(char *myComm, char **tokens, size_t *index)
{
    char *tok;
    const char *delim = " ";
    *index = 0;
        if (tok != NULL)
        {
            tokens[*index] = tok;
            (*index)++;
        }
        else
        {
            tokens[*index] = "\0";
            printf("%s\n", tok);
            return;
        }
    while(tok != NULL)
    {
        tok = strtok(NULL, delim);
        tokens[*index] = tok;
        (*index)++;
    }
    tokens[*index] = NULL;
    return myComm;
}
int isInternalCommand(char *c)
{
    int value = 0;
    if (strcmp(c, "exit") == 0)
    {
        value = 1;
        printf("%s\n", "yes");
    }
    return value;
}
void execInternalCommand(char *c)
{
    int val = 0;
    char a[4];
    if (strcmp(c,"exit") == 0)
    {
        val = val + 1;
    }
    switch(val)
    {
    case 1:
    _exit(0);
    break;
    default:
    break;
    }
}
void executeCommand (char * c, char * const arguments)
{
    execvp(c, arguments);
}
int main()
{
    char *myComm;
    char *parsed;
    size_t num_args = 100;
    char **tokens = malloc(sizeof(char *) * (num_args+1));
    size_t *idx = (size_t *) malloc(sizeof(size_t));
    pid_t pid, cpid;
    int *status;
    while(1)
    {
        printPrompt();
        myComm = readCommandLine();
        parsed = parseCmd(myComm, tokens, idx);
        char *const *args = tokens;
        if (isInternalCommand(parsed))
        {
            execInternalCommand(parsed);
        }
        else
        {
            pid = fork();
            if (pid == 0)
            {
                executeCommand(parsed, args);
                printf("%s\n", "executing...");
            }
            else if (pid > 0)
            {
                waitpid(cpid, &status, 0);
                printf("%s\n", "waiting...");
            }
            else
            {
            } 

    }
    free(idx);
    }
    return 0;
}

Итак, всякий раз, когда я пытаюсь разобрать команду, она работает только для команд из одного слова.Например, всякий раз, когда я пробую mkdir ./something, он отображает mkdir с «отсутствующим операндом», что означает, что он не будет создавать каталог.Кроме того, всякий раз, когда я набираю man что-то, например, man exec, он говорит что-то вроде "какую страницу man вы хотите посмотреть?"Как мне обойти это?Похоже, это проблема синтаксического анализа.

1 Ответ

0 голосов
/ 22 октября 2018
scanf("%s", myComm);

читает одно слово с пробелами из стандартного ввода.В результате ваши команды могут быть только отдельными словами.Если вы хотите прочитать строку, вы должны использовать fgets или, что еще лучше, getline, которая вызовет malloc для вас:

char *readCommandLine()
{
    char* myComm = 0;
    size_t buffer_length = 0;
    size_t length = getline(&myComm, &buffer_length, stdin);
    if (myComm[length] == '\n')
        myComm[length--] = 0;  /* strip off trailing newline, if any */
    return myComm;
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...