rtt更新
This commit is contained in:
20
rt-thread/components/drivers/scsi/Kconfig
Normal file
20
rt-thread/components/drivers/scsi/Kconfig
Normal file
@@ -0,0 +1,20 @@
|
||||
menuconfig RT_USING_SCSI
|
||||
bool "Using Small Computer System Interface (SCSI)"
|
||||
depends on RT_USING_DM
|
||||
default n
|
||||
|
||||
config RT_SCSI_SD
|
||||
bool "SD device on SCSI"
|
||||
depends on RT_USING_SCSI
|
||||
depends on RT_USING_BLK
|
||||
default y
|
||||
|
||||
config RT_SCSI_CDROM
|
||||
bool "CD-ROM device on SCSI"
|
||||
depends on RT_USING_SCSI
|
||||
depends on RT_USING_BLK
|
||||
default y
|
||||
|
||||
if RT_USING_SCSI
|
||||
osource "$(SOC_DM_SCSI_DIR)/Kconfig"
|
||||
endif
|
21
rt-thread/components/drivers/scsi/SConscript
Normal file
21
rt-thread/components/drivers/scsi/SConscript
Normal file
@@ -0,0 +1,21 @@
|
||||
from building import *
|
||||
|
||||
group = []
|
||||
|
||||
if not GetDepend(['RT_USING_SCSI']):
|
||||
Return('group')
|
||||
|
||||
cwd = GetCurrentDir()
|
||||
CPPPATH = [cwd + '/../include']
|
||||
|
||||
src = ['scsi.c']
|
||||
|
||||
if GetDepend(['RT_SCSI_SD']):
|
||||
src += ['scsi_sd.c']
|
||||
|
||||
if GetDepend(['RT_SCSI_CDROM']):
|
||||
src += ['scsi_cdrom.c']
|
||||
|
||||
group = DefineGroup('DeviceDrivers', src, depend = [''], CPPPATH = CPPPATH)
|
||||
|
||||
Return('group')
|
669
rt-thread/components/drivers/scsi/scsi.c
Normal file
669
rt-thread/components/drivers/scsi/scsi.c
Normal file
@@ -0,0 +1,669 @@
|
||||
/*
|
||||
* Copyright (c) 2006-2023, RT-Thread Development Team
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*
|
||||
* Change Logs:
|
||||
* Date Author Notes
|
||||
* 2023-02-25 GuEe-GUI the first version
|
||||
*/
|
||||
|
||||
#include <rthw.h>
|
||||
#include <rtthread.h>
|
||||
#include <rtdevice.h>
|
||||
|
||||
#define DBG_TAG "rtdm.scsi"
|
||||
#define DBG_LVL DBG_INFO
|
||||
#include <rtdbg.h>
|
||||
|
||||
/*
|
||||
* Since SCSI is used for storage (almost),
|
||||
* we do not want to implement SCSI as a system DM bus.
|
||||
*/
|
||||
struct scsi_driver
|
||||
{
|
||||
rt_err_t (*probe)(struct rt_scsi_device *sdev);
|
||||
rt_err_t (*remove)(struct rt_scsi_device *sdev);
|
||||
};
|
||||
|
||||
extern rt_err_t scsi_sd_probe(struct rt_scsi_device *sdev);
|
||||
extern rt_err_t scsi_sd_remove(struct rt_scsi_device *sdev);
|
||||
|
||||
extern rt_err_t scsi_cdrom_probe(struct rt_scsi_device *sdev);
|
||||
extern rt_err_t scsi_cdrom_remove(struct rt_scsi_device *sdev);
|
||||
|
||||
static struct scsi_driver driver_table[SCSI_DEVICE_TYPE_MAX] =
|
||||
{
|
||||
#ifdef RT_SCSI_SD
|
||||
[SCSI_DEVICE_TYPE_DIRECT] =
|
||||
{
|
||||
.probe = scsi_sd_probe,
|
||||
.remove = scsi_sd_remove,
|
||||
},
|
||||
#endif /* RT_SCSI_SD */
|
||||
#ifdef RT_SCSI_CDROM
|
||||
[SCSI_DEVICE_TYPE_CDROM] =
|
||||
{
|
||||
.probe = scsi_cdrom_probe,
|
||||
.remove = scsi_cdrom_remove,
|
||||
},
|
||||
#endif /* RT_SCSI_CDROM */
|
||||
};
|
||||
|
||||
static rt_err_t scsi_device_setup(struct rt_scsi_device *sdev)
|
||||
{
|
||||
rt_err_t err;
|
||||
rt_tick_t timeout;
|
||||
|
||||
if (sdev->host->ops->reset)
|
||||
{
|
||||
if ((err = sdev->host->ops->reset(sdev)))
|
||||
{
|
||||
return err;
|
||||
}
|
||||
}
|
||||
|
||||
if (!driver_table[sdev->devtype].probe)
|
||||
{
|
||||
LOG_E("Device type %x is not supported", sdev->devtype);
|
||||
|
||||
return -RT_ENOSYS;
|
||||
}
|
||||
|
||||
timeout = rt_tick_from_millisecond(5000);
|
||||
timeout += rt_tick_get();
|
||||
|
||||
while ((err = rt_scsi_test_unit_ready(sdev)))
|
||||
{
|
||||
if (rt_tick_get() >= timeout)
|
||||
{
|
||||
return -RT_ETIMEOUT;
|
||||
}
|
||||
}
|
||||
|
||||
return driver_table[sdev->devtype].probe(sdev);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_host_register(struct rt_scsi_host *scsi)
|
||||
{
|
||||
struct rt_scsi_device tmp_sdev, *sdev;
|
||||
|
||||
if (!scsi || !scsi->dev || !scsi->ops)
|
||||
{
|
||||
return -RT_EINVAL;
|
||||
}
|
||||
|
||||
if (!scsi->max_id || !scsi->max_lun)
|
||||
{
|
||||
return -RT_EINVAL;
|
||||
}
|
||||
|
||||
rt_list_init(&scsi->lun_nodes);
|
||||
|
||||
rt_memset(&tmp_sdev, 0, sizeof(tmp_sdev));
|
||||
tmp_sdev.host = scsi;
|
||||
|
||||
for (rt_size_t id = 0; id < scsi->max_id; ++id)
|
||||
{
|
||||
for (rt_size_t lun = 0; lun < scsi->max_lun; ++lun)
|
||||
{
|
||||
tmp_sdev.id = id;
|
||||
tmp_sdev.lun = lun;
|
||||
|
||||
if (rt_scsi_inquiry(&tmp_sdev, RT_NULL))
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
if (tmp_sdev.devtype >= SCSI_DEVICE_TYPE_MAX)
|
||||
{
|
||||
/*
|
||||
* This might seem odd, but we're only aiming to
|
||||
* support simple SCSI.
|
||||
* If devices appear on the bus out of order,
|
||||
* we won't perform additional scans.
|
||||
*/
|
||||
scsi->max_id = id;
|
||||
scsi->max_lun = lun;
|
||||
|
||||
LOG_D("Scan is end of ID: %u LUN: %u", id, lun);
|
||||
break;
|
||||
}
|
||||
|
||||
if (!(sdev = rt_malloc(sizeof(*sdev))))
|
||||
{
|
||||
if (!rt_list_isempty(&scsi->lun_nodes))
|
||||
{
|
||||
LOG_E("No memory to create device ID: %u, LUN: %u", id, lun);
|
||||
|
||||
return RT_EOK;
|
||||
}
|
||||
|
||||
return -RT_ENOMEM;
|
||||
}
|
||||
|
||||
rt_memcpy(sdev, &tmp_sdev, sizeof(*sdev));
|
||||
rt_list_init(&sdev->list);
|
||||
|
||||
if (scsi_device_setup(sdev))
|
||||
{
|
||||
rt_free(sdev);
|
||||
continue;
|
||||
}
|
||||
|
||||
rt_list_insert_before(&scsi->lun_nodes, &sdev->list);
|
||||
}
|
||||
}
|
||||
|
||||
return rt_list_isempty(&scsi->lun_nodes) ? -RT_EEMPTY : RT_EOK;
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_host_unregister(struct rt_scsi_host *scsi)
|
||||
{
|
||||
struct rt_scsi_device *sdev, *next_sdev;
|
||||
|
||||
if (!scsi)
|
||||
{
|
||||
return -RT_EINVAL;
|
||||
}
|
||||
|
||||
rt_list_for_each_entry_safe(sdev, next_sdev, &scsi->lun_nodes, list)
|
||||
{
|
||||
rt_list_remove(&sdev->list);
|
||||
|
||||
if (sdev->host->ops->reset)
|
||||
{
|
||||
sdev->host->ops->reset(sdev);
|
||||
}
|
||||
|
||||
if (!driver_table[sdev->devtype].remove)
|
||||
{
|
||||
driver_table[sdev->devtype].remove(sdev);
|
||||
}
|
||||
|
||||
rt_free(sdev);
|
||||
}
|
||||
|
||||
return RT_EOK;
|
||||
}
|
||||
|
||||
rt_inline rt_err_t scsi_transfer(struct rt_scsi_device *sdev, struct rt_scsi_cmd *cmd)
|
||||
{
|
||||
return sdev->host->ops->transfer(sdev, cmd);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_request_sense(struct rt_scsi_device *sdev,
|
||||
struct rt_scsi_request_sense_data *out_data)
|
||||
{
|
||||
rt_err_t err;
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.request_sense.opcode = RT_SCSI_CMD_REQUEST_SENSE;
|
||||
cmd.op.request_sense.config = 0;
|
||||
cmd.op.request_sense.alloc_length = 0x12;
|
||||
cmd.op.request_sense.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.request_sense);
|
||||
cmd.data.ptr = &cmd.data.request_sense;
|
||||
cmd.data.size = sizeof(cmd.data.request_sense);
|
||||
|
||||
err = scsi_transfer(sdev, &cmd);
|
||||
|
||||
if (!err && out_data)
|
||||
{
|
||||
rt_memcpy(out_data, &cmd.data.request_sense, sizeof(*out_data));
|
||||
}
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_test_unit_ready(struct rt_scsi_device *sdev)
|
||||
{
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.test_unit_ready.opcode = RT_SCSI_CMD_TEST_UNIT_READY;
|
||||
cmd.op.test_unit_ready.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.test_unit_ready);
|
||||
|
||||
return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_inquiry(struct rt_scsi_device *sdev,
|
||||
struct rt_scsi_inquiry_data *out_data)
|
||||
{
|
||||
rt_err_t err;
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.inquiry.opcode = RT_SCSI_CMD_INQUIRY;
|
||||
cmd.op.inquiry.config = 0;
|
||||
cmd.op.inquiry.page = 0;
|
||||
cmd.op.inquiry.alloc_length = 0x24;
|
||||
cmd.op.inquiry.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.inquiry);
|
||||
cmd.data.ptr = &cmd.data.inquiry;
|
||||
cmd.data.size = sizeof(cmd.data.inquiry);
|
||||
|
||||
err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
|
||||
if (!err)
|
||||
{
|
||||
sdev->devtype = cmd.data.inquiry.devtype & RT_SCSI_DEVTYPE_MASK;
|
||||
sdev->removable = cmd.data.inquiry.rmb >> RT_SCSI_REMOVABLE_BIT;
|
||||
|
||||
if (out_data)
|
||||
{
|
||||
rt_memcpy(out_data, &cmd.data.inquiry, sizeof(*out_data));
|
||||
}
|
||||
}
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_read_capacity10(struct rt_scsi_device *sdev,
|
||||
struct rt_scsi_read_capacity10_data *out_data)
|
||||
{
|
||||
rt_err_t err;
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.read_capacity10.opcode = RT_SCSI_CMD_READ_CAPACITY10;
|
||||
cmd.op.read_capacity10.config = 0;
|
||||
cmd.op.read_capacity10.logical_block_addr = 0;
|
||||
cmd.op.read_capacity10.pmi = 0;
|
||||
cmd.op.read_capacity10.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.read_capacity10);
|
||||
cmd.data.ptr = &cmd.data.read_capacity10;
|
||||
cmd.data.size = sizeof(cmd.data.read_capacity10);
|
||||
|
||||
err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
|
||||
if (!err)
|
||||
{
|
||||
sdev->last_block = rt_be32_to_cpu(cmd.data.read_capacity10.last_block);
|
||||
sdev->block_size = rt_be32_to_cpu(cmd.data.read_capacity10.block_size);
|
||||
|
||||
if (out_data)
|
||||
{
|
||||
rt_memcpy(out_data, &cmd.data.read_capacity10, sizeof(*out_data));
|
||||
}
|
||||
}
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_read_capacity16(struct rt_scsi_device *sdev,
|
||||
struct rt_scsi_read_capacity16_data *out_data)
|
||||
{
|
||||
rt_err_t err;
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.read_capacity16.opcode = RT_SCSI_CMD_READ_CAPACITY16;
|
||||
cmd.op.read_capacity16.config = 0x10;
|
||||
cmd.op.read_capacity16.logical_block_addr = 0;
|
||||
cmd.op.read_capacity16.alloc_len = rt_cpu_to_be32(sizeof(cmd.data.read_capacity16));
|
||||
cmd.op.read_capacity16.pmi = 0;
|
||||
cmd.op.read_capacity16.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.read_capacity16);
|
||||
cmd.data.ptr = &cmd.data.read_capacity16;
|
||||
cmd.data.size = sizeof(cmd.data.read_capacity16);
|
||||
|
||||
err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
|
||||
if (!err)
|
||||
{
|
||||
sdev->last_block = rt_be64_to_cpu(cmd.data.read_capacity16.last_block);
|
||||
sdev->block_size = rt_be32_to_cpu(cmd.data.read_capacity16.block_size);
|
||||
|
||||
if (out_data)
|
||||
{
|
||||
rt_memcpy(out_data, &cmd.data.read_capacity16, sizeof(*out_data));
|
||||
}
|
||||
}
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_read10(struct rt_scsi_device *sdev,
|
||||
rt_off_t lba, void *buffer, rt_size_t size)
|
||||
{
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.read10.opcode = RT_SCSI_CMD_READ10;
|
||||
cmd.op.read10.config = 0;
|
||||
cmd.op.read10.lba = rt_cpu_to_be32(lba);
|
||||
cmd.op.read10.size = rt_cpu_to_be16(size);
|
||||
cmd.op_size = sizeof(cmd.op.read10);
|
||||
cmd.data.ptr = buffer;
|
||||
cmd.data.size = size * sdev->block_size;
|
||||
|
||||
return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_read12(struct rt_scsi_device *sdev,
|
||||
rt_off_t lba, void *buffer, rt_size_t size)
|
||||
{
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.read12.opcode = RT_SCSI_CMD_READ12;
|
||||
cmd.op.read12.config = 0;
|
||||
cmd.op.read12.lba = rt_cpu_to_be32(lba);
|
||||
cmd.op.read12.size = rt_cpu_to_be32(size);
|
||||
cmd.op.read12.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.read12);
|
||||
cmd.data.ptr = buffer;
|
||||
cmd.data.size = size * sdev->block_size;
|
||||
|
||||
return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_read16(struct rt_scsi_device *sdev,
|
||||
rt_off_t lba, void *buffer, rt_size_t size)
|
||||
{
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.read16.opcode = RT_SCSI_CMD_READ16;
|
||||
cmd.op.read16.config = 0;
|
||||
cmd.op.read16.lba = rt_cpu_to_be64(lba);
|
||||
cmd.op.read16.size = rt_cpu_to_be32(size);
|
||||
cmd.op.read16.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.read16);
|
||||
cmd.data.ptr = buffer;
|
||||
cmd.data.size = size * sdev->block_size;
|
||||
|
||||
return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_write10(struct rt_scsi_device *sdev,
|
||||
rt_off_t lba, const void *buffer, rt_size_t size)
|
||||
{
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.write10.opcode = RT_SCSI_CMD_WRITE10;
|
||||
cmd.op.write10.config = 0;
|
||||
cmd.op.write10.lba = rt_cpu_to_be32(lba);
|
||||
cmd.op.write10.size = rt_cpu_to_be16(size);
|
||||
cmd.op_size = sizeof(cmd.op.write10);
|
||||
cmd.data.ptr = (void *)buffer;
|
||||
cmd.data.size = size * sdev->block_size;
|
||||
|
||||
return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_write12(struct rt_scsi_device *sdev,
|
||||
rt_off_t lba, const void *buffer, rt_size_t size)
|
||||
{
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.write12.opcode = RT_SCSI_CMD_WRITE12;
|
||||
cmd.op.write12.config = 0;
|
||||
cmd.op.write12.lba = rt_cpu_to_be32(lba);
|
||||
cmd.op.write12.size = rt_cpu_to_be32(size);
|
||||
cmd.op.write12.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.write12);
|
||||
cmd.data.ptr = (void *)buffer;
|
||||
cmd.data.size = size * sdev->block_size;
|
||||
|
||||
return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_write16(struct rt_scsi_device *sdev,
|
||||
rt_off_t lba, const void *buffer, rt_size_t size)
|
||||
{
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.write16.opcode = RT_SCSI_CMD_WRITE16;
|
||||
cmd.op.write16.config = 0;
|
||||
cmd.op.write16.lba = rt_cpu_to_be64(lba);
|
||||
cmd.op.write16.size = rt_cpu_to_be32(size);
|
||||
cmd.op.write16.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.write16);
|
||||
cmd.data.ptr = (void *)buffer;
|
||||
cmd.data.size = size * sdev->block_size;
|
||||
|
||||
return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_synchronize_cache10(struct rt_scsi_device *sdev,
|
||||
rt_off_t lba, rt_size_t size)
|
||||
{
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.synchronize_cache10.opcode = RT_SCSI_CMD_SYNCHRONIZE_CACHE10;
|
||||
cmd.op.synchronize_cache10.config = 0;
|
||||
cmd.op.synchronize_cache10.lba = rt_cpu_to_be32(lba);
|
||||
cmd.op.synchronize_cache10.size = rt_cpu_to_be16(size);
|
||||
cmd.op.synchronize_cache10.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.synchronize_cache10);
|
||||
|
||||
return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_synchronize_cache16(struct rt_scsi_device *sdev,
|
||||
rt_off_t lba, rt_size_t size)
|
||||
{
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.synchronize_cache16.opcode = RT_SCSI_CMD_SYNCHRONIZE_CACHE16;
|
||||
cmd.op.synchronize_cache16.config = 0;
|
||||
cmd.op.synchronize_cache16.lba = rt_cpu_to_be64(lba);
|
||||
cmd.op.synchronize_cache16.size = rt_cpu_to_be32(size);
|
||||
cmd.op.synchronize_cache16.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.synchronize_cache16);
|
||||
|
||||
return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_write_same10(struct rt_scsi_device *sdev,
|
||||
rt_off_t lba, rt_size_t size)
|
||||
{
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.write_same10.opcode = RT_SCSI_CMD_WRITE_SAME10;
|
||||
cmd.op.write_same10.config = RT_BIT(RT_SCSI_UNMAP_SHIFT);
|
||||
cmd.op.write_same10.lba = rt_cpu_to_be32(lba);
|
||||
cmd.op.write_same10.size = rt_cpu_to_be16(size);
|
||||
cmd.op.write_same10.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.write_same10);
|
||||
|
||||
return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_write_same16(struct rt_scsi_device *sdev,
|
||||
rt_off_t lba, rt_size_t size)
|
||||
{
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.write_same16.opcode = RT_SCSI_CMD_WRITE_SAME16;
|
||||
cmd.op.write_same16.config = RT_BIT(RT_SCSI_UNMAP_SHIFT);
|
||||
cmd.op.write_same16.lba = rt_cpu_to_be64(lba);
|
||||
cmd.op.write_same16.size = rt_cpu_to_be32(size);
|
||||
cmd.op.write_same16.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.write_same16);
|
||||
|
||||
return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_mode_select6(struct rt_scsi_device *sdev,
|
||||
rt_uint8_t pf, rt_uint8_t sp, void *buffer, rt_size_t size,
|
||||
struct rt_scsi_mode_select_data *data)
|
||||
{
|
||||
rt_err_t err;
|
||||
rt_uint8_t *real_buffer;
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
real_buffer = rt_malloc(4 + size);
|
||||
|
||||
if (!real_buffer)
|
||||
{
|
||||
return -RT_ENOMEM;
|
||||
}
|
||||
|
||||
rt_memcpy(real_buffer + 4, buffer, size);
|
||||
size += 4;
|
||||
real_buffer[0] = 0;
|
||||
real_buffer[1] = data->medium_type;
|
||||
real_buffer[2] = data->device_specific;
|
||||
real_buffer[3] = data->block_descriptor_length;
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.mode_select6.opcode = RT_SCSI_CMD_MODE_SELECT;
|
||||
cmd.op.mode_select6.config = pf ? RT_BIT(RT_SCSI_PF_SHIFT) : 0;
|
||||
cmd.op.mode_select6.config |= sp ? RT_BIT(RT_SCSI_SP_SHIFT) : 0;
|
||||
cmd.op.mode_select6.param_list_len = (rt_uint8_t)size;
|
||||
cmd.op.mode_select6.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.mode_select6);
|
||||
cmd.data.ptr = real_buffer;
|
||||
cmd.data.size = size;
|
||||
|
||||
err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
|
||||
rt_free(real_buffer);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_mode_select10(struct rt_scsi_device *sdev,
|
||||
rt_uint8_t pf, rt_uint8_t sp, void *buffer, rt_size_t size,
|
||||
struct rt_scsi_mode_select_data *data)
|
||||
{
|
||||
rt_err_t err;
|
||||
rt_uint8_t *real_buffer;
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
real_buffer = rt_malloc(8 + size);
|
||||
|
||||
if (!real_buffer)
|
||||
{
|
||||
return -RT_ENOMEM;
|
||||
}
|
||||
|
||||
rt_memcpy(real_buffer + 8, buffer, size);
|
||||
size += 8;
|
||||
real_buffer[0] = 0;
|
||||
real_buffer[1] = 0;
|
||||
real_buffer[2] = data->medium_type;
|
||||
real_buffer[3] = data->device_specific;
|
||||
real_buffer[4] = data->longlba ? 0x01 : 0;
|
||||
real_buffer[5] = 0;
|
||||
real_buffer[6] = rt_cpu_to_be16(data->block_descriptor_length);
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.mode_select10.opcode = RT_SCSI_CMD_MODE_SELECT10;
|
||||
cmd.op.mode_select10.config = pf ? RT_BIT(RT_SCSI_PF_SHIFT) : 0;
|
||||
cmd.op.mode_select10.config |= sp ? RT_BIT(RT_SCSI_SP_SHIFT) : 0;
|
||||
cmd.op.mode_select10.param_list_len = rt_cpu_to_be16(size);
|
||||
cmd.op.mode_select10.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.mode_select10);
|
||||
cmd.data.ptr = real_buffer;
|
||||
cmd.data.size = size;
|
||||
|
||||
err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
|
||||
rt_free(real_buffer);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static void scsi_mode_sense_fill(struct rt_scsi_mode_select_data *data,
|
||||
rt_uint8_t modepage, rt_uint8_t *buffer, rt_bool_t use10)
|
||||
{
|
||||
if (buffer[0] == 0x86 && buffer[1] == 0x0b && (modepage == 6 || modepage == 8))
|
||||
{
|
||||
data->header_length = 0;
|
||||
data->length = 13;
|
||||
data->medium_type = 0;
|
||||
data->device_specific = 0;
|
||||
data->longlba = 0;
|
||||
data->block_descriptor_length = 0;
|
||||
}
|
||||
else if (use10)
|
||||
{
|
||||
data->length = rt_be16_to_cpu(buffer[0]) + 2;
|
||||
data->medium_type = buffer[2];
|
||||
data->device_specific = buffer[3];
|
||||
data->longlba = buffer[4] & 0x01;
|
||||
data->block_descriptor_length = rt_be16_to_cpu(buffer[6]);
|
||||
}
|
||||
else
|
||||
{
|
||||
data->length = buffer[0] + 1;
|
||||
data->medium_type = buffer[1];
|
||||
data->device_specific = buffer[2];
|
||||
data->block_descriptor_length = buffer[3];
|
||||
}
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_mode_sense6(struct rt_scsi_device *sdev,
|
||||
rt_uint8_t dbd, rt_uint8_t modepage, rt_uint8_t subpage, void *buffer, rt_size_t size,
|
||||
struct rt_scsi_mode_select_data *data)
|
||||
{
|
||||
rt_err_t err;
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(buffer, 0, size);
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.mode_sense6.opcode = RT_SCSI_CMD_MODE_SENSE;
|
||||
cmd.op.mode_sense6.config = dbd & 0x18;
|
||||
cmd.op.mode_sense6.page_control_code = modepage;
|
||||
cmd.op.mode_sense6.subpage_code = subpage;
|
||||
cmd.op.mode_sense6.allocation_len = (rt_uint8_t)size;
|
||||
cmd.op.mode_sense6.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.mode_sense6);
|
||||
cmd.data.ptr = buffer;
|
||||
cmd.data.size = size;
|
||||
|
||||
err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
|
||||
if (!err)
|
||||
{
|
||||
data->header_length = 4;
|
||||
scsi_mode_sense_fill(data, modepage, buffer, RT_FALSE);
|
||||
}
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
rt_err_t rt_scsi_mode_sense10(struct rt_scsi_device *sdev,
|
||||
rt_uint8_t dbd, rt_uint8_t modepage, rt_uint8_t subpage, void *buffer, rt_size_t size,
|
||||
struct rt_scsi_mode_select_data *data)
|
||||
{
|
||||
rt_err_t err;
|
||||
struct rt_scsi_cmd cmd;
|
||||
|
||||
rt_memset(buffer, 0, size);
|
||||
|
||||
rt_memset(&cmd, 0, sizeof(cmd));
|
||||
cmd.op.mode_sense6.opcode = RT_SCSI_CMD_MODE_SENSE10;
|
||||
cmd.op.mode_sense6.config = dbd & 0x18;
|
||||
cmd.op.mode_sense6.page_control_code = modepage;
|
||||
cmd.op.mode_sense6.subpage_code = subpage;
|
||||
cmd.op.mode_sense6.allocation_len = rt_cpu_to_be16(size);
|
||||
cmd.op.mode_sense6.control = 0;
|
||||
cmd.op_size = sizeof(cmd.op.mode_sense6);
|
||||
cmd.data.ptr = buffer;
|
||||
cmd.data.size = size;
|
||||
|
||||
err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
|
||||
|
||||
if (!err)
|
||||
{
|
||||
data->header_length = 8;
|
||||
scsi_mode_sense_fill(data, modepage, buffer, RT_FALSE);
|
||||
}
|
||||
|
||||
return err;
|
||||
}
|
136
rt-thread/components/drivers/scsi/scsi_cdrom.c
Normal file
136
rt-thread/components/drivers/scsi/scsi_cdrom.c
Normal file
@@ -0,0 +1,136 @@
|
||||
/*
|
||||
* Copyright (c) 2006-2023, RT-Thread Development Team
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*
|
||||
* Change Logs:
|
||||
* Date Author Notes
|
||||
* 2023-02-25 GuEe-GUI the first version
|
||||
*/
|
||||
|
||||
#include <rthw.h>
|
||||
#include <rtthread.h>
|
||||
#include <rtdevice.h>
|
||||
|
||||
#define DBG_TAG "scsi.blk"
|
||||
#define DBG_LVL DBG_INFO
|
||||
#include <rtdbg.h>
|
||||
|
||||
struct scsi_cdrom
|
||||
{
|
||||
struct rt_blk_disk parent;
|
||||
struct rt_scsi_device *sdev;
|
||||
|
||||
int cdrom_id;
|
||||
struct rt_device_blk_geometry geometry;
|
||||
};
|
||||
|
||||
#define raw_to_scsi_cdrom(raw) rt_container_of(raw, struct scsi_cdrom, parent)
|
||||
|
||||
static struct rt_dm_ida cdrom_ida = RT_DM_IDA_INIT(CUSTOM);
|
||||
static struct rt_dm_ida scsi_cdrom_ida = RT_DM_IDA_INIT(SCSI_CDROM);
|
||||
|
||||
static rt_ssize_t scsi_cdrom_read(struct rt_blk_disk *disk, rt_off_t sector,
|
||||
void *buffer, rt_size_t sector_count)
|
||||
{
|
||||
rt_err_t err;
|
||||
struct scsi_cdrom *scdrom = raw_to_scsi_cdrom(disk);
|
||||
struct rt_scsi_device *sdev = scdrom->sdev;
|
||||
|
||||
sector_count &= RT_UINT32_MAX;
|
||||
|
||||
if (sector >> 32)
|
||||
{
|
||||
err = rt_scsi_read16(sdev, sector, buffer, sector_count);
|
||||
}
|
||||
else
|
||||
{
|
||||
err = rt_scsi_read12(sdev, sector, buffer, sector_count);
|
||||
}
|
||||
|
||||
return !err ? sector_count : (rt_ssize_t)err;
|
||||
}
|
||||
|
||||
static rt_err_t scsi_cdrom_getgeome(struct rt_blk_disk *disk,
|
||||
struct rt_device_blk_geometry *geometry)
|
||||
{
|
||||
struct scsi_cdrom *scdrom = raw_to_scsi_cdrom(disk);
|
||||
|
||||
rt_memcpy(geometry, &scdrom->geometry, sizeof(scdrom->geometry));
|
||||
|
||||
return RT_EOK;
|
||||
}
|
||||
|
||||
static const struct rt_blk_disk_ops scsi_cdrom_ops =
|
||||
{
|
||||
.read = scsi_cdrom_read,
|
||||
.getgeome = scsi_cdrom_getgeome,
|
||||
};
|
||||
|
||||
rt_err_t scsi_cdrom_probe(struct rt_scsi_device *sdev)
|
||||
{
|
||||
rt_err_t err;
|
||||
union
|
||||
{
|
||||
struct rt_scsi_read_capacity10_data capacity10;
|
||||
struct rt_scsi_read_capacity16_data capacity16;
|
||||
} data;
|
||||
struct scsi_cdrom *scdrom = rt_calloc(1, sizeof(*scdrom));
|
||||
|
||||
if (!scdrom)
|
||||
{
|
||||
return -RT_ENOMEM;
|
||||
}
|
||||
|
||||
if ((scdrom->cdrom_id = rt_dm_ida_alloc(&cdrom_ida)) < 0)
|
||||
{
|
||||
return -RT_EFULL;
|
||||
}
|
||||
|
||||
sdev->priv = scdrom;
|
||||
scdrom->sdev = sdev;
|
||||
scdrom->parent.ida = &scsi_cdrom_ida;
|
||||
scdrom->parent.read_only = RT_TRUE;
|
||||
scdrom->parent.parallel_io = RT_FALSE;
|
||||
scdrom->parent.ops = &scsi_cdrom_ops;
|
||||
scdrom->parent.max_partitions = RT_BLK_PARTITION_NONE;
|
||||
|
||||
if ((err = rt_scsi_read_capacity10(sdev, &data.capacity10)))
|
||||
{
|
||||
goto _fail;
|
||||
}
|
||||
if (data.capacity10.last_block == 0xffffffff)
|
||||
{
|
||||
if ((err = rt_scsi_read_capacity16(sdev, &data.capacity16)))
|
||||
{
|
||||
goto _fail;
|
||||
}
|
||||
}
|
||||
scdrom->geometry.bytes_per_sector = sdev->block_size;
|
||||
scdrom->geometry.block_size = sdev->block_size;
|
||||
scdrom->geometry.sector_count = sdev->last_block + 1;
|
||||
|
||||
rt_dm_dev_set_name(&scdrom->parent.parent, "cdrom%u", scdrom->cdrom_id);
|
||||
|
||||
if ((err = rt_hw_blk_disk_register(&scdrom->parent)))
|
||||
{
|
||||
goto _fail;
|
||||
}
|
||||
|
||||
return RT_EOK;
|
||||
|
||||
_fail:
|
||||
rt_dm_ida_free(&cdrom_ida, scdrom->cdrom_id);
|
||||
rt_free(scdrom);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
rt_err_t scsi_cdrom_remove(struct rt_scsi_device *sdev)
|
||||
{
|
||||
struct scsi_cdrom *scdrom = sdev->priv;
|
||||
|
||||
rt_dm_ida_free(&cdrom_ida, scdrom->cdrom_id);
|
||||
|
||||
return rt_hw_blk_disk_unregister(&scdrom->parent);
|
||||
}
|
251
rt-thread/components/drivers/scsi/scsi_sd.c
Normal file
251
rt-thread/components/drivers/scsi/scsi_sd.c
Normal file
@@ -0,0 +1,251 @@
|
||||
/*
|
||||
* Copyright (c) 2006-2023, RT-Thread Development Team
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*
|
||||
* Change Logs:
|
||||
* Date Author Notes
|
||||
* 2023-02-25 GuEe-GUI the first version
|
||||
*/
|
||||
|
||||
#include <rthw.h>
|
||||
#include <rtthread.h>
|
||||
#include <rtdevice.h>
|
||||
|
||||
#define DBG_TAG "scsi.blk"
|
||||
#define DBG_LVL DBG_INFO
|
||||
#include <rtdbg.h>
|
||||
|
||||
#include "../block/blk_dev.h"
|
||||
|
||||
struct scsi_sd
|
||||
{
|
||||
struct rt_blk_disk parent;
|
||||
struct rt_scsi_device *sdev;
|
||||
|
||||
int sd_id;
|
||||
rt_bool_t use16;
|
||||
struct rt_device_blk_geometry geometry;
|
||||
};
|
||||
|
||||
#define raw_to_scsi_sd(raw) rt_container_of(raw, struct scsi_sd, parent)
|
||||
|
||||
static struct rt_dm_ida sd_ida = RT_DM_IDA_INIT(CUSTOM);
|
||||
static struct rt_dm_ida scsi_sd_ida = RT_DM_IDA_INIT(SCSI_SD);
|
||||
|
||||
static rt_ssize_t scsi_sd_read(struct rt_blk_disk *disk, rt_off_t sector,
|
||||
void *buffer, rt_size_t sector_count)
|
||||
{
|
||||
rt_err_t err;
|
||||
struct scsi_sd *ssd = raw_to_scsi_sd(disk);
|
||||
struct rt_scsi_device *sdev = ssd->sdev;
|
||||
|
||||
sector_count &= RT_UINT32_MAX;
|
||||
|
||||
if (sector >> 32)
|
||||
{
|
||||
err = rt_scsi_read16(sdev, sector, buffer, sector_count);
|
||||
}
|
||||
else
|
||||
{
|
||||
err = rt_scsi_read10(sdev, sector, buffer, sector_count);
|
||||
}
|
||||
|
||||
return !err ? sector_count : (rt_ssize_t)err;
|
||||
}
|
||||
|
||||
static rt_ssize_t scsi_sd_write(struct rt_blk_disk *disk, rt_off_t sector,
|
||||
const void *buffer, rt_size_t sector_count)
|
||||
{
|
||||
rt_err_t err;
|
||||
struct scsi_sd *ssd = raw_to_scsi_sd(disk);
|
||||
struct rt_scsi_device *sdev = ssd->sdev;
|
||||
|
||||
sector_count &= RT_UINT32_MAX;
|
||||
|
||||
if (sector >> 32)
|
||||
{
|
||||
err = rt_scsi_write16(sdev, sector, buffer, sector_count);
|
||||
}
|
||||
else
|
||||
{
|
||||
err = rt_scsi_write10(sdev, sector, buffer, sector_count);
|
||||
}
|
||||
|
||||
return !err ? sector_count : (rt_ssize_t)err;
|
||||
}
|
||||
|
||||
static rt_err_t scsi_sd_getgeome(struct rt_blk_disk *disk,
|
||||
struct rt_device_blk_geometry *geometry)
|
||||
{
|
||||
struct scsi_sd *ssd = raw_to_scsi_sd(disk);
|
||||
|
||||
rt_memcpy(geometry, &ssd->geometry, sizeof(ssd->geometry));
|
||||
|
||||
return RT_EOK;
|
||||
}
|
||||
|
||||
static rt_err_t scsi_sd_sync(struct rt_blk_disk *disk)
|
||||
{
|
||||
rt_err_t err;
|
||||
rt_size_t lba_count;
|
||||
struct scsi_sd *ssd = raw_to_scsi_sd(disk);
|
||||
struct rt_scsi_device *sdev = ssd->sdev;
|
||||
|
||||
lba_count = ssd->geometry.sector_count;
|
||||
|
||||
if (ssd->use16)
|
||||
{
|
||||
err = rt_scsi_synchronize_cache16(sdev, 0, lba_count);
|
||||
}
|
||||
else
|
||||
{
|
||||
err = rt_scsi_synchronize_cache10(sdev, 0, lba_count);
|
||||
}
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static rt_err_t scsi_sd_erase(struct rt_blk_disk *disk)
|
||||
{
|
||||
rt_err_t err;
|
||||
rt_size_t lba_count;
|
||||
struct scsi_sd *ssd = raw_to_scsi_sd(disk);
|
||||
struct rt_scsi_device *sdev = ssd->sdev;
|
||||
|
||||
lba_count = ssd->geometry.sector_count;
|
||||
|
||||
if (ssd->use16)
|
||||
{
|
||||
err = rt_scsi_write_same16(sdev, 0, lba_count);
|
||||
}
|
||||
else
|
||||
{
|
||||
err = rt_scsi_write_same10(sdev, 0, lba_count);
|
||||
}
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static rt_err_t scsi_sd_autorefresh(struct rt_blk_disk *disk, rt_bool_t is_auto)
|
||||
{
|
||||
rt_err_t err;
|
||||
int sp;
|
||||
rt_size_t size;
|
||||
rt_uint8_t buffer[64];
|
||||
rt_uint8_t *buffer_data;
|
||||
rt_bool_t use6 = RT_TRUE;
|
||||
struct scsi_sd *ssd = raw_to_scsi_sd(disk);
|
||||
struct rt_scsi_device *sdev = ssd->sdev;
|
||||
struct rt_scsi_mode_select_data data;
|
||||
|
||||
err = rt_scsi_mode_sense6(sdev, 0x08, 8, 0, buffer, sizeof(buffer), &data);
|
||||
|
||||
if (err && err != -RT_ENOMEM)
|
||||
{
|
||||
use6 = RT_FALSE;
|
||||
err = rt_scsi_mode_sense10(sdev, 0x08, 8, 0, buffer, sizeof(buffer), &data);
|
||||
}
|
||||
if (err)
|
||||
{
|
||||
return err;
|
||||
}
|
||||
|
||||
size = rt_min_t(rt_size_t, sizeof(buffer),
|
||||
data.length - data.header_length - data.block_descriptor_length);
|
||||
buffer_data = buffer + data.header_length + data.block_descriptor_length;
|
||||
buffer_data[2] &= ~0x05;
|
||||
buffer_data[2] |= (!!is_auto) << 2 | (!!is_auto);
|
||||
sp = buffer_data[0] & 0x80 ? 1 : 0;
|
||||
buffer_data[0] &= ~0x80;
|
||||
data.device_specific = 0;
|
||||
|
||||
if (use6)
|
||||
{
|
||||
err = rt_scsi_mode_select6(sdev, 1, sp, buffer_data, size, &data);
|
||||
}
|
||||
else
|
||||
{
|
||||
err = rt_scsi_mode_select10(sdev, 1, sp, buffer_data, size, &data);
|
||||
}
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static const struct rt_blk_disk_ops scsi_sd_ops =
|
||||
{
|
||||
.read = scsi_sd_read,
|
||||
.write = scsi_sd_write,
|
||||
.getgeome = scsi_sd_getgeome,
|
||||
.sync = scsi_sd_sync,
|
||||
.erase = scsi_sd_erase,
|
||||
.autorefresh = scsi_sd_autorefresh,
|
||||
};
|
||||
|
||||
rt_err_t scsi_sd_probe(struct rt_scsi_device *sdev)
|
||||
{
|
||||
rt_err_t err;
|
||||
union
|
||||
{
|
||||
struct rt_scsi_read_capacity10_data capacity10;
|
||||
struct rt_scsi_read_capacity16_data capacity16;
|
||||
} data;
|
||||
struct scsi_sd *ssd = rt_calloc(1, sizeof(*ssd));
|
||||
|
||||
if (!ssd)
|
||||
{
|
||||
return -RT_ENOMEM;
|
||||
}
|
||||
|
||||
if ((ssd->sd_id = rt_dm_ida_alloc(&sd_ida)) < 0)
|
||||
{
|
||||
return -RT_EFULL;
|
||||
}
|
||||
|
||||
sdev->priv = ssd;
|
||||
ssd->sdev = sdev;
|
||||
ssd->parent.ida = &scsi_sd_ida;
|
||||
ssd->parent.parallel_io = RT_FALSE;
|
||||
ssd->parent.ops = &scsi_sd_ops;
|
||||
ssd->parent.max_partitions = RT_BLK_PARTITION_MAX;
|
||||
|
||||
if ((err = rt_scsi_read_capacity10(sdev, &data.capacity10)))
|
||||
{
|
||||
goto _fail;
|
||||
}
|
||||
if (data.capacity10.last_block == 0xffffffff)
|
||||
{
|
||||
if ((err = rt_scsi_read_capacity16(sdev, &data.capacity16)))
|
||||
{
|
||||
goto _fail;
|
||||
}
|
||||
ssd->use16 = RT_TRUE;
|
||||
}
|
||||
ssd->geometry.bytes_per_sector = sdev->block_size;
|
||||
ssd->geometry.block_size = sdev->block_size;
|
||||
ssd->geometry.sector_count = sdev->last_block + 1;
|
||||
|
||||
rt_dm_dev_set_name(&ssd->parent.parent, "sd%c%c", letter_name(ssd->sd_id));
|
||||
|
||||
if ((err = rt_hw_blk_disk_register(&ssd->parent)))
|
||||
{
|
||||
goto _fail;
|
||||
}
|
||||
|
||||
return RT_EOK;
|
||||
|
||||
_fail:
|
||||
rt_dm_ida_free(&sd_ida, ssd->sd_id);
|
||||
rt_free(ssd);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
rt_err_t scsi_sd_remove(struct rt_scsi_device *sdev)
|
||||
{
|
||||
struct scsi_sd *ssd = sdev->priv;
|
||||
|
||||
rt_dm_ida_free(&sd_ida, ssd->sd_id);
|
||||
|
||||
return rt_hw_blk_disk_unregister(&ssd->parent);
|
||||
}
|
Reference in New Issue
Block a user