2
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

format string 0 picoCTF writeup for beginner

Last updated at Posted at 2024-07-31

記事説明

CTF初めて1週間の主がPicoCTFのEasyをbeginnerむけに書いている記事です。CTFのwriteupではCTF常識が前提になっている記事が多く無知すぎてよくわからなかったので書くことにしました。至らぬところ等あると思いますのでご指摘いただけると嬉しいです。 #picoCTFbeginner

※補足

事前知識

書式文字列攻撃 (Format String Attack)

printfやsyslogなどでは%sや%dを使って変数を表示させることができる。 これを悪用して予期しない値を表示させるのがFormat String Attackです。 フォーマット指定子に対応する引数が不足した状態で表示させようとするとスタック内の他のデータを読み込んで出力しようとします。これにより機密情報が漏洩する可能性があります。
printf("I am %s", "name");
//I am nameと表示される。 
printf("I am %s");
//I am 予期しない値と表示される。

※参考

SIGSEGV

プログラムがメモリに対して不正なアクセスを試みたときに発生するシグナル。 具体的には無効なメモリアドレスの参照、配列の範囲外アクセスなどが含まれる。

関数の意味

printf("\n%s\n", flag)

変数flagに格納されている値を表示

fgets(flag, FLAGSIZE, f);

変数flagにf【file】の内容をFLAGSIZE【64】の字数だけ代入する。

signal(SIGSEGV, sigsegv_handler);

指定されたシグナルに対してシグナルハンドラを設定する関数。 SIGSEGV発生時にsigsegv_handlerを実行する。

flag獲得条件

format-string-0.c
void sigsegv_handler(int sig) {
    printf("\n%s\n", flag);
    fflush(stdout);
    exit(1);
}

int on_menu(char *burger, char *menu[], int count) {
    for (int i = 0; i < count; i++) {
        if (strcmp(burger, menu[i]) == 0)
            return 1;
    }
    return 0;
}

//-中略-

int main(int argc, char **argv){
    FILE *f = fopen("flag.txt", "r");
    if (f == NULL) {
        printf("%s %s", "Please create 'flag.txt' in this directory with your",
                        "own debugging flag.\n");
        exit(0);
    }

    fgets(flag, FLAGSIZE, f);
    signal(SIGSEGV, sigsegv_handler);

    gid_t gid = getegid();
    setresgid(gid, gid, gid);

    serve_patrick();
  
    return 0;
}

不正なメモリアクセスを行わせて、sigsegv_handler()を実行する。
そうするとprintf("\n%s\n", flag)されて、フラグが表示される。

コード概要

第一段階 serve_patrick()

format-string-0.c
void serve_patrick() {
    printf("%s %s\n%s\n%s %s\n%s",
            "Welcome to our newly-opened burger place Pico 'n Patty!",
            "Can you help the picky customers find their favorite burger?",
            "Here comes the first customer Patrick who wants a giant bite.",
            "Please choose from the following burgers:",
            "Breakf@st_Burger, Gr%114d_Cheese, Bac0n_D3luxe",
            "Enter your recommendation: ");
    fflush(stdout);

    char choice1[BUFSIZE];
    scanf("%s", choice1);
    char *menu1[3] = {"Breakf@st_Burger", "Gr%114d_Cheese", "Bac0n_D3luxe"};
    if (!on_menu(choice1, menu1, 3)) {
        printf("%s", "There is no such burger yet!\n");
        fflush(stdout);
    } else {
        int count = printf(choice1);
        if (count > 2 * BUFSIZE) {
            serve_bob();
        } else {
            printf("%s\n%s\n",
                    "Patrick is still hungry!",
                    "Try to serve him something of larger size!");
            fflush(stdout);
        }
    }
}

1.入力された値がmenu1内から選ばれているか確認。falseならその場で終了
2.入力された値の文字数が64より大きいか確認。falseならその場で終了
3.serve_bob()を実行。第2段階へ。

menu1の{"Breakf@st_Burger", "Gr%114d_Cheese", "Bac0n_D3luxe"}の中から文字数が64より大きいものを選ばなければいけない。
printf("Gr%114d_Cheese")をすると%114dに対応する引数がないため、ランダムな数字が114文字幅で表示される。

114文字の文字列例
"Gr                                                                                                           4202954_Cheese"

第二段階 serve_bob()

format-string-0.c
void serve_bob() {
    printf("\n%s %s\n%s %s\n%s %s\n%s",
            "Good job! Patrick is happy!",
            "Now can you serve the second customer?",
            "Sponge Bob wants something outrageous that would break the shop",
            "(better be served quick before the shop owner kicks you out!)",
            "Please choose from the following burgers:",
            "Pe%to_Portobello, $outhwest_Burger, Cla%sic_Che%s%steak",
            "Enter your recommendation: ");
    fflush(stdout);

    char choice2[BUFSIZE];
    scanf("%s", choice2);
    char *menu2[3] = {"Pe%to_Portobello", "$outhwest_Burger", "Cla%sic_Che%s%steak"};
    if (!on_menu(choice2, menu2, 3)) {
        printf("%s", "There is no such burger yet!\n");
        fflush(stdout);
    } else {
        printf(choice2);
        fflush(stdout);
    }
}

1.入力された値がmenu2内から選ばれているか確認。falseならその場で終了
2.入力された値を表示
3.SIGSEGVを起こして、flagを表示させる。

menu2の{"Pe%to_Portobello", "$outhwest_Burger", "Cla%sic_Che%s%steak"}の中の文字列を表示させて、SIGSEGVを起こす必要がある。
printf("Cla%sic_Che%s%steak")をすると%sに対応する引数がないため、ランダムなメモリ内の値が表示される。それがSIGSEGVを起こす。

答え

cmd
Welcome to our newly-opened burger place Pico 'n Patty! Can you help the picky customers find their favorite burger?
Here comes the first customer Patrick who wants a giant bite.
Please choose from the following burgers: Breakf@st_Burger, Gr%114d_Cheese, Bac0n_D3luxe
Enter your recommendation: Gr%114d_Cheese
Gr                                                                                                           4202954_Cheese
Good job! Patrick is happy! Now can you serve the second customer?
Sponge Bob wants something outrageous that would break the shop (better be served quick before the shop owner kicks you out!)
Please choose from the following burgers: Pe%to_Portobello, $outhwest_Burger, Cla%sic_Che%s%steak
Enter your recommendation: Cla%sic_Che%s%steak
ClaCla%sic_Che%s%steakic_Che(null)
picoCTF{7h3_cu570m3r_15_n3v3r_SEGFAULT_dc0f36c4}

2
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
2
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?