Delphi-PRAXiS
Seite 1 von 3  1 23      

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Win32/Win64 API (native code) (https://www.delphipraxis.net/17-win32-win64-api-native-code/)
-   -   Delphi Sektoren auslesen (https://www.delphipraxis.net/3281-sektoren-auslesen.html)

webby 4. Mär 2003 21:09


Sektoren auslesen
 
Hallo,
wie kann ich von einem Datenträger gezielt einen Sektor auslesen.
Ich möchte nämlich dann herausfinden, ob der entsprechende Sektor fehlerhaft ist.

sakura 5. Mär 2003 10:48

Für welches Betriebssystem und was für ein Dateiformat (NTFS, FAT16, FAT32, ...) Ist schon wichtig dass zu wissen, ist schliesslich immer wieder anders ;)

...:cat:...

webby 5. Mär 2003 12:40

Also wenn ich in die Windows API Sparte poste, werde ich als Betriebssystem Windows benutzen :) (oder hab ich die falsche Sparte erwischt?!).
Und die meisten Disketten nutzen ja FAT32.

Luckie 5. Mär 2003 12:56

Du hast nur "Datenträger" geschrieben und das kann auch eine Festplatte oder CD-ROM sein. :wink:

sakura 5. Mär 2003 14:53

Bei Disketten muss ich leider passen, sorry ;)

...:cat:...

jbg 5. Mär 2003 18:09

Zitat:

Zitat von webby
Also wenn ich in die Windows API Sparte poste, werde ich als Betriebssystem Windows benutzen

Das ist toll, aber welches Windows? Windows 9x/ME oder NT/2000/XP.
Für NT-basierte ist es um einiges einfacher als für 9x-basierte.

ab WinNT 3.5 (also kein 9x/ME)
Delphi-Quellcode:
uses Windows;
type
  TDiskGeometry = packed record
    Cylinders: Int64;
    MediaType: Integer;
    TracksPerCylinder: DWORD;
    SectorsPerTrack: DWORD;
    BytesPerSector: Integer; // wichtig für die Reservierung des Buffer-Speichers
  end;

  TRawDrive = record
    DiskGeometry: TDiskGeometry;
    Handle: THandle;
  end;

var
  IOCTL_DISK_GET_DRIVE_GEOMETRY = $00070000;
  FSCTL_LOCK_VOLUME = $00090018;
  FSCTL_UNLOCK_VOLUME = $0009001C;

function RawOpenDrive(DriveLetter: Char): TRawDrive;
var num: Cardinal;
begin
  FillChar(Result, SizeOf(TRawDrive), 0);

  Result.Handle := CreateFile(PChar('\\.\' + DriveLetter + ':'),
                       GENERIC_READ or GENERIC_WRITE,
                       FILE_SHARE_READ or FILE_SHARE_WRITE,
                       nil,
                       OPEN_EXISTING,
                       0,
                       0);
  if Result.Handle = INVALID_HANDLE_VALUE then RaiseLastOSError;

 // Laufwerk für andere sperren
  if not DeviceIoControl(Result, FSCTL_LOCK_VOLUME, nil, 0, nil, 0, num, nil) then
    RaiseLastOSError;

 { Disk-Geometry ermitteln. Vor allem ist das Feld BytesPerSector wichtig, da nur vielfache Bytes gelesen werden können. Ist also BytesPerSector z.B: 512 und man will nur 1 Byte lesen, muss man 512 Bytes lesen. }
  if not DeviceIoControl(h, IOCTL_DISK_GET_DRIVE_GEOMETRY, nil, 0, @Result.DiskGeometry,
                         SizeOf(TDiskGeometry), num, nil) then
    RaiseLastOSError;
end;

procedure RawCloseDrive(var RawDrive: TRawDrive);
var num: Cardinal;
begin
  DeviceIoControl(RawDrive.Handle, FSCTL_UNLOCK_VOLUME, nil, 0, nil, 0, num, nil);
  CloseHandle(RawDrive.Handle);
  RawDrive.Handle := 0;
end;

procedure RawReadSectors(const RawDrive: TRawDrive; var Buf; Count: Integer);
var num: Cardinal;
begin
  if not ReadFile(RawDrive.Handle, Buf, Count * RawDrive.DiskGeometry.BytesPerSector, num, nil) then
    RaiseLastOSError;
end;

procedure RawWriteSectors(const RawDrive: TRawDrive; var Buf; Count: Integer);
var num: Cardinal;
begin
  if not WriteFile(RawDrive.Handle, Buf, Count * RawDrive.DiskGeometry.BytesPerSector, num, nil) then
    RaiseLastOSError;
end;
Hier fehlt noch das RawSeek, jedoch brauchte ich bei meinem Diskettenkopierprogramm dieses nicht und habe es somit auch nicht programmiert. Die benötigte API Funktion wäre SetFilePointer. Hier ist aber auch wieder darauf zu achten, dass nur an Sektorgrenzen gesprungen werden kann.

webby 5. Mär 2003 18:29

Zitat:

Zitat von Luckie
Du hast nur "Datenträger" geschrieben und das kann auch eine Festplatte oder CD-ROM sein. :wink:

Stimmt, sorry, ich war davon überzeugt, dass ich Diskette geschrieben hätte...
Tja so kann man sich täuschen... ;)


Achso, stimmt ja in solche Sachen gehen 9x und NT anders ran.
Ne ich habe (leider) Windows XP Home Edition

@jbg:
Danke ich werde mir später den Code nochmal genau anschauen!

MadMax 24. Apr 2003 09:08

Genau das!!
 
Das ist genau das Programm das ich brauche.

Frage:
Muss das Prog als Konsolen-Anwendung kompiliert werden?

Das erste Problem tritt schon bei der Variablen Definition auf Delphi 5.0 checkt das irgendwie nicht.

Code:
var
  IOCTL_DISK_GET_DRIVE_GEOMETRY = $00070000;
  FSCTL_LOCK_VOLUME = $00090018;
  FSCTL_UNLOCK_VOLUME = $0009001C;
[Fehler] Project2.dpr(19): ',' oder ':' erwartet, aber '=' gefunden
[Fehler] Project2.dpr(19): '..' erwartet, aber ';' gefunden

usw....

Luckie 24. Apr 2003 11:01

Jeder aufmerksame Programmierer sieht, dass es sich hier:
Delphi-Quellcode:
IOCTL_DISK_GET_DRIVE_GEOMETRY = $00070000;
um keine Variabledeklaration, sondern um eine Konstantendeklaration handelt. Und jeder clevere Programmierer, kommt dann auf die Idee und ersetzt das var durch const. :wink:

War wohl der ScripptKiddie-CopyAndPaste-Schutz und du bist voll reingetappt in die Falle. :mrgreen:

nice 26. Jan 2004 16:45

Re: Sektoren auslesen
 
nabend erstmal

ich hätt mal ne frage zu dem source:
Delphi-Quellcode:
if not DeviceIoControl(Result, FSCTL_LOCK_VOLUME, nil, 0, nil, 0, num, nil) then
    RaiseLastOSError;
bei mir kommt bei der result ein fehler,es wäre eine Trawdrive - ist ja klar deklariert
er will aber an der stelle eine cardinal

wie kann ich das beheben ?

oder gibts ne schönere variante sektoren einer cd auszulesen(als rückgabe reicht auch defekt : true/false)
soll ein kleiner "kopierschutz" werden


Alle Zeitangaben in WEZ +1. Es ist jetzt 15:12 Uhr.
Seite 1 von 3  1 23      

Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz