Проблема с прерыванием сборки: int 13h (ah = 42h) не удалось вернуть cf = 1, ah = 1 - PullRequest
0 голосов
/ 06 июня 2011

Я хочу написать программу на ассемблере для чтения основных разделов жесткого диска. За последние несколько дней я много гуглил и обнаружил, что, может быть, int 13h (ah = 42h) для меня. Но я потерпел неудачу в начале. После вызова INT 13H для CF было установлено значение 1, а для AH - 1. Из документов, которые я знаю, прерывание было неудачным.

Вот мой код:

ASSUME CS:CodeSeg, DS:DataSeg, SS:StackSeg

DataSeg SEGMENT

BSBuffer:             ; Abbr for Boot Sector Buffer.
MBRecord:             ; Master Boot Record.
    MBR DB 446 DUP (0)

PartitionA:
    StatusA      DB 0 ;
    BeginHeadA   DB 0 ;
    BeginSeclynA DW 0 ;
    FileSystemA  DB 0 ;
    FinalHeadA   DB 0 ;
    FinalSeclynA DW 0 ;
    BeginSectorA DD 0 ;
    SectorCountA DD 0 ;

PartitionB:
    StatusB      DB 0 ;
    BeginHeadB   DB 0 ;
    BeginSeclynB DW 0 ;
    FileSystemB  DB 0 ;
    FinalHeadB   DB 0 ;
    FinalSeclynB DW 0 ;
    BeginSectorB DD 0 ;
    SectorCountB DD 0 ;

PartitionC:
    StatusC      DB 0 ;
    BeginHeadC   DB 0 ;
    BeginSeclynC DW 0 ;
    FileSystemC  DB 0 ;
    FinalHeadC   DB 0 ;
    FinalSeclynC DW 0 ;
    BeginSectorC DD 0 ;
    SectorCountC DD 0 ;

PartitionD:
    StatusD      DB 0 ;
    BeginHeadD   DB 0 ;
    BeginSeclynD DW 0 ;
    FileSystemD  DB 0 ;
    FinalHeadD   DB 0 ;
    FinalSeclynD DW 0 ;
    BeginSectorD DD 0 ;
    SectorCountD DD 0 ;

Validation:
    VALID DW 0 ; Should be 55AAH.

; DAPacket is used as the input parameter of ReadBootSector PROC

DAPacket:                ; Abbr for Disk Address Packet.
    PacketSize    DB 16  ; Always 16.
    Reserved      DB 0   ; Reserved.
    SectorCount   DW 1   ; Should be 1 to read boot sector.
    BufferOffset  DW 0
    BufferSegment DW 0
    BlockNumber DB 8 DUP (0)

DataSeg ENDS

StackSeg SEGMENT
    DB 4096 DUP (0)
StackSeg ENDS

CodeSeg SEGMENT
START:

    MOV AX, DataSeg
    MOV DS, AX
    MOV AX, StackSeg
    MOV SS, AX
    MOV SP, 4096

    MOV DL, 80H
    CALL ReadDisk

    MOV CX, VALID

    MOV AX, 4C00H
    INT 21H

; This process is used to read the boot sector of a given disk.
; Input:
;     DL - Disk ID, 0~79H for floppies, 80H~FFH for hds.
; Output:
;     BSBuffer - Boot sector of the disk indicated by DL.

ReadDisk:

    PUSH AX
    PUSH SI
    MOV SI, DAPacket
    MOV PacketSize, 16
    MOV SectorCount, 1
    MOV BufferOffset, BSBuffer
    MOV BufferSegment, DataSeg

    MOV AH, 42H

    INT 13H

    POP SI
    POP AX

    RET

CodeSeg ENDS
END START

Спасибо!

Ответы [ 2 ]

1 голос
/ 16 сентября 2011

Сектор имеет 512 (0x200) байтов, и если вы хотите записать его в сегмент данных, вы должны сделать блок длиной не менее 512 байтов.В противном случае вы перезапишете КОД / ДАННЫЕ, которые пытаетесь выполнить.

1 голос
/ 13 июля 2011

Вы использовали две функции, которые не являются частью одного и того же API.

int 13h - ах: 42h => Это функция BIOS (расширение диска чтения IBM / MS)

int 21h - ах: 4Ch => Это функция DOS (метод завершения процесса)

Эта программа нигде не может работать!

Редактировать : Это неверно.Ты прав @ninjalj, я не знал.Это работа на DOS.Виноват.Спасибо за исправление.

Если вы пишете код для WinXP, использование ассемблера действительно мало интересует.Используйте C и встроенную сборку, если вы хотите для критических секций.К сожалению, я не знаю, как читать на физическом диске с помощью Win32API, но я уже видел это где-то, так что я догадываюсь, что это возможно ...

...