サイトトップへ
OSASK.NET
  サイトトップへ       新掲示板(閉鎖済)   Wiki(凍結済)   旧掲示板(廃止済)   ニュース(廃止済)  
1: 2009-08-03 (月) 21:02:26 ソース 2: 2009-08-03 (月) 22:23:40 ソース
Line 3: Line 3:
-メモのうち重要な部分をそのうちまとめてまともなページを作る -メモのうち重要な部分をそのうちまとめてまともなページを作る
*** (42) naskで.g01アプリを作るには?(1) *** (42) naskで.g01アプリを作るには?(1)
--[[impressions]]+-[[impressions]]の2009-07-26ごろのfshinoさんのリクエストに答えるために、このセクションは用意されました。 
 +-文脈的には[[GUIGUI01/memo27]]の続きだけど、あっちはC言語。こっちはアセンブラ主体です。 
 +---- 
 +-何はともあれ、まずは文字列表示を使って"hello, world\n"をやってみることにします。・・・あれ?[[GUIGUI01/memo18]]に今から僕が書こうとしていることに近いことが書いてありますね。ああでも、今回はサイズ重視ではなく、汎用かつ簡潔性重視で行きます。 
 + ; ex0025.nas 
 + [FORMAT "WCOFF"] 
 + [FILE "ex0025.nas"] 
 + [INSTRSET "i486p"] 
 + [BITS 32] 
 +         GLOBAL  _G01Main 
 +         EXTERN  _g01_execcmd0 
 + 
 + [SECTION .text] 
 + 
 + _G01Main: 
 +         MOV    EAX,msg 
 +         CALL    _g01_execcmd0 
 +         DB      0x53, 0x00    ; g01_putstr0((char *) EAX); 
 +         RET 
 + 
 + [SECTION .data] 
 + 
 + msg    DB      "hello, world", 0x0a, 0 
 +-これです。makeすると64バイトになるでしょう。 
 +-要点は次の通りです。 
 +--「CALL _g01_execcmd0 DB 0x53, 0x00」が、EAX番地からの文字列表示API 
 +--(ポインタで指定する)文字列は.dataかもしくはスタック内に置かなければいけない。 
 +--「CALL _g01_execcmd0」すると、ESIはある特別な値が代入され、EDIは強制的に0になる。これが困る場合は、PUSHなどでレジスタを退避すること。 
 +--アプリの実行開始番地は_G01Mainに固定。 
 +---他のレジスタは原則として影響がない。 
 +-EAX以外のレジスタを指定したい場合は次の通りです。 
 +--「CALL _g01_execcmd0 DB 0x53, 0x10」が、ECX番地からの文字列表示API 
 +--「CALL _g01_execcmd0 DB 0x53, 0x20」が、EDX番地からの文字列表示API 
 +--「CALL _g01_execcmd0 DB 0x53, 0x30」が、EBX番地からの文字列表示API 
 +---- 
 +-次は1文字表示APIです。 
 + ; ex0026.nas 
 + [FORMAT "WCOFF"] 
 + [FILE "ex0026.nas"] 
 + [INSTRSET "i486p"] 
 + [BITS 32] 
 +         GLOBAL  _G01Main 
 +         EXTERN  _g01_execcmd0 
 + 
 + [SECTION .text] 
 + 
 + _G01Main: 
 +         MOV    AL,0x20 
 + .putc_loop 
 +         CALL    _g01_execcmd0 
 +         DB      0x55, 0x16, 0x00    ; g01_putc(EAX); 
 +         INC    EAX 
 +         CMP    AL,0x7e 
 +         JBE    .putc_loop 
 +         MOV    AL,0x0a 
 +         CALL    _g01_execcmd0 
 +         DB      0x55, 0x16, 0x00    ; g01_putc(EAX); 
 +         RET 
 +-これです。makeすると57バイトになるでしょう。 
 +-要点は次の通りです。 
 +--「CALL _g01_execcmd0 DB 0x55, 0x16, 0x00」が、putc(EAX);相当のAPI 
 +-ちなみにこれもレジスタを選べます。 
 +--「CALL _g01_execcmd0 DB 0x55, 0x16, 0x10」が、putc(ECX);相当のAPI 
 +--「CALL _g01_execcmd0 DB 0x55, 0x16, 0x20」が、putc(EDX);相当のAPI 
 +--「CALL _g01_execcmd0 DB 0x55, 0x16, 0x30」が、putc(EBX);相当のAPI 
 +-こんな感じで分かるでしょうか? 
 +---- 
 +-ここで、「CALL _g01_execcmd0」の正体を明かしたいと思います。これはライブラリで用意されている関数なのですが、中身はこうなっています。 
 +         EXTERN _g01_esi0 
 + 
 + _g01_execcmd0: 
 +         XOR    EDI,EDI 
 +         MOV    ESI,[_g01_esi0] 
 +         JMP    [ESI]          ; これがAPI呼び出し。RETで帰ると_g01_execcmd0の呼び出し元へ帰る。 
 +-だからEDIが0になるとか、ESIの値が上書きされるなんてことになっていたわけです。 
 +-これを利用すると、ex0026.nasは次のように書き換えられます。 
 + ; ex0026.nas 
 + [FORMAT "WCOFF"] 
 + [FILE "ex0026.nas"] 
 + [INSTRSET "i486p"] 
 + [BITS 32] 
 +         GLOBAL  _G01Main 
 +         EXTERN  _g01_esi0 
 + 
 + [SECTION .text] 
 + 
 + _G01Main: 
 +         MOV    AL,0x20 
 +         XOR    EDI,EDI 
 +         MOV    ESI,[_g01_esi0] 
 + .putc_loop 
 +         CALL    [ESI] 
 +         DB      0x55, 0x16, 0x00    ; g01_putc(EAX); 
 +         INC    EAX 
 +         CMP    AL,0x7e 
 +         JBE    .putc_loop 
 +         MOV    AL,0x0a 
 +         CALL    [ESI] 
 +         DB      0x55, 0x16, 0x00    ; g01_putc(EAX); 
 +         RET 
 +-これです。makeすると56バイトになるでしょう。ちょっとだけ小さくなったわけです。 
 +-さらに小さくすることもできます。実は.g01アプリが実行される際には、EDI=0,ESI=[_g01_esi0]の初期値で始まることが保証されているのです。つまり、_G01Main:に入った時点でこれらの値は保証されているので、わざわざ初期化する必要はないのです。そうすると、こう書けます。 
 + ; ex0026.nas 
 + [FORMAT "WCOFF"] 
 + [FILE "ex0026.nas"] 
 + [INSTRSET "i486p"] 
 + [BITS 32] 
 +         GLOBAL  _G01Main 
 + 
 + [SECTION .text] 
 + 
 + _G01Main: 
 +         MOV    AL,0x20 
 + .putc_loop 
 +         CALL    [ESI] 
 +         DB      0x55, 0x16, 0x00    ; g01_putc(EAX); 
 +         INC    EAX 
 +         CMP    AL,0x7e 
 +         JBE    .putc_loop 
 +         MOV    AL,0x0a 
 +         CALL    [ESI] 
 +         DB      0x55, 0x16, 0x00    ; g01_putc(EAX); 
 +         RET 
 +-短くなりました。
* こめんと欄 * こめんと欄
#comment #comment

トップ   差分 バックアップ 複製 名前変更 リロード印刷に適した表示   ページ新規作成 全ページ一覧 単語検索 最新ページの一覧   ヘルプ
新着

目次
メンバー一覧


最新の20件
2016-10-01 2016-09-08
  • @MenuBar.
2016-09-07 2016-09-04 2016-08-15 2015-09-23 2014-07-30 2014-07-04 2014-02-04 2013-10-26 2013-06-21 2013-06-17 2013-06-15 2013-04-02 2013-02-09 2013-02-04 2012-12-25 2012-12-01 2012-05-28 2012-03-31

トピック一覧
一般用コメント最新
新掲示板lina
2016/9/5 20:58
SandBoxゲスト
2016/9/4 12:01
RecentDeletedlina
2015/6/2 19:29
Old-OSASK-MLlina
2014/6/29 9:14
hideyosi/メールhideyosi
2014/1/6 20:17
hideyosi/募集中lina
2013/11/8 19:56

このサイトは川合秀実から委託を受けて、OSASKコミュニティによって管理・運営されています。