【问题标题】:SCSI Read (10) and Write (10) with the SCSI Generic Interface使用 SCSI 通用接口的 SCSI 读取 (10) 和写入 (10)
【发布时间】:2012-08-05 17:53:49
【问题描述】:

我尝试向 SSD 发出 scsi read(10)write(10)。我使用this example code 作为参考/基本代码。

这是我的 scsi 阅读:

#define READ_REPLY_LEN 32
#define READ_CMDLEN 10
void scsi_read()
{
  unsigned char Readbuffer[ SCSI_OFF + READ_REPLY_LEN ];
  unsigned char cmdblk [ READ_CMDLEN ] =
      {        0x28,  /* command */
                  0,  /* lun/reserved */
                  0,  /* lba */
                  0,  /* lba */
                  0,  /* lba */
                  0,  /* lba */
                  0,  /* reserved */
                  0,  /* transfer length */
     READ_REPLY_LEN,  /* transfer length */
                  0 };/* reserved/flag/link */
  memset(Readbuffer,0,sizeof(Readbuffer));
  memcpy( cmd + SCSI_OFF, cmdblk, sizeof(cmdblk) );

  /*
   * +------------------+
   * | struct sg_header | <- cmd
   * +------------------+
   * | copy of cmdblk   | <- cmd + SCSI_OFF
   * +------------------+
   */

  if (handle_scsi_cmd(sizeof(cmdblk), 0, cmd,
                      sizeof(Readbuffer) - SCSI_OFF, Readbuffer )) {
      fprintf( stderr, "read failed\n" );
      exit(2);
  }
  hex_dump(Readbuffer,sizeof(Readbuffer));
}

这是我的 scsi 写的:

void scsi_write ( void )
{
  unsigned char Writebuffer[SCSI_OFF];
  unsigned char cmdblk [] =
      {        0x2A,  /* 0: command */
                  0,  /* 1: lun/reserved */
                  0,  /* 2: LBA */
                  0,  /* 3: LBA */
                  0,  /* 4: LBA */
                  0,  /* 5: LBA */
                  0,  /* 6: reserved */
                  0,  /* 7: transfer length */
                  0,  /* 8: transfer length */
                  0 };/* 9: control */

  memset(Writebuffer,0,sizeof(Writebuffer));
  memcpy( cmd + SCSI_OFF, cmdblk, sizeof(cmdblk) );
  cmd[SCSI_OFF+sizeof(cmdblk)+0] = 'A';
  cmd[SCSI_OFF+sizeof(cmdblk)+1] = 'b';
  cmd[SCSI_OFF+sizeof(cmdblk)+2] = 'c';
  cmd[SCSI_OFF+sizeof(cmdblk)+3] = 'd';
  cmd[SCSI_OFF+sizeof(cmdblk)+4] = 'e';
  cmd[SCSI_OFF+sizeof(cmdblk)+5] = 'f';
  cmd[SCSI_OFF+sizeof(cmdblk)+6] = 'g';
  cmd[SCSI_OFF+sizeof(cmdblk)+7] = 0;
  /*
   * +------------------+
   * | struct sg_header | <- cmd
   * +------------------+
   * | copy of cmdblk   | <- cmd + SCSI_OFF
   * +------------------+
   * | data to write    | 
   * +------------------+
   */

  if (handle_scsi_cmd(sizeof(cmdblk), 8, cmd, 
                      sizeof(Writebuffer) - SCSI_OFF, Writebuffer )) {
      fprintf( stderr, "write failed\n" );
      exit(2);
  }
}

在下面的例子中我会这样做

  1. scsi 读取
  2. scsi 写入
  3. scsi 读取

然后我打印写入数据(scsi write)和读取数据(scsi read)的 hexdumps

Read(10)
[0000]   00 00 00 44 00 00 00 44   00 00 00 00 00 00 00 00   ...D...D ........
[0010]   00 2C 00 00 00 00 00 00   00 00 00 00 00 00 00 00   ........ ........
[0020]   00 00 00 00 00 00 00 00   00 00 00 00 00 00 00 00   ........ ........
[0030]   00 00 00 00 00 00 00 00   00 00 00 00 00 00 00 00   ........ ........
[0040]   00 00 00 00                                         ....

Write(10):
[0000]   00 00 00 00 00 00 00 24   00 00 00 00 00 00 00 00   ........ ........
[0010]   00 00 00 00 00 00 00 00   00 00 00 00 00 00 00 00   ........ ........
[0020]   00 00 00 00 2A 00 00 00   00 00 00 00 00 00 41 62   ........ ......Ab
[0030]   63 64 65 66 67 00                                   cdefg.

Read(10):
[0000]   00 00 00 44 00 00 00 44   00 00 00 00 00 00 00 00   ...D...D ........
[0010]   04 00 20 00 70 00 02 00   00 00 00 0A 00 00 00 00   ....p... ........
[0020]   04 00 00 00 41 62 63 64   65 66 67 00 00 00 00 00   ....Abcd efg.....
[0030]   00 00 00 00 00 00 00 00   00 00 00 00 00 00 00 00   ........ ........
[0040]   00 00 00 00                                         ....

再次运行这三个命令后,我应该在第一次阅读时阅读Abcdefg。正确的?但是再次运行它们并没有改变。您现在可以假设,我使用的内存仍然具有来自先前函数的数据,但即使我在 sys_read() 发生之前运行 memset(Readbuff,0,sizeof(Readbuff)),我也会得到相同的结果。

我假设,我尝试写入的 LBA 可能被禁止写入,我读取了缓存。但是从 0x00-0xFF 对 LBA 地址进行交互没有任何改变 - 这意味着,我读取了相同的数据 (Abcdefg)。

您知道使用 scsi 通用接口进行 scsi 读取或写入的示例实现吗?

【问题讨论】:

    标签: linux linux-device-driver scsi


    【解决方案1】:

    在 SCSI 中,LBA 的单位和传输长度以块为单位,有时称为扇区。这几乎总是 512 字节。因此,您不能只读取或写入 32 个字节。至少,您必须执行 512 字节 == 一个块。这一点是您需要解决的大部分问题。

    在您的 scsi_write 实现中,您的传输长度为零,因此它实际上不会写入任何数据。

    您应该为 CDB 和写入/读取数据使用不同的缓冲区。我怀疑对这些缓冲区的混淆导致您的实现写到您的静态分配数组之一的末尾和您的 ReadBuffer 上。在valgrind 下运行它,看看会出现什么。

    最后,handle_scsi_cmd 中的任何内容都可能出现很多问题。设置数据传输可能很棘手......特别是,请确保您直接了解数据在 I/O 标头的 dxfer_direction 中的传输方式:SG_DXFER_TO_DEV 用于写入,SG_DXFER_FROM_DEV 用于读取。

    查看此示例,了解如何进行读取 (16)。这更符合您想要完成的目标。

    https://github.com/hreinecke/sg3_utils/blob/master/examples/sg_simple16.c

    【讨论】:

    • 非常感谢。 sg3_utils 的例子对我帮助很大,现在可以了。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2020-09-06
    • 1970-01-01
    • 2017-08-22
    • 2013-05-02
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多