4 * Created on: Oct 27, 2008
9 * This file contains exec style device, usbmss.device, which is available as
10 * soon as the mass storage class gets initialized. The device translates the
11 * trackdisk commands into mass storage methods.
13 #include <aros/symbolsets.h>
14 #include <aros/macros.h>
16 #include <exec/types.h>
17 #include <exec/nodes.h>
18 #include <exec/devices.h>
20 #include <exec/errors.h>
21 #include <devices/trackdisk.h>
22 #include <devices/scsidisk.h>
23 #include <devices/newstyle.h>
26 #include <dos/dosextens.h>
28 #include <hidd/hidd.h>
30 #include <usb/storage.h>
34 #include <proto/exec.h>
35 #include <proto/oop.h>
36 #include <proto/dos.h>
37 #include <aros/debug.h>
40 #include LC_LIBDEFS_FILE
43 #define SD(x) (&LIBBASE->sd)
45 /* Invalid command - reports error */
46 static void cmd_Invalid(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
48 D(bug("[MSS-dev] Invalid command %d for unit %04x\n", io
->io_Command
, unit
->msu_unitNum
));
49 io
->io_Error
= IOERR_NOCMD
;
52 /* NULL commad - always success. */
53 static void cmd_NULL(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
55 IOStdReq(io
)->io_Actual
= 0;
58 static void cmd_Reset(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
60 HIDD_USBStorage_Reset(unit
->msu_object
);
61 IOStdReq(io
)->io_Actual
= 0;
64 static void cmd_Remove(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
66 if (unit
->msu_removeInt
)
67 io
->io_Error
= TDERR_DriveInUse
;
69 unit
->msu_removeInt
= (struct Interrupt
*)IOStdReq(io
)->io_Data
;
72 static void cmd_AddChangeInt(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
75 AddHead(&unit
->msu_diskChangeList
, (struct Node
*)io
);
78 io
->io_Flags
&= ~IOF_QUICK
;
79 unit
->msu_unit
.unit_flags
&= ~UNITF_ACTIVE
;
82 static void cmd_RemChangeInt(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
85 Remove((struct Node
*)io
);
89 static void cmd_ChangeNum(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
91 IOStdReq(io
)->io_Actual
= unit
->msu_changeNum
;
94 static void cmd_ProtStatus(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
96 IOStdReq(io
)->io_Actual
= ((unit
->msu_inquiry
[0] & 0x1f) == DG_CDROM
) ? -1 : 0;
99 static void cmd_ChangeState(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
101 IOStdReq(io
)->io_Actual
= unit
->msu_flags
& MSF_DiskPresent
? 0:1;
104 static void cmd_GetGeometry(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
106 if (!(unit
->msu_flags
& MSF_DiskPresent
))
108 io
->io_Error
= TDERR_DiskChanged
;
110 else if (IOStdReq(io
)->io_Length
== sizeof(struct DriveGeometry
))
112 struct DriveGeometry
*dg
= (struct DriveGeometry
*)IOStdReq(io
)->io_Data
;
114 dg
->dg_SectorSize
= unit
->msu_blockSize
;
115 dg
->dg_TotalSectors
= unit
->msu_blockCount
;
116 dg
->dg_Cylinders
= unit
->msu_blockCount
/ (255*63);
118 dg
->dg_TrackSectors
= 63;
119 dg
->dg_CylSectors
= 255*63;
120 dg
->dg_BufMemType
= MEMF_PUBLIC
;
121 dg
->dg_DeviceType
= unit
->msu_inquiry
[0] & 0x1f;
122 dg
->dg_Flags
= (dg
->dg_DeviceType
== DG_CDROM
) ? DGF_REMOVABLE
: 0;
125 IOStdReq(io
)->io_Actual
= sizeof(struct DriveGeometry
);
128 io
->io_Error
= TDERR_NotSpecified
;
131 static void cmd_Read32(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
133 uint32_t block
= IOStdReq(io
)->io_Offset
;
134 uint32_t count
= IOStdReq(io
)->io_Length
;
136 if (!(unit
->msu_flags
& MSF_DiskPresent
))
138 io
->io_Error
= TDERR_DiskChanged
;
142 // D(bug("[%s] Read32(%08x, %08x)\n", FindTask(NULL)->tc_Node.ln_Name, IOStdReq(io)->io_Offset, IOStdReq(io)->io_Length));
144 if (!IOStdReq(io
)->io_Data
)
146 D(bug("[%s] No buffer!\n", FindTask(NULL
)->tc_Node
.ln_Name
));
147 io
->io_Error
= IOERR_BADADDRESS
;
149 else if ((block
& ((1 << unit
->msu_blockShift
) - 1)) || (count
& ((1 << unit
->msu_blockShift
) - 1)))
151 D(bug("[%s] offset or length not sector-aligned\n", FindTask(NULL
)->tc_Node
.ln_Name
));
152 cmd_Invalid(io
, dev
, unit
);
156 block
>>= unit
->msu_blockShift
;
157 count
>>= unit
->msu_blockShift
;
159 if (block
+ count
- 1> unit
->msu_blockCount
)
161 D(bug("[%s] ERROR! Requested read outside the available area\n", FindTask(NULL
)->tc_Node
.ln_Name
));
162 io
->io_Error
= IOERR_BADADDRESS
;
166 if (!HIDD_USBStorage_Read(unit
->msu_object
, unit
->msu_lun
, IOStdReq(io
)->io_Data
, block
, count
))
168 D(bug("[%s] READ ERROR: block=0x%08x count=0x%08x\n", FindTask(NULL
)->tc_Node
.ln_Name
, block
, count
));
170 HIDD_USBStorage_Reset(unit
->msu_object
);
172 if (!HIDD_USBStorage_Read(unit
->msu_object
, unit
->msu_lun
, IOStdReq(io
)->io_Data
, block
, count
))
174 IOStdReq(io
)->io_Error
= TDERR_NotSpecified
;
175 IOStdReq(io
)->io_Actual
= 0;
180 IOStdReq(io
)->io_Actual
= IOStdReq(io
)->io_Length
;
186 static void cmd_Read64(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
188 uint64_t block
= (uint64_t)(IOStdReq(io
)->io_Offset
) | ((uint64_t)(IOStdReq(io
)->io_Actual
) << 32);
189 uint32_t count
= IOStdReq(io
)->io_Length
;
191 if (!(unit
->msu_flags
& MSF_DiskPresent
))
193 io
->io_Error
= TDERR_DiskChanged
;
198 // D(bug("[%s] Read64(%04x%08x, %08x)\n", FindTask(NULL)->tc_Node.ln_Name, IOStdReq(io)->io_Actual, IOStdReq(io)->io_Offset, IOStdReq(io)->io_Length));
202 if (!IOStdReq(io
)->io_Data
)
204 D(bug("[%s] No buffer!\n", FindTask(NULL
)->tc_Node
.ln_Name
));
205 io
->io_Error
= IOERR_BADADDRESS
;
207 else if ((block
& ((1 << unit
->msu_blockShift
) - 1)) || (count
& ((1 << unit
->msu_blockShift
) - 1)))
209 D(bug("[%s] offset or length not sector-aligned\n", FindTask(NULL
)->tc_Node
.ln_Name
));
210 cmd_Invalid(io
, dev
, unit
);
214 block
>>= unit
->msu_blockShift
;
215 count
>>= unit
->msu_blockShift
;
217 if (block
+ count
- 1> unit
->msu_blockCount
)
219 D(bug("[%s] ERROR! Requested read outside the available area\n", FindTask(NULL
)->tc_Node
.ln_Name
));
220 io
->io_Error
= IOERR_BADADDRESS
;
224 if (!HIDD_USBStorage_Read(unit
->msu_object
, unit
->msu_lun
, IOStdReq(io
)->io_Data
, block
, count
))
226 D(bug("[%s] READ ERROR: block=0x%08x count=0x%08x\n", FindTask(NULL
)->tc_Node
.ln_Name
, (int32_t)block
, count
));
228 HIDD_USBStorage_Reset(unit
->msu_object
);
230 if (!HIDD_USBStorage_Read(unit
->msu_object
, unit
->msu_lun
, IOStdReq(io
)->io_Data
, block
, count
))
232 IOStdReq(io
)->io_Error
= TDERR_NotSpecified
;
233 IOStdReq(io
)->io_Actual
= 0;
238 IOStdReq(io
)->io_Actual
= IOStdReq(io
)->io_Length
;
244 static void cmd_Write32(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
246 uint32_t block
= IOStdReq(io
)->io_Offset
;
247 uint32_t count
= IOStdReq(io
)->io_Length
;
249 if (!(unit
->msu_flags
& MSF_DiskPresent
))
251 io
->io_Error
= TDERR_DiskChanged
;
256 // D(bug("[%s] Write32(%08x, %08x)\n", FindTask(NULL)->tc_Node.ln_Name, IOStdReq(io)->io_Offset, IOStdReq(io)->io_Length));
258 if (!IOStdReq(io
)->io_Data
)
260 D(bug("[%s] No buffer!\n", FindTask(NULL
)->tc_Node
.ln_Name
));
261 io
->io_Error
= IOERR_BADADDRESS
;
263 else if ((block
& ((1 << unit
->msu_blockShift
) - 1)) || (count
& ((1 << unit
->msu_blockShift
) - 1)))
265 D(bug("[%s] offset or length not sector-aligned\n", FindTask(NULL
)->tc_Node
.ln_Name
));
266 cmd_Invalid(io
, dev
, unit
);
270 block
>>= unit
->msu_blockShift
;
271 count
>>= unit
->msu_blockShift
;
273 if (block
+ count
- 1> unit
->msu_blockCount
)
275 D(bug("[%s] ERROR! Requested write outside the available area\n", FindTask(NULL
)->tc_Node
.ln_Name
));
276 io
->io_Error
= IOERR_BADADDRESS
;
280 if (!HIDD_USBStorage_Write(unit
->msu_object
, unit
->msu_lun
, IOStdReq(io
)->io_Data
, block
, count
))
282 D(bug("[%s] WRITE ERROR: block=0x%08x count=0x%08x\n", FindTask(NULL
)->tc_Node
.ln_Name
, block
, count
));
284 HIDD_USBStorage_Reset(unit
->msu_object
);
286 if (!HIDD_USBStorage_Write(unit
->msu_object
, unit
->msu_lun
, IOStdReq(io
)->io_Data
, block
, count
))
288 IOStdReq(io
)->io_Error
= TDERR_NotSpecified
;
289 IOStdReq(io
)->io_Actual
= 0;
294 IOStdReq(io
)->io_Actual
= IOStdReq(io
)->io_Length
;
300 static void cmd_Write64(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
302 uint64_t block
= (uint64_t)(IOStdReq(io
)->io_Offset
) | ((uint64_t)(IOStdReq(io
)->io_Actual
) << 32);
303 uint32_t count
= IOStdReq(io
)->io_Length
;
305 if (!(unit
->msu_flags
& MSF_DiskPresent
))
307 io
->io_Error
= TDERR_DiskChanged
;
311 // D(bug("[%s] Write64(%04x%08x, %08x)\n", FindTask(NULL)->tc_Node.ln_Name, IOStdReq(io)->io_Actual, IOStdReq(io)->io_Offset, IOStdReq(io)->io_Length));
315 if (!IOStdReq(io
)->io_Data
)
317 D(bug("[%s] No buffer!\n", FindTask(NULL
)->tc_Node
.ln_Name
));
318 io
->io_Error
= IOERR_BADADDRESS
;
320 else if ((block
& ((1 << unit
->msu_blockShift
) - 1)) || (count
& ((1 << unit
->msu_blockShift
) - 1)))
322 D(bug("[%s] offset or length not sector-aligned\n", FindTask(NULL
)->tc_Node
.ln_Name
));
323 cmd_Invalid(io
, dev
, unit
);
327 block
>>= unit
->msu_blockShift
;
328 count
>>= unit
->msu_blockShift
;
330 if (block
+ count
- 1 > unit
->msu_blockCount
)
332 D(bug("[%s] ERROR! Requested write outside the available area\n", FindTask(NULL
)->tc_Node
.ln_Name
));
333 io
->io_Error
= IOERR_BADADDRESS
;
337 if (!HIDD_USBStorage_Write(unit
->msu_object
, unit
->msu_lun
, IOStdReq(io
)->io_Data
, block
, count
))
339 D(bug("[%s] WRITE ERROR: block=0x%08x count=0x%08x\n", FindTask(NULL
)->tc_Node
.ln_Name
, (int32_t)block
, count
));
341 HIDD_USBStorage_Reset(unit
->msu_object
);
343 if (!HIDD_USBStorage_Write(unit
->msu_object
, unit
->msu_lun
, IOStdReq(io
)->io_Data
, block
, count
))
345 IOStdReq(io
)->io_Error
= TDERR_NotSpecified
;
346 IOStdReq(io
)->io_Actual
= 0;
351 IOStdReq(io
)->io_Actual
= IOStdReq(io
)->io_Length
;
357 static void cmd_DirectSCSI(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
)
359 struct SCSICmd
*cmd
= (struct SCSICmd
*)IOStdReq(io
)->io_Data
;
361 IOStdReq(io
)->io_Error
= 0;
364 D(bug("[%s] DirectSCSI(", FindTask(NULL
)->tc_Node
.ln_Name
));
365 for (i
=0; i
< cmd
->scsi_CmdLength
; i
++)
367 D(bug("%02x%c", cmd
->scsi_Command
[i
], i
< (cmd
->scsi_CmdLength
-1) ? ',':')'));
373 cmd
->scsi_Actual
= 0;
374 cmd
->scsi_Status
= 0;
377 Handle the USB SCSI quirks. The code is bad because it assumes the
378 cmd->scsi_Command is writable
380 switch(cmd
->scsi_Command
[0])
382 case 0x12: /* INQUIRY: USB devices deliver the first 36 bytes only */
383 if (cmd
->scsi_Command
[4] > 36)
385 cmd
->scsi_Command
[4] = 36;
386 cmd
->scsi_Length
= 36;
390 case 0x5a: /* MODE SENSE 10: Only 8 bytes are delivered */
391 if (((cmd
->scsi_Command
[7] << 8) | cmd
->scsi_Command
[8]) > 8)
393 cmd
->scsi_Command
[7] = 0;
394 cmd
->scsi_Command
[8] = 8;
395 cmd
->scsi_Length
= 8;
399 case 0x03: /* REQUEST SENSE: USB devices deliver up to 18 bytes only*/
400 if (cmd
->scsi_Command
[4] > 18)
402 cmd
->scsi_Command
[4] = 18;
403 cmd
->scsi_Length
= 18;
407 case 0xec: /* Hack to issue ATA Inquiry call. Ignore it. */
408 if (cmd
->scsi_CmdLength
== 1)
410 IOStdReq(io
)->io_Error
= IOERR_NOCMD
;
416 if (!HIDD_USBStorage_DirectSCSI(unit
->msu_object
, unit
->msu_lun
, cmd
->scsi_Command
, cmd
->scsi_CmdLength
, cmd
->scsi_Data
, cmd
->scsi_Length
, cmd
->scsi_Flags
& SCSIF_READ
))
419 char *c
= cmd
->scsi_Command
;
420 D(bug("[MSS-dev] DirectSCSI failed: "));
422 for (i
=0; i
< cmd
->scsi_CmdLength
; i
++)
423 D(bug("%02x ", c
[i
]));
425 D(bug("data=%08x len=%08x\n", cmd
->scsi_Data
, cmd
->scsi_Length
));
427 IOStdReq(io
)->io_Error
= HFERR_BadStatus
;
429 cmd
->scsi_Status
= 2;
431 if (cmd
->scsi_Flags
& SCSIF_AUTOSENSE
)
433 if (cmd
->scsi_SenseData
&& cmd
->scsi_SenseLength
> 0)
437 if (HIDD_USBStorage_RequestSense(unit
->msu_object
, unit
->msu_lun
, sense
, 18))
439 if (cmd
->scsi_SenseLength
>= 18)
441 CopyMem(sense
, cmd
->scsi_SenseData
, 18);
442 cmd
->scsi_SenseActual
= 18;
446 CopyMem(sense
, cmd
->scsi_SenseData
, cmd
->scsi_SenseLength
);
447 cmd
->scsi_SenseActual
= cmd
->scsi_SenseLength
;
455 IOStdReq(io
)->io_Error
= 0;
456 cmd
->scsi_Actual
= cmd
->scsi_Length
;
457 IOStdReq(io
)->io_Actual
= IOStdReq(io
)->io_Length
;
462 static const uint16_t supported_commands
[] = {
463 CMD_RESET
, CMD_READ
, CMD_WRITE
, CMD_UPDATE
, CMD_CLEAR
, CMD_START
, CMD_STOP
, CMD_FLUSH
,
464 TD_MOTOR
, TD_SEEK
, TD_FORMAT
, TD_REMOVE
, TD_CHANGENUM
, TD_CHANGESTATE
, TD_PROTSTATUS
,
465 TD_GETNUMTRACKS
, TD_ADDCHANGEINT
, TD_REMCHANGEINT
, TD_GETGEOMETRY
, TD_EJECT
,
466 TD_READ64
, TD_WRITE64
, TD_SEEK64
, TD_FORMAT64
,
467 HD_SCSICMD
, TD_GETDRIVETYPE
,
468 NSCMD_DEVICEQUERY
, NSCMD_TD_READ64
, NSCMD_TD_WRITE64
, NSCMD_TD_SEEK64
, NSCMD_TD_FORMAT64
,
472 static void (*map32
[HD_SCSICMD
+1])(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
) = {
473 [CMD_INVALID
] = cmd_Invalid
,
474 [CMD_RESET
] = cmd_Reset
,
475 [CMD_READ
] = cmd_Read32
,
476 [CMD_WRITE
] = cmd_Write32
,
477 [CMD_UPDATE
] = cmd_Reset
,
478 [CMD_CLEAR
] = cmd_Reset
,
479 [CMD_STOP
] = cmd_Reset
,
480 [CMD_START
] = cmd_Reset
,
481 [CMD_FLUSH
] = cmd_Reset
, /* Implement flush! */
482 [TD_MOTOR
] = cmd_NULL
,
483 [TD_SEEK
] = cmd_NULL
,
484 [TD_FORMAT
] = cmd_Write32
,
485 [TD_REMOVE
] = cmd_Remove
,
486 [TD_CHANGENUM
] = cmd_ChangeNum
,
487 [TD_CHANGESTATE
] = cmd_ChangeState
,
488 [TD_PROTSTATUS
] = cmd_ProtStatus
,
489 [TD_RAWREAD
] = cmd_Invalid
,
490 [TD_RAWWRITE
] = cmd_Invalid
,
491 [TD_GETNUMTRACKS
] = cmd_Invalid
,
492 [TD_ADDCHANGEINT
] = cmd_AddChangeInt
,
493 [TD_REMCHANGEINT
] = cmd_RemChangeInt
,
494 [TD_GETGEOMETRY
] = cmd_GetGeometry
,
495 [TD_EJECT
] = cmd_Invalid
,
496 [TD_READ64
] = cmd_Read64
,
497 [TD_WRITE64
] = cmd_Write64
,
498 [TD_SEEK64
] = cmd_NULL
,
499 [TD_FORMAT64
] = cmd_Write64
,
500 [HD_SCSICMD
] = cmd_DirectSCSI
,
503 static void (*map64
[4])(struct IORequest
*io
, mss_device_t
*dev
, mss_unit_t
*unit
) = {
504 [NSCMD_TD_READ64
- 0xc000] = cmd_Read64
,
505 [NSCMD_TD_WRITE64
- 0xc000] = cmd_Write64
,
506 [NSCMD_TD_SEEK64
- 0xc000] = cmd_NULL
,
507 [NSCMD_TD_FORMAT64
- 0xc000] = cmd_Write64
,
511 * IO handler. It may be either called directly, or through the mass storage
514 void HandleIO(struct IORequest
*io
, mss_device_t
*device
, mss_unit_t
*unit
)
516 unit
->msu_unit
.unit_flags
|= UNITF_ACTIVE
;
520 /* Some commands will be handled directly */
521 switch (io
->io_Command
)
524 New Style Devices query. Introduce self as trackdisk and provide list of
527 case NSCMD_DEVICEQUERY
:
529 struct NSDeviceQueryResult
*nsdq
= (struct NSDeviceQueryResult
*)IOStdReq(io
)->io_Data
;
530 nsdq
->DevQueryFormat
= 0;
531 nsdq
->SizeAvailable
= sizeof(struct NSDeviceQueryResult
);
532 nsdq
->DeviceType
= NSDEVTYPE_TRACKDISK
;
533 nsdq
->DeviceSubType
= 0;
534 nsdq
->SupportedCommands
= (uint16_t *)supported_commands
;
535 IOStdReq(io
)->io_Actual
= sizeof(struct NSDeviceQueryResult
);
540 New Style Devices report here the 'NSTY' - only if such value is
541 returned here, the NSCMD_DEVICEQUERY might be called. Otherwice it should
544 case TD_GETDRIVETYPE
:
545 IOStdReq(io
)->io_Actual
= DRIVE_NEWSTYLE
;
549 if (io
->io_Command
<= HD_SCSICMD
)
551 if (map32
[io
->io_Command
])
552 map32
[io
->io_Command
](io
, device
, unit
);
554 cmd_Invalid(io
, device
, unit
);
556 else if (io
->io_Command
>= NSCMD_TD_READ64
&& io
->io_Command
<= NSCMD_TD_FORMAT64
)
558 if (map64
[io
->io_Command
- NSCMD_TD_READ64
])
559 map64
[io
->io_Command
- NSCMD_TD_READ64
](io
, device
, unit
);
561 cmd_Invalid(io
, device
, unit
);
566 unit
->msu_unit
.unit_flags
&= ~(UNITF_ACTIVE
);
572 * Exec device functions
576 AROS_LH3 (void, dev_OpenLib
,
577 AROS_LHA(struct IORequest
*, ioreq
, A1
),
578 AROS_LHA(ULONG
, unitnum
, D0
),
579 AROS_LHA(ULONG
, flags
, D1
),
580 mss_device_t
*, lh
, 1, Storage
)
584 mss_unit_t
*unit
, *found
= NULL
;
586 ioreq
->io_Error
= IOERR_OPENFAIL
;
588 D(bug("[MSS-dev] Open(%p, %x, %x)\n", ioreq
, unitnum
, flags
));
590 /* Lock the whole mass storage class and search for a requested unit */
591 ObtainSemaphore(&lh
->mss_static
->Lock
);
592 ForeachNode(&lh
->mss_static
->unitList
, unit
)
594 if (unit
->msu_unitNum
== unitnum
)
602 ioreq
->io_Device
= &lh
->mss_device
;
603 ioreq
->io_Unit
= (struct Unit
*)found
;
606 found
->msu_unit
.unit_OpenCnt
++;
608 ReleaseSemaphore(&lh
->mss_static
->Lock
);
610 D(bug("[MSS-dev] %sfound unit %d at %p\n", found
?"":"not ", unitnum
, found
));
616 AROS_LH1(BPTR
, dev_CloseLib
,
617 AROS_LHA(struct IORequest
*, ioreq
, A1
),
618 mss_device_t
*, lh
, 2, Storage
)
622 /* Release the unit */
623 ObtainSemaphore(&lh
->mss_static
->Lock
);
624 mss_unit_t
*unit
= (mss_unit_t
*)ioreq
->io_Unit
;
626 ioreq
->io_Unit
= (struct Unit
*)~0;
627 if (unit
&& unit
->msu_unit
.unit_OpenCnt
)
628 unit
->msu_unit
.unit_OpenCnt
--;
630 ReleaseSemaphore(&lh
->mss_static
->Lock
);
638 AROS_LH1 (BPTR
, dev_ExpungeLib
,
639 AROS_LHA(mss_device_t
*, extralh
, D0
),
640 mss_device_t
*, lh
, 3, Storage
)
648 AROS_LH0 (LIBBASETYPEPTR
, dev_ExtFuncLib
,
649 mss_device_t
*, lh
, 4, Storage
)
657 * Helper function used to test, whether the specified command is slow or not.
658 * Slow commands will be transfered to the unit task, all others will be handled
661 * NOTE: Every command performing *any* USB transfer/operation is considered slow.
663 static const uint64_t immediate_cmds
= (
670 (1 << TD_CHANGENUM
) |
671 (1 << TD_CHANGESTATE
) |
672 (1 << TD_ADDCHANGEINT
) |
673 (1 << TD_REMCHANGEINT
) |
674 (1 << TD_GETGEOMETRY
) |
676 (1 << TD_GETDRIVETYPE
)
679 static BOOL
isSlow(uint32_t cmd
)
685 if (immediate_cmds
& (1 << cmd
))
688 else if (cmd
== NSCMD_TD_SEEK64
|| cmd
== NSCMD_DEVICEQUERY
)
691 // D(bug("[MSS-dev] isSlow(%04x) = %d\n", cmd, slow));
697 AROS_LH1(void, dev_BeginIO
,
698 AROS_LHA(struct IORequest
*, io
, A1
),
699 mss_device_t
*, mss_dev
, 5, Storage
)
703 // D(bug("[MSS-dev] BeginIO. CMD=%04x\n", io->io_Command));
707 mss_unit_t
*unit
= (mss_unit_t
*)io
->io_Unit
;
709 if (unit
&& unit
!= (void *)0xffffffff)
711 if (isSlow(io
->io_Command
))
713 // D(bug("[MSS-dev] Forwarding IO call to device task\n"));
714 unit
->msu_unit
.unit_flags
|= UNITF_INTASK
;
715 io
->io_Flags
&= ~IOF_QUICK
;
717 PutMsg(&unit
->msu_unit
.unit_MsgPort
, &io
->io_Message
);
719 unit
->msu_unit
.unit_flags
&= ~UNITF_INTASK
;
722 HandleIO(io
, mss_dev
, unit
);
730 AROS_LH1(LONG
, dev_AbortIO
,
731 AROS_LHA(struct IORequest
*, io
, A1
),
732 mss_device_t
*, mss_dev
, 6, Storage
)
736 /* Cannot Abort IO */
743 AROS_LH1(ULONG
, dev_GetRdskLba
,
744 AROS_LHA(struct IORequest
*, io
, A1
),
745 mss_device_t
*, mss_dev
, 7, Storage
)
755 AROS_LH1(ULONG
, dev_GetBlkSize
,
756 AROS_LHA(struct IORequest
*, io
, A1
),
757 mss_device_t
*, mss_dev
, 8, Storage
)
761 mss_unit_t
*unit
= (mss_unit_t
*)io
->io_Unit
;
763 if (unit
->msu_blockSize
)
764 return AROS_LEAST_BIT_POS(unit
->msu_blockSize
);
771 static const APTR Storage_dev_FuncTable
[]=
773 &AROS_SLIB_ENTRY(dev_OpenLib
,Storage
,1),
774 &AROS_SLIB_ENTRY(dev_CloseLib
,Storage
,2),
775 &AROS_SLIB_ENTRY(dev_ExpungeLib
,Storage
,3),
776 &AROS_SLIB_ENTRY(dev_ExtFuncLib
,Storage
,4),
777 &AROS_SLIB_ENTRY(dev_BeginIO
,Storage
,5),
778 &AROS_SLIB_ENTRY(dev_AbortIO
,Storage
,6),
779 &AROS_SLIB_ENTRY(dev_GetRdskLba
,Storage
,7),
780 &AROS_SLIB_ENTRY(dev_GetBlkSize
,Storage
,8),
784 static int StorageDev_Init(LIBBASETYPEPTR LIBBASE
)
786 D(bug("[MSS] USB Mass Storage device layer init\n"));
788 mss_device_t
*dev
= (mss_device_t
*)
789 ((intptr_t)AllocPooled(SD(LIBBASE
)->MemPool
,
790 sizeof(mss_device_t
) + sizeof(Storage_dev_FuncTable
)) + sizeof(Storage_dev_FuncTable
));
792 dev
->mss_static
= SD(LIBBASE
);
794 dev
->mss_device
.dd_Library
.lib_Node
.ln_Name
= "usbmss.device";
795 dev
->mss_device
.dd_Library
.lib_Node
.ln_Type
= NT_DEVICE
;
796 dev
->mss_device
.dd_Library
.lib_Node
.ln_Pri
= 0;
797 dev
->mss_device
.dd_Library
.lib_Version
= 1;
798 dev
->mss_device
.dd_Library
.lib_Revision
= 0;
800 MakeFunctions(dev
, Storage_dev_FuncTable
, NULL
);
801 AddDevice((struct Device
*)dev
);
806 static int StorageDev_Expunge(LIBBASETYPEPTR LIBBASE
)
811 ADD2INITLIB(StorageDev_Init
, 1)
812 ADD2EXPUNGELIB(StorageDev_Expunge
, 1)