init commit for spc/sbc
This commit is contained in:
208
pkg/scsi/sbc.go
208
pkg/scsi/sbc.go
@@ -17,6 +17,8 @@ limitations under the License.
|
||||
// SCSI block command processing
|
||||
package scsi
|
||||
|
||||
import "encoding/binary"
|
||||
|
||||
type SBCSCSIDeviceProtocol struct {
|
||||
BaseSCSIDeviceProtocol
|
||||
}
|
||||
@@ -37,10 +39,12 @@ func (sbc *SBCSCSIDeviceProtocol) OfflineLu(lu *SCSILu) error {
|
||||
return nil
|
||||
}
|
||||
|
||||
func NewSBCDevice() (SBCSCSIDeviceProtocol, error) {
|
||||
func NewSBCDevice() SBCSCSIDeviceProtocol {
|
||||
var sbc = SBCSCSIDeviceProtocol{
|
||||
BaseSCSIDeviceProtocol{Type: TYPE_DISK,
|
||||
SCSIDeviceOps: make([]SCSIDeviceOperation, 256)},
|
||||
BaseSCSIDeviceProtocol{
|
||||
Type: TYPE_DISK,
|
||||
SCSIDeviceOps: make([]SCSIDeviceOperation, 256),
|
||||
},
|
||||
}
|
||||
for i := 0; i <= 256; i++ {
|
||||
sbc.SCSIDeviceOps = append(sbc.SCSIDeviceOps, NewSCSIDeviceOperation(SPCIllegalOp, nil, 0))
|
||||
@@ -95,57 +99,201 @@ func NewSBCDevice() (SBCSCSIDeviceProtocol, error) {
|
||||
sbc.SCSIDeviceOps[WRITE_VERIFY_12] = NewSCSIDeviceOperation(SBCReadWrite, nil, PR_EA_FA|PR_EA_FN)
|
||||
sbc.SCSIDeviceOps[VERIFY_12] = NewSCSIDeviceOperation(SBCVerify, nil, PR_EA_FA|PR_EA_FN)
|
||||
|
||||
return sbc, nil
|
||||
return sbc
|
||||
}
|
||||
|
||||
func SBCModeSelect(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
func SBCModeSelect(host int, cmd *SCSICommand) SAMStat {
|
||||
return SAMStatGood
|
||||
}
|
||||
|
||||
func SBCModeSense(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
func SBCModeSense(host int, cmd *SCSICommand) SAMStat {
|
||||
// DPOFUA = 0x10
|
||||
var deviceSpecific uint8 = 0x10
|
||||
|
||||
if err := SPCModeSense(host, cmd); err.Err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// If this is a read-only lun, we must set the write protect bit
|
||||
if cmd.Device.Attrs.Readonly || cmd.Device.Attrs.SWP {
|
||||
deviceSpecific |= 0x80
|
||||
}
|
||||
|
||||
data := cmd.InSDBBuffer.Buffer
|
||||
data.Next(2)
|
||||
|
||||
if cmd.SCB.Bytes()[0] == 0x1a {
|
||||
data.WriteByte(deviceSpecific)
|
||||
} else {
|
||||
data.Next(1)
|
||||
data.WriteByte(deviceSpecific)
|
||||
}
|
||||
|
||||
return SAMStatGood
|
||||
}
|
||||
|
||||
func SBCFormatUnit(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
// The FORMAT UNIT command requests that the device server format the medium into application client
|
||||
// accessible logical blocks as specified in the number of blocks and block length values received
|
||||
// in the last mode parameter block descriptor in a MODE SELECT command (see SPC-3). In addition,
|
||||
// the device server may certify the medium and create control structures for the management of the medium and defects.
|
||||
// The degree that the medium is altered by this command is vendor-specific.
|
||||
func SBCFormatUnit(host int, cmd *SCSICommand) SAMStat {
|
||||
var (
|
||||
key = ILLEGAL_REQUEST
|
||||
asc = ASC_INVALID_FIELD_IN_CDB
|
||||
)
|
||||
|
||||
if err := deviceReserve(cmd); err != nil {
|
||||
return SAMStatReservationConflict
|
||||
}
|
||||
|
||||
if !cmd.Device.Attrs.Online {
|
||||
key = NOT_READY
|
||||
asc = ASC_MEDIUM_NOT_PRESENT
|
||||
goto sense
|
||||
}
|
||||
|
||||
if cmd.Device.Attrs.Readonly || cmd.Device.Attrs.SWP {
|
||||
key = DATA_PROTECT
|
||||
asc = ASC_WRITE_PROTECT
|
||||
goto sense
|
||||
}
|
||||
|
||||
if cmd.SCB.Bytes()[1]&0x80 != 0 {
|
||||
// we dont support format protection information
|
||||
goto sense
|
||||
}
|
||||
if cmd.SCB.Bytes()[1]&0x10 != 0 {
|
||||
// we dont support format data
|
||||
goto sense
|
||||
}
|
||||
if cmd.SCB.Bytes()[1]&0x07 != 0 {
|
||||
// defect list format must be 0
|
||||
goto sense
|
||||
}
|
||||
|
||||
return SAMStatGood
|
||||
sense:
|
||||
BuildSenseData(cmd, key, asc)
|
||||
return SAMStatCheckCondition
|
||||
}
|
||||
|
||||
func SBCUnmap(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
func SBCUnmap(host int, cmd *SCSICommand) SAMStat {
|
||||
return SAMStatGood
|
||||
}
|
||||
|
||||
func SBCReadWrite(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
func SBCReadWrite(host int, cmd *SCSICommand) SAMStat {
|
||||
return SAMStatGood
|
||||
}
|
||||
|
||||
func SBCReserve(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
func SBCReserve(host int, cmd *SCSICommand) SAMStat {
|
||||
if err := deviceReserve(cmd); err != nil {
|
||||
return SAMStatReservationConflict
|
||||
}
|
||||
return SAMStatGood
|
||||
}
|
||||
|
||||
func SBCRelease(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
func SBCRelease(host int, cmd *SCSICommand) SAMStat {
|
||||
if err := deviceRelease(cmd.Target.TID, cmd.CommandITNID, cmd.Device.Lun, false); err != nil {
|
||||
return SAMStatReservationConflict
|
||||
}
|
||||
|
||||
return SAMStatGood
|
||||
}
|
||||
|
||||
func SBCReadCapacity(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
// The READ CAPACITY (10) command requests that the device server transfer 8 bytes of parameter data
|
||||
// describing the capacity and medium format of the direct-access block device to the data-in buffer.
|
||||
// This command may be processed as if it has a HEAD OF QUEUE task attribute. If the logical unit supports
|
||||
// protection information, the application client should use the READ CAPACITY (16) command instead of
|
||||
// the READ CAPACITY (10) command.
|
||||
func SBCReadCapacity(host int, cmd *SCSICommand) SAMStat {
|
||||
var (
|
||||
scb = cmd.SCB.Bytes()
|
||||
key = ILLEGAL_REQUEST
|
||||
asc = ASC_LUN_NOT_SUPPORTED
|
||||
data = cmd.InSDBBuffer.Buffer
|
||||
bshift = cmd.Device.BlockShift
|
||||
size = cmd.Device.Size >> bshift
|
||||
)
|
||||
|
||||
if cmd.Device.Attrs.Removable && !cmd.Device.Attrs.Online {
|
||||
key = NOT_READY
|
||||
asc = ASC_MEDIUM_NOT_PRESENT
|
||||
goto sense
|
||||
}
|
||||
|
||||
if (scb[8]&0x1 == 0) && (scb[2]|scb[3]|scb[4]|scb[5]) != 0 {
|
||||
asc = ASC_INVALID_FIELD_IN_CDB
|
||||
goto sense
|
||||
}
|
||||
|
||||
if cmd.InSDBBuffer.Length < 8 {
|
||||
goto overflow
|
||||
}
|
||||
|
||||
// data[0] = (size >> 32) ? __cpu_to_be32(0xffffffff) : __cpu_to_be32(size - 1);
|
||||
if size>>32 != 0 {
|
||||
binary.Write(data, binary.BigEndian, uint32(0xffffffff))
|
||||
} else {
|
||||
binary.Write(data, binary.BigEndian, uint32(size-1))
|
||||
}
|
||||
|
||||
// data[1] = __cpu_to_be32(1U << bshift);
|
||||
binary.Write(data, binary.BigEndian, uint32(1<<bshift))
|
||||
overflow:
|
||||
cmd.InSDBBuffer.Resid = 8
|
||||
return SAMStatGood
|
||||
sense:
|
||||
cmd.InSDBBuffer.Resid = 0
|
||||
BuildSenseData(cmd, key, asc)
|
||||
return SAMStatCheckCondition
|
||||
}
|
||||
|
||||
func SBCVerify(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
// The VERIFY (10) command requests that the device server verify the specified logical block(s) on the medium.
|
||||
func SBCVerify(host int, cmd *SCSICommand) SAMStat {
|
||||
var (
|
||||
key = ILLEGAL_REQUEST
|
||||
asc = ASC_INVALID_FIELD_IN_CDB
|
||||
)
|
||||
if cmd.Device.Attrs.Removable && !cmd.Device.Attrs.Online {
|
||||
key = NOT_READY
|
||||
asc = ASC_MEDIUM_NOT_PRESENT
|
||||
goto sense
|
||||
}
|
||||
|
||||
if cmd.SCB.Bytes()[1]&0xe0 != 0 {
|
||||
// We only support protection information type 0
|
||||
key = ILLEGAL_REQUEST
|
||||
asc = ASC_INVALID_FIELD_IN_CDB
|
||||
goto sense
|
||||
}
|
||||
|
||||
if cmd.SCB.Bytes()[1]&0x02 == 0 {
|
||||
// no data compare with the media
|
||||
return SAMStatGood
|
||||
}
|
||||
// TODO
|
||||
sense:
|
||||
cmd.InSDBBuffer.Resid = 0
|
||||
BuildSenseData(cmd, key, asc)
|
||||
return SAMStatCheckCondition
|
||||
}
|
||||
|
||||
func SBCReadCapacity16(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
func SBCReadCapacity16(host int, cmd *SCSICommand) SAMStat {
|
||||
return SAMStatGood
|
||||
}
|
||||
|
||||
func SBCGetLbaStatus(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
func SBCGetLbaStatus(host int, cmd *SCSICommand) SAMStat {
|
||||
return SAMStatGood
|
||||
}
|
||||
|
||||
func SBCServiceAction(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
func SBCServiceAction(host int, cmd *SCSICommand) SAMStat {
|
||||
return SAMStatGood
|
||||
}
|
||||
|
||||
func SBCSyncCache(host int, cmd *SCSICommand) error {
|
||||
return nil
|
||||
// The SYNCHRONIZE CACHE (10) command requests that the device server ensure that
|
||||
// the specified logical blocks have their most recent data values recorded in
|
||||
// non-volatile cache and/or on the medium, based on the SYNC_NV bit.
|
||||
func SBCSyncCache(host int, cmd *SCSICommand) SAMStat {
|
||||
return SAMStatGood
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user