ReactOS 0.4.16-dev-937-g7afcd2a
hwidep.h
Go to the documentation of this file.
1/*
2 * PROJECT: FreeLoader
3 * LICENSE: MIT (https://spdx.org/licenses/MIT)
4 * PURPOSE: Private header for ATA/ATAPI programmed I/O driver.
5 * COPYRIGHT: Copyright 2019-2025 Dmitry Borisov (di.sean@protonmail.com)
6 */
7
8#pragma once
9
10#define TAG_ATA_DEVICE 'dATA'
11
12#define ATA_STATUS_SUCCESS 0
13#define ATA_STATUS_PENDING 1
14#define ATA_STATUS_ERROR 2
15#define ATA_STATUS_RESET 3
16#define ATA_STATUS_RETRY 4
17
18#if defined(SARCH_PC98)
19/* Master/Slave devices for Bank 0 and Bank 1 */
20#define CHANNEL_MAX_DEVICES 4
21#define DEV_SLAVE(DeviceNumber) ((DeviceNumber) & 1)
22#else
23/* Master/Slave devices */
24#define CHANNEL_MAX_DEVICES 2
25#define DEV_SLAVE(DeviceNumber) (DeviceNumber)
26#endif
27
28#if defined(SARCH_XBOX)
29/* It's safe to enable the multiple mode */
30#define ATA_ENABLE_MULTIPLE_MODE
31#endif
32
33/* Delay of 400ns */
34#if defined(SARCH_PC98)
35#define ATA_IO_WAIT() WRITE_PORT_UCHAR((PUCHAR)0x5F, 0)
36#else
37#define ATA_IO_WAIT() StallExecutionProcessor(1)
38#endif
39
40#if defined(_M_IX86) || defined(_M_AMD64)
41/* x86 I/O address space spans from 0 to 0xFFFF */
42typedef USHORT IDE_REG;
43#else
45#endif
46
47#define ATA_MAX_LBA_28 0x0FFFFFFFULL
48#define ATA_MAX_LBA_48 (1ULL << 48)
49
50#define IDE_FEATURE_PIO 0x00
51
52#define IDE_DC_ALWAYS 0x08
53
54#define IDE_DRIVE_SELECT 0xA0
55
56#define ATAPI_INT_REASON_COD 0x01
57#define ATAPI_INT_REASON_IO 0x02
58#define ATAPI_INT_REASON_MASK (ATAPI_INT_REASON_IO | ATAPI_INT_REASON_COD)
59
60#define ATAPI_INT_REASON_STATUS_NEC 0x00
61#define ATAPI_INT_REASON_STATUS (ATAPI_INT_REASON_IO | ATAPI_INT_REASON_COD)
62#define ATAPI_INT_REASON_AWAIT_CDB (IDE_STATUS_DRQ | ATAPI_INT_REASON_COD)
63#define ATAPI_INT_REASON_DATA_IN (ATAPI_INT_REASON_IO | IDE_STATUS_DRQ)
64
65#define MAXIMUM_CDROM_SIZE 804 // == sizeof(CDROM_TOC)
66
67#define ATA_TIME_BUSY_SELECT 2000
68#define ATA_TIME_BUSY_POLL 500000
69#define ATA_TIME_BUSY_ENUM 100
70#define ATA_TIME_BUSY_RESET 1000000
71#define ATA_TIME_RESET_SELECT 200000
72#define ATA_TIME_DRQ_CLEAR 100
73#define ATA_TIME_PHASE_CHANGE 100
74
75#define ATA_WRITE(Port, Value) \
76 WRITE_PORT_UCHAR((PUCHAR)(ULONG_PTR)(Port), (Value))
77
78#define ATA_WRITE_BLOCK_16(Port, Buffer, Count) \
79 WRITE_PORT_BUFFER_USHORT((PUSHORT)(ULONG_PTR)(Port), (PUSHORT)(Buffer), (Count))
80
81#define ATA_WRITE_BLOCK_32(Port, Buffer, Count) \
82 WRITE_PORT_BUFFER_ULONG((PULONG)(ULONG_PTR)(Port), (PULONG)(Buffer), (Count))
83
84#define ATA_READ(Port) \
85 READ_PORT_UCHAR((PUCHAR)(ULONG_PTR)(Port))
86
87#define ATA_READ_BLOCK_16(Port, Buffer, Count) \
88 READ_PORT_BUFFER_USHORT((PUSHORT)(ULONG_PTR)(Port), (PUSHORT)(Buffer), (Count))
89
90#define ATA_READ_BLOCK_32(Port, Buffer, Count) \
91 READ_PORT_BUFFER_ULONG((PULONG)(ULONG_PTR)(Port), (PULONG)(Buffer), (Count))
92
94{
99
100typedef struct _IDE_REGISTERS
101{
103 union
104 {
107 };
108 union
109 {
112 };
114 union
115 {
119 };
120 union
121 {
125 };
127 union
128 {
131 };
132 union
133 {
136 };
138
139typedef struct _ATA_TASKFILE
140{
143 struct
144 {
150 } Data[2]; // 0 - low part, 1 - high part
152
154{
155 union
156 {
159 };
162#define REQUEST_FLAG_LBA48 0x00000001
163#define REQUEST_FLAG_READ_WRITE_MULTIPLE 0x00000002
164#define REQUEST_FLAG_PACKET_COMMAND 0x00000004
165#define REQUEST_FLAG_SET_DEVICE_REGISTER 0x00000008
166#define REQUEST_FLAG_AWAIT_CDB 0x00000010
167#define REQUEST_FLAG_READ_COMMAND 0x00000020
168#define REQUEST_FLAG_IDENTIFY_COMMAND 0x00000040
169
172
173typedef struct _HW_DEVICE_UNIT
174{
175 /* Public data, must be the first member */
177
187 union
188 {
191 };
193
197 _In_ PIDENTIFY_DEVICE_DATA IdentifyData)
198{
199 ULONG i;
200 UCHAR Crc;
201
202 /* Bits 0:8 of word 255 */
203 if (IdentifyData->Signature != 0xA5)
204 {
205 /* The integrity word is missing, assume the data provided by the device is valid */
206 return TRUE;
207 }
208
209 /* Verify the checksum */
210 Crc = 0;
211 for (i = 0; i < sizeof(*IdentifyData); ++i)
212 {
213 Crc += ((PUCHAR)IdentifyData)[i];
214 }
215
216 return (Crc == 0);
217}
218
220UCHAR
222 _In_ PIDENTIFY_PACKET_DATA IdentifyPacketData)
223{
224 /* Bits 0:2 of word 0 */
225 return (IdentifyPacketData->GeneralConfiguration.PacketType != 0) ? 8 : 6;
226}
227
231 _In_ PIDENTIFY_DEVICE_DATA IdentifyData)
232{
233 /* Bit 9 of word 49 */
234 return IdentifyData->Capabilities.LbaSupported;
235}
236
238ULONG
240 _In_ PIDENTIFY_DEVICE_DATA IdentifyData)
241{
242 /* Words 60-61 */
243 return IdentifyData->UserAddressableSectors;
244}
245
249 _In_ PIDENTIFY_DEVICE_DATA IdentifyData)
250{
251 /* Words 100-103 */
252 return ((ULONG64)IdentifyData->Max48BitLBA[1] << 32) | IdentifyData->Max48BitLBA[0];
253}
254
258 _In_ PIDENTIFY_DEVICE_DATA IdentifyData)
259{
260 /* Word 83: 15 = 0, 14 = 1 */
261 if (IdentifyData->CommandSetSupport.WordValid83 == 1)
262 {
263 /* Bit 10 of word 83 */
264 return IdentifyData->CommandSetSupport.BigLba;
265 }
266
267 return FALSE;
268}
269
273 _In_ PIDENTIFY_DEVICE_DATA IdentifyData)
274{
275 return ((IdentifyData->TranslationFieldsValid & 1) &&
276 (IdentifyData->NumberOfCurrentCylinders != 0) &&
277 (IdentifyData->NumberOfCurrentCylinders <= 63) &&
278 (IdentifyData->NumberOfCurrentHeads != 0) &&
279 (IdentifyData->NumberOfCurrentHeads <= 16) &&
280 (IdentifyData->CurrentSectorsPerTrack != 0));
281}
282
284VOID
286 _In_ PIDENTIFY_DEVICE_DATA IdentifyData,
287 _Out_ PUSHORT Cylinders,
288 _Out_ PUSHORT Heads,
290{
291 /* Word 1 */
292 *Cylinders = IdentifyData->NumCylinders;
293
294 /* Word 3 */
295 *Heads = IdentifyData->NumHeads;
296
297 /* Word 6 */
298 *SectorsPerTrack = IdentifyData->NumSectorsPerTrack;
299}
300
302VOID
304 _In_ PIDENTIFY_DEVICE_DATA IdentifyData,
305 _Out_ PUSHORT Cylinders,
306 _Out_ PUSHORT Heads,
308{
309 /* Word 54 */
310 *Cylinders = IdentifyData->NumberOfCurrentCylinders;
311
312 /* Word 55 */
313 *Heads = IdentifyData->NumberOfCurrentHeads;
314
315 /* Word 55 */
316 *SectorsPerTrack = IdentifyData->CurrentSectorsPerTrack;
317}
318
320UCHAR
322 _In_ PIDENTIFY_DEVICE_DATA IdentifyData)
323{
324 UCHAR MultiSectorCurrent;
325
326 /* Bit 8 of word 59 */
327 if (!(IdentifyData->MultiSectorSettingValid))
328 return 0;
329
330 /* The word 59 should be a power of 2 */
331 MultiSectorCurrent = IdentifyData->CurrentMultiSectorSetting;
332
333 if ((MultiSectorCurrent > 0) && ((MultiSectorCurrent & (MultiSectorCurrent - 1)) == 0))
334 return MultiSectorCurrent;
335
336 return 0;
337}
338
340UCHAR
342 _In_ PIDENTIFY_DEVICE_DATA IdentifyData)
343{
344 UCHAR MultiSectorMax;
345
346 /* The word 47 should be a power of 2 */
347 MultiSectorMax = IdentifyData->MaximumBlockTransfer;
348
349 if ((MultiSectorMax > 0) && ((MultiSectorMax & (MultiSectorMax - 1)) == 0))
350 return MultiSectorMax;
351
352 return 0;
353}
354
356ULONG
358 _In_ PIDENTIFY_DEVICE_DATA IdentifyData)
359{
360 ULONG WordCount;
361
362 /* Word 106: 15 = 0, 14 = 1, 12 = 1 */
363 if (IdentifyData->PhysicalLogicalSectorSize.Reserved1 == 1 &&
364 IdentifyData->PhysicalLogicalSectorSize.LogicalSectorLongerThan256Words)
365 {
366 /* Words 116-117 */
367 WordCount = IdentifyData->WordsPerLogicalSector[0];
368 WordCount |= (ULONG)IdentifyData->WordsPerLogicalSector[1] << 16;
369 }
370 else
371 {
372 /* 256 words = 512 bytes */
373 WordCount = 256;
374 }
375
376 return WordCount * sizeof(USHORT);
377}
378
382 _In_ ULONG64 SectorNumber,
384{
385 /* Use the 48-bit command when reasonable */
386 return (((SectorNumber + SectorCount) >= ATA_MAX_LBA_28) || (SectorCount > 0x100));
387}
unsigned char BOOLEAN
#define TRUE
Definition: types.h:120
#define FALSE
Definition: types.h:117
GLsizei GLenum const GLvoid GLsizei GLenum GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLint GLint GLint GLshort GLshort GLshort GLubyte GLubyte GLubyte GLuint GLuint GLuint GLushort GLushort GLushort GLbyte GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLfloat GLint GLint GLint GLint GLshort GLshort GLshort GLshort GLubyte GLubyte GLubyte GLubyte GLuint GLuint GLuint GLuint GLushort GLushort GLushort GLushort GLboolean const GLdouble const GLfloat const GLint const GLshort const GLbyte const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLdouble const GLfloat const GLfloat const GLint const GLint const GLshort const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort GLenum GLenum GLenum GLfloat GLenum GLint GLenum GLenum GLenum GLfloat GLenum GLenum GLint GLenum GLfloat GLenum GLint GLint GLushort GLenum GLenum GLfloat GLenum GLenum GLint GLfloat const GLubyte GLenum GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLint GLint GLsizei GLsizei GLint GLenum GLenum const GLvoid GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLenum const GLdouble GLenum GLenum const GLfloat GLenum GLenum const GLint GLsizei GLuint GLfloat GLuint GLbitfield GLfloat GLint GLuint GLboolean GLenum GLfloat GLenum GLbitfield GLenum GLfloat GLfloat GLint GLint const GLfloat GLenum GLfloat GLfloat GLint GLint GLfloat GLfloat GLint GLint const GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat const GLdouble const GLfloat const GLdouble const GLfloat GLint i
Definition: glfuncs.h:248
FORCEINLINE VOID AtaDevDefaultChsTranslation(_In_ PIDENTIFY_DEVICE_DATA IdentifyData, _Out_ PUSHORT Cylinders, _Out_ PUSHORT Heads, _Out_ PUSHORT SectorsPerTrack)
Definition: hwidep.h:285
struct _IDE_REGISTERS * PIDE_REGISTERS
FORCEINLINE ULONG AtaDevBytesPerLogicalSector(_In_ PIDENTIFY_DEVICE_DATA IdentifyData)
Definition: hwidep.h:357
enum _ATA_DEVICE_CLASS ATA_DEVICE_CLASS
struct _HW_DEVICE_UNIT * PHW_DEVICE_UNIT
FORCEINLINE BOOLEAN AtaDevIsIdentifyDataValid(_In_ PIDENTIFY_DEVICE_DATA IdentifyData)
Definition: hwidep.h:196
struct _IDE_REGISTERS IDE_REGISTERS
FORCEINLINE UCHAR AtaDevCurrentSectorsPerDrq(_In_ PIDENTIFY_DEVICE_DATA IdentifyData)
Definition: hwidep.h:321
FORCEINLINE BOOLEAN AtaCommandUseLba48(_In_ ULONG64 SectorNumber, _In_ ULONG SectorCount)
Definition: hwidep.h:381
FORCEINLINE BOOLEAN AtaDevHas48BitAddressFeature(_In_ PIDENTIFY_DEVICE_DATA IdentifyData)
Definition: hwidep.h:257
_ATA_DEVICE_CLASS
Definition: hwidep.h:94
@ DEV_ATA
Definition: hwidep.h:95
@ DEV_ATAPI
Definition: hwidep.h:96
@ DEV_NONE
Definition: hwidep.h:97
struct _ATA_DEVICE_REQUEST ATA_DEVICE_REQUEST
FORCEINLINE ULONG64 AtaDevUserAddressableSectors48Bit(_In_ PIDENTIFY_DEVICE_DATA IdentifyData)
Definition: hwidep.h:248
enum _ATA_DEVICE_CLASS * PATA_DEVICE_CLASS
FORCEINLINE BOOLEAN AtaDevHasLbaTranslation(_In_ PIDENTIFY_DEVICE_DATA IdentifyData)
Definition: hwidep.h:230
FORCEINLINE ULONG AtaDevUserAddressableSectors28Bit(_In_ PIDENTIFY_DEVICE_DATA IdentifyData)
Definition: hwidep.h:239
FORCEINLINE UCHAR AtaDevMaximumSectorsPerDrq(_In_ PIDENTIFY_DEVICE_DATA IdentifyData)
Definition: hwidep.h:341
#define ATA_MAX_LBA_28
Definition: hwidep.h:47
ULONG_PTR IDE_REG
Definition: hwidep.h:44
FORCEINLINE BOOLEAN AtaDevIsCurrentGeometryValid(_In_ PIDENTIFY_DEVICE_DATA IdentifyData)
Definition: hwidep.h:272
struct _ATA_TASKFILE ATA_TASKFILE
struct _ATA_TASKFILE * PATA_TASKFILE
FORCEINLINE VOID AtaDevCurrentChsTranslation(_In_ PIDENTIFY_DEVICE_DATA IdentifyData, _Out_ PUSHORT Cylinders, _Out_ PUSHORT Heads, _Out_ PUSHORT SectorsPerTrack)
Definition: hwidep.h:303
struct _HW_DEVICE_UNIT HW_DEVICE_UNIT
struct _ATA_DEVICE_REQUEST * PATA_DEVICE_REQUEST
FORCEINLINE UCHAR AtaDevCdbSizeInWords(_In_ PIDENTIFY_PACKET_DATA IdentifyPacketData)
Definition: hwidep.h:221
unsigned __int64 ULONG64
Definition: imports.h:198
#define _Out_
Definition: no_sal2.h:160
#define _In_
Definition: no_sal2.h:158
ULONG SectorCount
Definition: part_xbox.c:31
unsigned short USHORT
Definition: pedump.c:61
ATA_TASKFILE TaskFile
Definition: hwidep.h:158
UCHAR Cdb[16]
Definition: hwidep.h:157
ULONG DataTransferLength
Definition: hwidep.h:170
UCHAR SectorCount
Definition: hwidep.h:146
UCHAR Feature
Definition: hwidep.h:145
UCHAR MidLba
Definition: hwidep.h:148
UCHAR HighLba
Definition: hwidep.h:149
UCHAR Command
Definition: hwidep.h:142
UCHAR DriveSelect
Definition: hwidep.h:141
UCHAR LowLba
Definition: hwidep.h:147
Data structure for the ATA device.
Definition: hwide.h:12
ULONG MaximumTransferLength
Definition: hwidep.h:182
ULONG DrqByteCount
Definition: hwidep.h:181
ULONG BytesToTransfer
Definition: hwidep.h:179
UCHAR DeviceNumber
Definition: hwidep.h:183
UCHAR CdbSize
Definition: hwidep.h:185
UCHAR MultiSectorTransfer
Definition: hwidep.h:186
DEVICE_UNIT P
Definition: hwidep.h:176
UCHAR DeviceSelect
Definition: hwidep.h:184
IDENTIFY_PACKET_DATA IdentifyPacketData
Definition: hwidep.h:190
IDE_REGISTERS Registers
Definition: hwidep.h:178
IDENTIFY_DEVICE_DATA IdentifyDeviceData
Definition: hwidep.h:189
PUCHAR DataBuffer
Definition: hwidep.h:180
IDE_REG Error
Definition: hwidep.h:106
IDE_REG SignatureLow
Definition: hwidep.h:118
IDE_REG Control
Definition: hwidep.h:134
IDE_REG LbaLow
LBA bits 0-7, 24-31.
Definition: hwidep.h:113
IDE_REG LbaMid
LBA bits 8-15, 32-39.
Definition: hwidep.h:116
IDE_REG InterruptReason
Definition: hwidep.h:111
IDE_REG Data
Definition: hwidep.h:102
IDE_REG Command
Definition: hwidep.h:129
IDE_REG SignatureHigh
Definition: hwidep.h:124
IDE_REG Status
Definition: hwidep.h:130
IDE_REG LbaHigh
LBA bits 16-23, 40-47.
Definition: hwidep.h:122
IDE_REG Device
Definition: hwidep.h:126
IDE_REG ByteCountHigh
Definition: hwidep.h:123
IDE_REG AlternateStatus
Definition: hwidep.h:135
IDE_REG SectorCount
Definition: hwidep.h:110
IDE_REG ByteCountLow
Definition: hwidep.h:117
IDE_REG Features
Definition: hwidep.h:105
uint16_t * PUSHORT
Definition: typedefs.h:56
uint32_t ULONG_PTR
Definition: typedefs.h:65
unsigned char * PUCHAR
Definition: typedefs.h:53
uint32_t ULONG
Definition: typedefs.h:59
#define FORCEINLINE
Definition: wdftypes.h:67
_In_ ULONG _In_ ULONG SectorsPerTrack
Definition: iofuncs.h:2071
unsigned char UCHAR
Definition: xmlstorage.h:181