View Full Version : [?] lettura hard disk
:help:
qualcono sa come si puo leggere o scrivere sull' hard disk in modalita protetta quindi senza utilizzare gli interrupt del bios (int 13h).
grazie per le vostre risposte
su Windows devi aprire l'unità fisica o logica utilizzando CreateFile, dopodiché leggi normalmente con ReadFile. nel caso di unità logica il formato del nome del file deve essere "\\\\.\\X:" dove X è la lettera dell'unità. nel caso invece di unità fisica il formato del nome deve essere "\\\\.\\PHYSICALDRIVEx", dove x è il numero dell'unità fisica.
su Windows devi aprire l'unità fisica o logica utilizzando CreateFile, dopodiché leggi normalmente con ReadFile. nel caso di unità logica il formato del nome del file deve essere "\\\\.\\X:" dove X è la lettera dell'unità. nel caso invece di unità fisica il formato del nome deve essere "\\\\.\\PHYSICALDRIVEx", dove x è il numero dell'unità fisica.
scusa non sono stato chiaro.
il mio problema è che sto provando a scrivere un "mini sistema operativo" a 32 bit
ora dovrei leggere e scrivere sull 'hard disk ma non so come fare perche l' int 13 non funziona a 32 bit
scusa non sono stato chiaro.
il mio problema è che sto provando a scrivere un "mini sistema operativo" a 32 bit
ora dovrei leggere e scrivere sull 'hard disk ma non so come fare perche l' int 13 non funziona a 32 bitAllora hai almeno 2 possibilità:
- Implementare un tuo driver, in protected-mode, che "parla" direttamente con il controller dei dischi.
- Switchare in real-mode, eseguire quello che ti serve e poi switchare nuovamente in protected-mode.
Ovviamente la soluzione migliore sarebbe la prima. ;)
DanieleC88
13-01-2007, 15:08
Allora hai almeno 2 possibilità:
- Implementare un tuo driver, in protected-mode, che "parla" direttamente con il controller dei dischi.
- Switchare in real-mode, eseguire quello che ti serve e poi switchare nuovamente in protected-mode.
Ovviamente la soluzione migliore sarebbe la prima. ;)
Concordo pienamente, anche perché la prima soluzione è abbastanza semplice nella sua implementazione (almeno, rispetto alla seconda).
Allego una guida rapida (non dettagliata, ma molto semplice).
grazie per la guida
ho provato a scrivere questo ma come immaginavo non funziona
:muro:
outp(0X1f6,0x0); //drive 0 head 0
outp(0X1f2,0x1); //scrivi un settore
outp(0X1f3,0x02); //scrivi dal 2 settore
outp(0X1f4,0x00); //cilindro 0
outp(0X1f5,0x00); //cilindro 0
outp(0X1f7,0x30); //comando
do
{
/* wait until BSY signal is set to a 1 and DRQ is set to a 1 */
} while ( !inp(0x1f7) );
printf("scrivo");
int i;
for (i=0;i<512;i++)
{
outp(0x1f0,10); /* output 16-bits at a time */
}
quali sono gli errori?
DanieleC88
14-01-2007, 13:13
Ci sono alcuni errori: ad esempio, non devi leggere la porta 0x1F7 per attendere che il drive sia pronto, ma la porta 0x3F6.
/* Attesa */
while ((inp(0x3F6) & (0x8 | 0x80)) != 0);
/* Cilindro */
outp(0x1F4, 0);
outp(0x1F5, 0);
/* Drive e testina */
outp(0x1F6, 0);
/* Numero di settori e settore iniziale */
outp(0x1F2, 1);
outp(0x1F3, 2);
/* Comando (lettura) */
outp(0x1F7, 0x20);
if ((inp(0x3F6) & 1) != 0) {
exit(-1);
}
/* Lettura di un settore di 512 byte, una word alla volta */
for (x = 0; x < (512 / 2); x++) {
buffer[x] = inp(0x1F0); // ** Attento: questo deve leggere una WORD, non un BYTE
}
Stai attento però, ad eccezione dell'ultimo inp() che deve leggere una word, tutti gli inp() o gli outp() devono leggere o scrivere byte. Meglio quindi se fai delle routine separate con nomi diversi (tipo inportb() per i byte e inportw() per le word).
L'esempio che ho scritto è per leggere dal disco, per scrivere è quasi lo stesso, cambia solo il comando (0x30) e l'ultima chiamata (ad esempio outportw() invece che inportw()).
ciao ;)
mi fa piacere vederti così preparato sull'argomento Daniele :D :D :D
Db., sto arrivando... :cry: :cry: :cry:
DanieleC88
14-01-2007, 15:04
:D
Faccio il possibile, ma ho ancora il solito ostacolo. Anzi, credo che chiederò qui sul forum... :cool:
grazie danielec88
ho provato funziona tutto perfettamente.
adesso stavo provando a scrivere una funzione che mi converte gli indirizzi LBA in CHS.
ho utilizzato queste formule per convertire
unsigned short cilindri = (LBA/SecPerTrk)/HeadsPerCyl;
unsigned char head = (LBA/SecPerTrk)% HeadsPerCyl;
unsigned char settore = (LBA % SecPerTrk)+1;
dove
HeadsPerCyl = 16
SecPerTrk = 255
pero fino a LBA 255 funziona dopo scrive da tutt'altra parte.
credo che le formule siano corrette ma le due costanti no
DanieleC88
16-01-2007, 12:50
Allora, se il disco supporta il LBA puoi usare anche indirizzi logici direttamente.
Per controllare se il disco supporta il LBA, manda il comando 0xEC al controller e leggi i 512 byte (256 word) che restituisce di ritorno, poi verifica che la sessantesima double word non sia uguale a zero:
DWORD info[128];
leggi_info(&info);
if (info[60] != 0) {
printf("LBA supportato");
}
Fatto questo, un codice del genere dovrebbe essere OK:
imposta_settore_iniziale(lba & 0xFF);
imposta_cilindro((lba & 0xFFFF00) >> 8);
imposta_drive_e_testina(HD0, (lba & 0xF000000) >> 24 | 0xE0); // questo 0xE0 dice al controller di usare il LBA
Altrimenti, per convertire in CHS fai una cosa come questa (comunque, in rete se fai una ricerca trovi un casino di risultati interessanti):
DWORD tmp = (lba % (DRIVEHEADS * DRIVESECTORS));
hd_set_sector_start(tmp % hd0.DRIVESECTORS + 1);
hd_set_cylinder(lba / (DRIVEHEADS * DRIVESECTORS));
hd_set_drive_and_head(HD0, (tmp / DRIVESECTORS));
ciao ;)
vBulletin® v3.6.4, Copyright ©2000-2025, Jelsoft Enterprises Ltd.