1

独自のシェルを作成する方法に関するチュートリアルに従っていますが、数日間立ち往生しています。

2つのこと:

  1. このコードをコンパイルして実行すると、ランダムにセグメンテーション違反が発生し、その理由がわかりません。
  2. if 文 if (ferror != 0) は常に true のようです。main()関数でfgets()が失敗する理由がわからないので、これは奇妙です。

これらのトピック (またはこのシェルの作成に関するその他のトピック) に関する情報は大歓迎です。

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

#define MAXSIZE 512

int parseCmd(char *cmd, char *args[])
{
    printf("LOGGER: parseCmd(cmd=%s, args=%p)\n", cmd, args);

    char cmdDelims[] = {' ','>'};

    char *cmdReader;
    cmdReader = strtok(cmd, cmdDelims);

    int i = 0;
    while (cmdReader != NULL)
    {       
        args[i] = strdup(cmdReader);

        printf("LOGGER: args[%d]=%s\n", i, args[i]);

        cmdReader = strtok(NULL, cmdDelims);
        i++;
    }
    return 0;
}

void printToLine(char *args[])
{
    int length;
    length = sizeof(args) / sizeof(char);

    int i = 0;
    while (i < length)
    {
        printf("%s\n", args[i]);
        i++;
    }
}

int main(int argc, char *argv[]) 
{   
    char *in;
    in = malloc(MAXSIZE);

    char *args[15];
    char *cmd = NULL;

    int errorBit = 0;
    int terminationBit = 1;
    char error_message[30] = "An error has occurred\n";

    char inDelims[] = "\n";

    while (terminationBit)
    {
        printf("mysh>");

        // get input from command line
        fgets(in, MAXSIZE, stdin);
        if (ferror != 0)
        {
            perror(error_message);
        }

        // get pointer to command line input w/o the newline
        cmd = strtok(in, inDelims);

        // parse the command into separate arguments
        errorBit = parseCmd(cmd, args);
        if (errorBit)
        {
            perror(error_message);
            exit(1);
        }

        printToLine(args);

        // check if the user wants to exit the shell
        if (strcmp(*args, "exit") == 0)
        {
            terminationBit = 0;
        }
    }
    return 0;
}

ここにいくつかの出力があります:

**[ray@12] (6)$ mysh**
mysh>1 2 3
An error has occurred
: Success
LOGGER: parseCmd(cmd=1 2 3, args=0x7fff4a50b080)
LOGGER: args[0]=1
LOGGER: args[1]=2
LOGGER: args[2]=3
1
2
3
Segmentation fault (core dumped)
**[ray@12] (7)$ mysh**
mysh>1 2 3 4 5 6 7 8 9 10
An error has occurred
: Success
LOGGER: parseCmd(cmd=1 2 3 4 5 6 7 8 9 10, args=0x7fffba053d70)
LOGGER: args[0]=1
LOGGER: args[1]=2
LOGGER: args[2]=3
LOGGER: args[3]=4
LOGGER: args[4]=5
LOGGER: args[5]=6
LOGGER: args[6]=7
LOGGER: args[7]=8
LOGGER: args[8]=9
LOGGER: args[9]=10
1
2
3
4
5
6
7
8
mysh>1 2 3
An error has occurred
: Success
LOGGER: parseCmd(cmd=1 2 3, args=0x7fffba053d70)
LOGGER: args[0]=1
LOGGER: args[1]=2
LOGGER: args[2]=3
1
2
3
4
5
6
7
8
4

1 に答える 1

3

エラーについては、ではなくferrorテストする必要があります。後者は関数アドレスであり、ゼロになることはありません。ferror(stdin)ferror

if (ferror(stdin) != 0)
{
    perror(error_message);
}

少なくともいくつかのsegfaultについては、これはあなたが思っていることをしません:

length = sizeof(args) / sizeof(char);

これにより、ポインタを格納するために使用されるバイト数がわかります。これは、引数の数ではなく、 4または8に依存します。

したがって、4つ(または8つ)の引数がある場合は、機能しているように見えます。あなたがもっと持っているなら、それはいくつかの議論を無視しているように見えるでしょう。また、数が少ない場合は、Void全体から欠落している引数をフェッチし、(ほぼ確実に)セグメンテーション違反が発生します。

独立して計算lengthして渡すか、ターミネータをに格納する必要がありargsます。たとえば、最後に見つかった有効な引数の後にNULL引数を追加します。

        cmdReader = strtok(NULL, cmdDelims);
        i++;
    }
    args[i] = NULL;
    return 0;
}

void printToLine(char *args[])
{
    int i = 0;
    while (args[i])
    {
        printf("%s\n", args[i]);
        i++;
    }
}
于 2012-10-03T20:25:51.767 に答える