4: 2005-07-22 (金) 08:08:57 |
現: 2024-01-06 (土) 22:39:10 |
| | DW| 2| ;ヘッドの数はいくつか(一般に1440KB-FDの場合は2)| | | | DW| 2| ;ヘッドの数はいくつか(一般に1440KB-FDの場合は2)| |
| | DD| 0| ;このBPBセクタはデバイスの先頭から数えてどこにあるか(HDDのパーティション開始位置を表すために用いられる、セクタ単位、一般に1440KB-FDの場合は0)| | | | DD| 0| ;このBPBセクタはデバイスの先頭から数えてどこにあるか(HDDのパーティション開始位置を表すために用いられる、セクタ単位、一般に1440KB-FDの場合は0)| |
- | | DD| 2880| ;| | + | | DD| 2880| ;セクターが全部で何個あるか| |
| | DB| 0,0| ;よくわからないけどとりあえず0にしておくといいらしい| | | | DB| 0,0| ;よくわからないけどとりあえず0にしておくといいらしい| |
| | DB| 29H| ;よくわからないけどとりあえず0x29を書くらしい| | | | DB| 29H| ;よくわからないけどとりあえず0x29を書くらしい| |
| | | |
| -FAT領域 | | -FAT領域 |
- | --ファイルの情報(データではない)が格納されます。ここは、「FATの長さ(1つ分の長さをセクタ数で)」の値から解ります。9です。つまり、上記のスタート地点(セクター2)から9個分のセクターを使うということになります。なので、「セクター1〜セクター9」ということになります。 | + | --ファイルの情報(データではない)が格納されます。ここは、「FATの長さ(1つ分の長さをセクタ数で)」の値から解ります。9です。つまり、上記のスタート地点(セクター1)から9個分のセクターを使うということになります。なので、「セクター1〜セクター9」ということになります。 |
| --さらに、「FATがいくつあるか」という情報も参照します。2ですね。もう一個は予備用らしいですが、ちゃんと領域を喰います。二つ目の予備FATも同じく9個なので、「セクター10〜セクター18」となります。 | | --さらに、「FATがいくつあるか」という情報も参照します。2ですね。もう一個は予備用らしいですが、ちゃんと領域を喰います。二つ目の予備FATも同じく9個なので、「セクター10〜セクター18」となります。 |
| --結果、FAT領域は、「セクター1〜セクター18」となります。 | | --結果、FAT領域は、「セクター1〜セクター18」となります。 |
| となります。・・・じゃ、論理セクター18は?っていうと、 | | となります。・・・じゃ、論理セクター18は?っていうと、 |
| | | |
- | |シリンダ| 1| (一番内側から2つ目)| | + | |シリンダ| 0| (一番内側)| |
| |セクタ| 1 | (詳しいことは省略。FDはちゃんとスタート位置があります。)| | | |セクタ| 1 | (詳しいことは省略。FDはちゃんとスタート位置があります。)| |
- | |ヘッド| 0 | (表面のこと)| | + | |ヘッド| 1 | (裏面のこと)| |
| | | |
| と、こう続いていくわけです。もうお解りですね? (^^) | | と、こう続いていくわけです。もうお解りですね? (^^) |
| + | |
| + | 同じくセクタ34はどこかっていうと、 |
| + | |
| + | |シリンダ| 0| (一番内側)| |
| + | |セクタ| 18 | (詳しいことは省略。FDはちゃんとスタート位置があります。)| |
| + | |ヘッド| 1 | (裏面のこと)| |
| + | |
| + | じゃ、セクタ35は? というと・・・・ |
| + | |
| + | |シリンダ| 1| (一番内側から2番目)| |
| + | |セクタ| 1 | (詳しいことは省略。FDはちゃんとスタート位置があります。)| |
| + | |ヘッド| 0 | (表面のこと)| |
| + | |
| + | とこう続いていくわけです。 |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| | | |
| | | |
| これを、ちょうどベーシックのFor〜Nextみたいにできないものだろうか・・・ | | これを、ちょうどベーシックのFor〜Nextみたいにできないものだろうか・・・ |
| | | |
| + | ここで、論理セクタから自動的に物理セクタに変換して、かつ、とりあえず連番なら自動で読み込んでくれるような式を試案してみる。 |
| | | |
| + | ;AXレジスタに開始番号、DXレジスタに終了番号を入れてコールすると読んでくれる |
| + | ;なんてできないものかな・・・ |
| + | |
| + | readsec1: |
| + | PUSH AX ;AXの値を一旦バックアップ |
| + | |
| + | SUB AX,DX ;まず引き算して、AXに「個数」を入れる |
| + | MOV CX,AX ;CXレジスタに個数を入れる。(LOOP用のカウンタ) |
| + | |
| + | POP AX ;リストア |
| + | |
| + | ;ヘッダの値を計算する |
| + | PUSH AX ;AXの値を一旦バックアップ |
| + | DIV AX,2 ;値を半分に割る。 |
| + | SUB AX,1439 ;片面のセクタ数を引き算する。 |
| + | |
| + | ;・・・あれぇ。引き算の場合、マイナスになるような場合って・・・ |
| | | |
| + | あかん・・・だめだ。ちょっと整理しよう。 |
| | | |
| + | -読み込みたいセクターの位置指定を求める計算式 |
| + | --ヘッド |
| + | +++読みたい論理セクターの値 ÷ (総セクター数÷2) |
| + | --シリンダ |
| + | +++読みたい論理セクター値 ÷ セクター数(18) |
| + | --セクター値 |
| + | +++読みたい論理セクター値 mod セクター数(18) |
| | | |
| + | えーっと・・・これでいいかな? ちょっと検算中・・・ |
| | | |
| + | あークソ!だめだ。ちょっと別に式を組み立てないとわからんわい!(あったまわる・・・) |
| | | |
| + | えーっと。逆に、論理セクタを計算する式を考えてみる。 |
| | | |
| + | -論理セクタ : R |
| + | -シリンダ :c |
| + | -セクタ :s |
| + | -ヘッド :h |
| | | |
| + | だとすると・・・こうかな?? |
| | | |
| + | R = c × 18 + s +(2880/2*h) -1 |
| | | |
| + | おぉぉ! これでいいらしいぞ!!! (^^;) |
| | | |
| + | では、この式を展開して・・・・あ”だめだ。ちょっとこれじゃ解りにくいな。ちと変更する。 |
| | | |
| | | |
| + | -論理セクタ : R |
| + | -シリンダ :c |
| + | -セクタ :s |
| + | -ヘッド :h |
| + | -シリンダ内のセクター数:A |
| + | -セクター総数:B |
| | | |
| + | R = cA + s +(B/2*h) -1 |
| | | |
| + | こうだ。これを展開。(なつかすいなぁ。こんなの中学以来かも・・・(^^;)) |
| | | |
| | | |
| + | h = (R-cA-s+1)÷B÷2 |
| + | s = R-cA+1-(B÷2h) |
| + | c = (R-s-(B÷2h)+1)÷A |
| | | |
| + | よーし!これで求められるぞ! ばんざーい!ばんざーい! |
| | | |
| + | ・・・・''ダメじゃん''・・・・ orz |
| + | |
| + | あー。むずかしく考えすぎたかもしれん。こうかな??? |
| + | |
| + | ;まず、hを求める |
| + | h = R ÷ (B ÷ 2) (余りは破棄) |
| + | |
| + | ・・・うん。これでよさそうだ。 |
| + | で、これでhが求められると。 |
| + | |
| + | ;セクタも求める |
| + | s = R mod A + 1 |
| + | |
| + | ・・・おぉ? これでいいのかな??? |
| + | |
| + | そうするってーと、シリンダはこうかな? |
| + | |
| + | ;シリンダを求める |
| + | Q = B ÷ A ÷ 2 ;先に一面のシリンダ数を求める |
| + | c = R ÷ A - (Q * h) |
| + | |
| + | うん!これでよさそうだぞ! |
| + | |
| + | では、これらを実際のアセンブラにするってーと・・・・ |
| + | |
| + | まずはh。 |
| + | ;h = R ÷ (B ÷ 2) (余りは破棄)だから、 |
| + | |
| + | PUSH AX ;AXには論理セクタ値が入っている。一旦バックアップ。 |
| + | MOV AX,2880 ;まず、AXに総セクタ数を入れる。 |
| + | MOV DX,2 ;DXレジスタに2を入れる |
| + | DIV DX ;AX ÷ DX を実行し、商をAXへ。余りをDXへ。 |
| + | MOV DX,AX ;AXレジスタの値をDXにコピー |
| + | POP AX ;AXレジスタを元に戻す(論理セクタ値をセット) |
| + | PUSH AX ;もう一回バックアップ。 |
| + | DIV DX ;これで、AXにはhの値が入っているはず。 |
| + | |
| + | ふむ・・・一旦DOSのDEBUGで確認してみよう。 |
| + | |
| + | ・・・あっれえええええ??? 割り算(DIV)がどうしてもうまくいかないぞ?? |
| + | |
| + | なになに??? 「÷2」などは shr を使うのが早いと? じゃあこうか。 |
| + | |
| + | PUSH AX ;AXには論理セクタ値が入っている。一旦バックアップ。 |
| + | MOV AX,2880 ;まず、AXに総セクタ数を入れる。 |
| + | SHR AX,1 ;AX ÷ 2 を実行ししたのと同じ結果になる。 |
| + | |
| + | MOV DX,AX ;AXレジスタの値をDXにコピー |
| + | POP AX ;AXレジスタを元に戻す(論理セクタ値をセット) |
| + | PUSH AX ;もう一回バックアップ。 |
| + | DIV DX ;これで、AXにはhの値が入っているはず。 |
| + | |
| + | うーーーむ・・・・ 16bit同士の割り算ってどうやったらいいんだ・・・ |
| + | |
| + | まいったなぁ。やっぱどっかメモリを使わないといけないのかなぁ。メモリは遅いからできれば使いたくない。・・・うーん。遅いっていえばそもそもDIV命令も評判がよくないなぁ。なんかうまい工夫ないもんかな。 |
| + | |
| + | こういう時って、CXやBX使っちゃいけないのかなぁ・・・・。例えば |
| + | |
| + | |
| + | PUSH CX ;CXレジスタの値をバックアップ |
| + | PUSH AX ;AXには論理セクタ値が入っている。一旦バックアップ。 |
| + | MOV AX,2880 ;まず、AXに総セクタ数を入れる。 |
| + | SHR AX,1 ;AX ÷ 2 を実行ししたのと同じ結果になる。 |
| + | MOV CX,AX ;AXの内容をCXにコピー |
| + | |
| + | POP AX ;AXの値を論理セクタに戻す |
| + | PUSH AX ;もう一回バックアップ |
| + | MOV DX,0 ;これでDX:AXの値は32bitながら論理セクタ値となるはず。 |
| + | |
| + | DIV CX ;これで、AXにはhの値が入っているはず。 |
| + | |
| + | MOV DX,AX ;AXの値(h)をDXレジスタに移す |
| + | |
| + | POP AX ;AXを初期値に戻す(論理セクター値) |
| + | POP CX ;CXを初期値に戻す |
| + | |
| + | ;これで全て元通りで、かつ、DXにはhがセットされているはず。 |
| + | |
| + | うーむ。とりあえずは成功かな??? |
| + | |
| + | そんなわけで、[[こんなコード>IPL研究/コード1]]を妄想してみる。 |
| + | |
| + | debugコマンドで検証してみないと・・・ |
| + | |
| + | くそ!だめだ。割り算がメタメタだ。たぶん桁のせいだろう・・・ |
| + | |
| + | シリンダやセクタは同時に計算できないだろうか。 |
| + | |
| + | そんなわけで、[[こういうコード>IPL研究/コード2]]コードを書いてみた。 |
| + | |
| + | キタ━━━━━━(゚∀゚)━━━━━━!!!!!!!! |
| + | |
| + | たぶん、これで大丈夫なはずだぞ!!!!! |
| + | |
| + | |
| + | ''がーーーん!'' |
| + | 大勘違いしてたかも!?????。スタックって、メモリなの?????? |
| + | それじゃ、どこかに適当にメモリを確保して書き込んだって同じじゃんかよ!!! |
| + | |
| + | 課題としては、「DIV命令は遅いので最小で」と、「メモリも遅いので出来るだけレジスタ内で完結」だなこりゃ。 |
| + | |
| + | コード2を見直そう・・・・・ orz |
| + | |
| + | そんなわけで[[こんなコード>IPL研究/コード3]]になった。スタックの使用を最小限にして、割り算も減らした。OK。あとは、実際に繰り返しを行うプログラムを書いた時に、一回一回割り算をしなくてもいいように調整すると。 |
| + | |
| + | |
| + | **どこに読み込むか?(格納メモリ番地) [#j27da20b] |
| + | 上記のオベンキョでセクタの指定はできるようになったけど、読んだセクタはどこに格納したらいいでしょう??? |
| + | |
| + | まずはメモリマップを読んでお勉強 |
| + | |
| + | K氏のheboOSのIPLでは、読み込みは 0x800:0x100からスタートしている。このアドレスはなんだろう? (たぶんMS-DOSの実行ファイルとの互換だろうが・・・) |
| + | |
| + | えーーーーっと・・・・・ |
| + | |
| + | 8086時には、セグメント:オフセットという指定でアドレスを指定できるんだけど、本来は0hからいくらでもアドレスは増やせるはず。(たぶんプロテクトモードなんかではそうなんだろう。 |
| + | |
| + | セグメントの指定は5桁まで。で、オフセットは4桁まで。そう考えると、8086時には |
| + | |
| + | 0h 〜 FFFFFh までの空間がアクセル可能と。(5桁だから) |
| + | |
| + | 0h 〜 FFFFF までのアドレスにアクセルできる。で、この中に勝手に使っていい部分とそうでない部分があるはず。 |
| + | |
| + | heboOSで読み込んでいるアドレスは、0x800:0100 ってことは、 |
| + | |
| + | 00800 + 0100 = 900番地 |
| + | |
| + | からは、勝手に使っていい・・・・のかな? いやたぶんそんなことはないはず。0〜900番地の間はどうなっているんだろう??? |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | |
| + | ~ |
| + | ~ |
| + | ~ |
| + | ~ |
| + | ~ |
| + | ~ |
| + | ~ |
| + | ~ |
| | | |
| | | |
| | | |
| ---- | | ---- |
| + | |
| + | ~ |
| + | ~ |
| + | ~ |
| + | ~ |
| + | ~ |
| + | ~ |
| + | ~ |
| + | ~ |
| + | |
| | | |
| | | |