2012-01-14 16:36:32 +04:00
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// $Id$
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
2012-08-31 00:41:25 +04:00
|
|
|
|
// SCSI emulation layer (ported from QEMU)
|
|
|
|
|
//
|
|
|
|
|
// Copyright (C) 2006 CodeSourcery.
|
|
|
|
|
// Based on code by Fabrice Bellard
|
|
|
|
|
//
|
|
|
|
|
// Written by Paul Brook
|
|
|
|
|
//
|
2015-08-23 19:09:24 +03:00
|
|
|
|
// Copyright (C) 2007-2015 The Bochs Project
|
2012-01-14 16:36:32 +04:00
|
|
|
|
//
|
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
|
// version 2 of the License, or (at your option) any later version.
|
|
|
|
|
//
|
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
|
//
|
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
|
// License along with this library; if not, write to the Free Software
|
|
|
|
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
2012-08-31 00:41:25 +04:00
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
2012-01-14 16:36:32 +04:00
|
|
|
|
|
|
|
|
|
// Define BX_PLUGGABLE in files that can be compiled into plugins. For
|
|
|
|
|
// platforms that require a special tag on exported symbols, BX_PLUGGABLE
|
|
|
|
|
// is used to know when we are exporting symbols and when we are importing.
|
|
|
|
|
#define BX_PLUGGABLE
|
|
|
|
|
|
|
|
|
|
#include "iodev.h"
|
|
|
|
|
|
|
|
|
|
#if BX_SUPPORT_PCI && BX_SUPPORT_PCIUSB
|
2012-08-05 22:13:38 +04:00
|
|
|
|
#include "hdimage/hdimage.h"
|
|
|
|
|
#include "hdimage/cdrom.h"
|
2012-01-14 16:36:32 +04:00
|
|
|
|
#include "scsi_device.h"
|
|
|
|
|
|
|
|
|
|
#define LOG_THIS
|
|
|
|
|
|
|
|
|
|
#define DEVICE_NAME "SCSI drive"
|
|
|
|
|
|
|
|
|
|
static SCSIRequest *free_requests = NULL;
|
|
|
|
|
static Bit32u serial_number = 12345678;
|
|
|
|
|
|
2014-01-29 00:29:12 +04:00
|
|
|
|
Bit64s scsireq_save_handler(void *class_ptr, bx_param_c *param)
|
|
|
|
|
{
|
|
|
|
|
char fname[BX_PATHNAME_LEN];
|
|
|
|
|
char path[BX_PATHNAME_LEN];
|
|
|
|
|
|
|
|
|
|
param->get_param_path(fname, BX_PATHNAME_LEN);
|
|
|
|
|
if (!strncmp(fname, "bochs.", 6)) {
|
|
|
|
|
strcpy(fname, fname+6);
|
|
|
|
|
}
|
|
|
|
|
if (SIM->get_param_string(BXPN_RESTORE_PATH)->isempty()) {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
sprintf(path, "%s/%s", SIM->get_param_string(BXPN_RESTORE_PATH)->getptr(), fname);
|
|
|
|
|
return ((scsi_device_t*)class_ptr)->save_requests(path);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void scsireq_restore_handler(void *class_ptr, bx_param_c *param, Bit64s value)
|
|
|
|
|
{
|
|
|
|
|
char fname[BX_PATHNAME_LEN];
|
|
|
|
|
char path[BX_PATHNAME_LEN];
|
|
|
|
|
|
|
|
|
|
if (value != 0) {
|
|
|
|
|
param->get_param_path(fname, BX_PATHNAME_LEN);
|
|
|
|
|
if (!strncmp(fname, "bochs.", 6)) {
|
|
|
|
|
strcpy(fname, fname+6);
|
|
|
|
|
}
|
|
|
|
|
sprintf(path, "%s/%s", SIM->get_param_string(BXPN_RESTORE_PATH)->getptr(), fname);
|
|
|
|
|
((scsi_device_t*)class_ptr)->restore_requests(path);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-01-14 16:36:32 +04:00
|
|
|
|
scsi_device_t::scsi_device_t(device_image_t *_hdimage, int _tcq,
|
|
|
|
|
scsi_completionfn _completion, void *_dev)
|
|
|
|
|
{
|
|
|
|
|
type = SCSIDEV_TYPE_DISK;
|
|
|
|
|
cdrom = NULL;
|
|
|
|
|
hdimage = _hdimage;
|
|
|
|
|
requests = NULL;
|
|
|
|
|
sense = 0;
|
|
|
|
|
tcq = _tcq;
|
|
|
|
|
completion = _completion;
|
|
|
|
|
dev = _dev;
|
|
|
|
|
cluster_size = 1;
|
|
|
|
|
locked = 0;
|
|
|
|
|
inserted = 1;
|
|
|
|
|
max_lba = (hdimage->hd_size / 512) - 1;
|
2015-09-13 23:04:41 +03:00
|
|
|
|
curr_lba = max_lba;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
sprintf(drive_serial_str, "%d", serial_number++);
|
2014-01-24 23:07:51 +04:00
|
|
|
|
seek_timer_index =
|
|
|
|
|
DEV_register_timer(this, seek_timer_handler, 1000, 0, 0, "USB HD seek");
|
2012-01-14 16:36:32 +04:00
|
|
|
|
|
2013-12-29 16:56:52 +04:00
|
|
|
|
put("SCSIHD");
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
|
2013-11-05 22:06:39 +04:00
|
|
|
|
scsi_device_t::scsi_device_t(cdrom_base_c *_cdrom, int _tcq,
|
2012-01-14 16:36:32 +04:00
|
|
|
|
scsi_completionfn _completion, void *_dev)
|
|
|
|
|
{
|
|
|
|
|
type = SCSIDEV_TYPE_CDROM;
|
|
|
|
|
cdrom = _cdrom;
|
|
|
|
|
hdimage = NULL;
|
|
|
|
|
requests = NULL;
|
|
|
|
|
sense = 0;
|
|
|
|
|
tcq = _tcq;
|
|
|
|
|
completion = _completion;
|
|
|
|
|
dev = _dev;
|
|
|
|
|
cluster_size = 4;
|
|
|
|
|
locked = 0;
|
2014-01-22 00:56:50 +04:00
|
|
|
|
inserted = 0;
|
|
|
|
|
max_lba = 0;
|
2015-09-13 23:04:41 +03:00
|
|
|
|
curr_lba = 0;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
sprintf(drive_serial_str, "%d", serial_number++);
|
2014-01-24 23:07:51 +04:00
|
|
|
|
seek_timer_index =
|
|
|
|
|
DEV_register_timer(this, seek_timer_handler, 1000, 0, 0, "USB CD seek");
|
2012-01-14 16:36:32 +04:00
|
|
|
|
|
2013-12-29 16:56:52 +04:00
|
|
|
|
put("SCSICD");
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
scsi_device_t::~scsi_device_t(void)
|
|
|
|
|
{
|
|
|
|
|
SCSIRequest *r, *next;
|
|
|
|
|
|
|
|
|
|
if (requests) {
|
|
|
|
|
r = requests;
|
|
|
|
|
while (r != NULL) {
|
|
|
|
|
next = r->next;
|
2014-01-29 00:29:12 +04:00
|
|
|
|
delete [] r->dma_buf;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
delete r;
|
|
|
|
|
r = next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (free_requests) {
|
|
|
|
|
r = free_requests;
|
|
|
|
|
while (r != NULL) {
|
|
|
|
|
next = r->next;
|
2014-01-29 00:29:12 +04:00
|
|
|
|
delete [] r->dma_buf;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
delete r;
|
|
|
|
|
r = next;
|
|
|
|
|
}
|
|
|
|
|
free_requests = NULL;
|
|
|
|
|
}
|
2014-01-24 23:07:51 +04:00
|
|
|
|
bx_pc_system.deactivate_timer(seek_timer_index);
|
|
|
|
|
bx_pc_system.unregisterTimer(seek_timer_index);
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void scsi_device_t::register_state(bx_list_c *parent, const char *name)
|
|
|
|
|
{
|
2012-02-12 22:43:20 +04:00
|
|
|
|
bx_list_c *list = new bx_list_c(parent, name, "");
|
2012-01-14 16:36:32 +04:00
|
|
|
|
new bx_shadow_num_c(list, "sense", &sense);
|
2014-01-31 22:49:10 +04:00
|
|
|
|
new bx_shadow_bool_c(list, "locked", &locked);
|
2015-09-13 23:04:41 +03:00
|
|
|
|
new bx_shadow_num_c(list, "curr_lba", &curr_lba);
|
2014-01-29 00:29:12 +04:00
|
|
|
|
bx_param_bool_c *requests = new bx_param_bool_c(list, "requests", NULL, NULL, 0);
|
|
|
|
|
requests->set_sr_handlers(this, scsireq_save_handler, scsireq_restore_handler);
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
|
2014-01-29 00:29:12 +04:00
|
|
|
|
// SCSI request handling
|
|
|
|
|
|
2012-01-14 16:36:32 +04:00
|
|
|
|
SCSIRequest* scsi_device_t::scsi_new_request(Bit32u tag)
|
|
|
|
|
{
|
|
|
|
|
SCSIRequest *r;
|
|
|
|
|
|
|
|
|
|
if (free_requests) {
|
|
|
|
|
r = free_requests;
|
|
|
|
|
free_requests = r->next;
|
|
|
|
|
} else {
|
|
|
|
|
r = new SCSIRequest;
|
2014-01-29 00:29:12 +04:00
|
|
|
|
r->dma_buf = new Bit8u[SCSI_DMA_BUF_SIZE];
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
r->tag = tag;
|
|
|
|
|
r->sector_count = 0;
|
2015-09-13 23:04:41 +03:00
|
|
|
|
r->async_supported = 0;
|
|
|
|
|
r->seek_active = 0;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
r->buf_len = 0;
|
|
|
|
|
r->status = 0;
|
|
|
|
|
|
|
|
|
|
r->next = requests;
|
|
|
|
|
requests = r;
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void scsi_device_t::scsi_remove_request(SCSIRequest *r)
|
|
|
|
|
{
|
|
|
|
|
SCSIRequest *last;
|
|
|
|
|
|
|
|
|
|
if (requests == r) {
|
|
|
|
|
requests = r->next;
|
|
|
|
|
} else {
|
|
|
|
|
last = requests;
|
|
|
|
|
while (last != NULL) {
|
|
|
|
|
if (last->next != r)
|
|
|
|
|
last = last->next;
|
|
|
|
|
else
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (last) {
|
|
|
|
|
last->next = r->next;
|
|
|
|
|
} else {
|
|
|
|
|
BX_ERROR(("orphaned request"));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
r->next = free_requests;
|
|
|
|
|
free_requests = r;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SCSIRequest* scsi_device_t::scsi_find_request(Bit32u tag)
|
|
|
|
|
{
|
|
|
|
|
SCSIRequest *r = requests;
|
|
|
|
|
while (r != NULL) {
|
|
|
|
|
if (r->tag != tag)
|
|
|
|
|
r = r->next;
|
|
|
|
|
else
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-29 00:29:12 +04:00
|
|
|
|
bx_bool scsi_device_t::save_requests(const char *path)
|
|
|
|
|
{
|
|
|
|
|
char tmppath[BX_PATHNAME_LEN];
|
2014-01-31 22:49:10 +04:00
|
|
|
|
FILE *fp, *fp2;
|
2014-01-29 00:29:12 +04:00
|
|
|
|
|
|
|
|
|
if (requests != NULL) {
|
2014-01-31 22:49:10 +04:00
|
|
|
|
fp = fopen(path, "w");
|
|
|
|
|
if (fp != NULL) {
|
2014-01-29 00:29:12 +04:00
|
|
|
|
SCSIRequest *r = requests;
|
|
|
|
|
Bit32u i = 0;
|
|
|
|
|
while (r != NULL) {
|
2014-01-31 22:49:10 +04:00
|
|
|
|
fprintf(fp, "%u = {\n", i);
|
|
|
|
|
fprintf(fp, " tag = %u\n", r->tag);
|
2014-07-09 19:39:47 +04:00
|
|
|
|
fprintf(fp, " sector = " FMT_LL "u\n", r->sector);
|
2014-01-31 22:49:10 +04:00
|
|
|
|
fprintf(fp, " sector_count = %u\n", r->sector_count);
|
|
|
|
|
fprintf(fp, " buf_len = %d\n", r->buf_len);
|
|
|
|
|
fprintf(fp, " status = %u\n", r->status);
|
2015-09-13 23:04:41 +03:00
|
|
|
|
fprintf(fp, " async_supported = %u\n", r->async_supported);
|
|
|
|
|
fprintf(fp, " seek_active = %u\n", r->seek_active);
|
2014-01-31 22:49:10 +04:00
|
|
|
|
fprintf(fp, "}\n");
|
2014-01-29 00:29:12 +04:00
|
|
|
|
if (r->buf_len > 0) {
|
|
|
|
|
sprintf(tmppath, "%s.%u", path, i);
|
2014-01-31 22:49:10 +04:00
|
|
|
|
fp2 = fopen(tmppath, "wb");
|
2014-01-29 00:29:12 +04:00
|
|
|
|
if (fp2 != NULL) {
|
|
|
|
|
fwrite(r->dma_buf, 1, (size_t)r->buf_len, fp2);
|
|
|
|
|
}
|
|
|
|
|
fclose(fp2);
|
|
|
|
|
}
|
|
|
|
|
r = r->next;
|
|
|
|
|
i++;
|
|
|
|
|
}
|
2014-01-31 22:49:10 +04:00
|
|
|
|
fclose(fp);
|
2014-01-29 00:29:12 +04:00
|
|
|
|
return 1;
|
|
|
|
|
} else {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void scsi_device_t::restore_requests(const char *path)
|
|
|
|
|
{
|
2014-01-31 22:49:10 +04:00
|
|
|
|
char line[512], pname[16], tmppath[BX_PATHNAME_LEN];
|
|
|
|
|
char *ret, *ptr;
|
|
|
|
|
FILE *fp, *fp2;
|
|
|
|
|
int i, reqid = -1;
|
|
|
|
|
Bit64s value;
|
|
|
|
|
Bit32u tag = 0;
|
|
|
|
|
SCSIRequest *r = NULL;
|
|
|
|
|
bx_bool rrq_error = 0;
|
|
|
|
|
|
|
|
|
|
fp = fopen(path, "r");
|
|
|
|
|
if (fp != NULL) {
|
|
|
|
|
do {
|
|
|
|
|
ret = fgets(line, sizeof(line)-1, fp);
|
|
|
|
|
line[sizeof(line) - 1] = '\0';
|
|
|
|
|
int len = strlen(line);
|
|
|
|
|
if ((len > 0) && (line[len-1] < ' '))
|
|
|
|
|
line[len-1] = '\0';
|
|
|
|
|
i = 0;
|
|
|
|
|
if ((ret != NULL) && strlen(line) > 0) {
|
|
|
|
|
ptr = strtok(line, " ");
|
|
|
|
|
while (ptr) {
|
|
|
|
|
if (i == 0) {
|
|
|
|
|
if (!strcmp(ptr, "}")) {
|
|
|
|
|
if (r != NULL) {
|
|
|
|
|
if (r->buf_len > 0) {
|
|
|
|
|
sprintf(tmppath, "%s.%u", path, reqid);
|
|
|
|
|
fp2 = fopen(tmppath, "wb");
|
|
|
|
|
if (fp2 != NULL) {
|
|
|
|
|
fread(r->dma_buf, 1, (size_t)r->buf_len, fp2);
|
|
|
|
|
}
|
|
|
|
|
fclose(fp2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
reqid = -1;
|
|
|
|
|
r = NULL;
|
|
|
|
|
tag = 0;
|
|
|
|
|
break;
|
|
|
|
|
} else if (reqid < 0) {
|
|
|
|
|
reqid = (int)strtol(ptr, NULL, 10);
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
strcpy(pname, ptr);
|
|
|
|
|
}
|
|
|
|
|
} else if (i == 2) {
|
|
|
|
|
if (reqid >= 0) {
|
|
|
|
|
if (!strcmp(pname, "tag")) {
|
|
|
|
|
if (tag == 0) {
|
|
|
|
|
tag = (Bit32u)strtoul(ptr, NULL, 10);
|
|
|
|
|
r = scsi_new_request(tag);
|
|
|
|
|
if (r == NULL) {
|
|
|
|
|
BX_ERROR(("restore_requests(): cannot create request"));
|
|
|
|
|
rrq_error = 1;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
BX_ERROR(("restore_requests(): data format error"));
|
|
|
|
|
rrq_error = 1;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
value = (Bit64s)strtoll(ptr, NULL, 10);
|
|
|
|
|
if (!strcmp(pname, "sector")) {
|
|
|
|
|
r->sector = (Bit64u)value;
|
|
|
|
|
} else if (!strcmp(pname, "sector_count")) {
|
|
|
|
|
r->sector_count = (Bit32u)value;
|
|
|
|
|
} else if (!strcmp(pname, "buf_len")) {
|
|
|
|
|
r->buf_len = (int)value;
|
|
|
|
|
} else if (!strcmp(pname, "status")) {
|
|
|
|
|
r->status = (Bit32u)value;
|
2015-09-13 23:04:41 +03:00
|
|
|
|
} else if (!strcmp(pname, "async_supported")) {
|
|
|
|
|
r->async_supported = (bx_bool)value;
|
|
|
|
|
} else if (!strcmp(pname, "seek_active")) {
|
|
|
|
|
r->seek_active = (bx_bool)value;
|
2014-01-31 22:49:10 +04:00
|
|
|
|
} else {
|
|
|
|
|
BX_ERROR(("restore_requests(): data format error"));
|
|
|
|
|
rrq_error = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
BX_ERROR(("restore_requests(): data format error"));
|
|
|
|
|
rrq_error = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
i++;
|
|
|
|
|
ptr = strtok(NULL, " ");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} while (!feof(fp) && !rrq_error);
|
|
|
|
|
fclose(fp);
|
|
|
|
|
} else {
|
|
|
|
|
BX_ERROR(("restore_requests(): error in file open"));
|
|
|
|
|
}
|
2014-01-29 00:29:12 +04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// SCSI command implementation
|
|
|
|
|
|
2012-01-14 16:36:32 +04:00
|
|
|
|
void scsi_device_t::scsi_command_complete(SCSIRequest *r, int status, int _sense)
|
|
|
|
|
{
|
|
|
|
|
Bit32u tag;
|
|
|
|
|
BX_DEBUG(("command complete tag=0x%x status=%d sense=%d", r->tag, status, sense));
|
|
|
|
|
sense = _sense;
|
|
|
|
|
tag = r->tag;
|
|
|
|
|
scsi_remove_request(r);
|
|
|
|
|
completion(dev, SCSI_REASON_DONE, tag, status);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void scsi_device_t::scsi_cancel_io(Bit32u tag)
|
|
|
|
|
{
|
|
|
|
|
BX_DEBUG(("cancel tag=0x%x", tag));
|
|
|
|
|
SCSIRequest *r = scsi_find_request(tag);
|
|
|
|
|
if (r) {
|
|
|
|
|
scsi_remove_request(r);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void scsi_device_t::scsi_read_complete(void *req, int ret)
|
|
|
|
|
{
|
|
|
|
|
SCSIRequest *r = (SCSIRequest *)req;
|
|
|
|
|
|
|
|
|
|
if (ret) {
|
|
|
|
|
BX_ERROR(("IO error"));
|
|
|
|
|
completion(r, SCSI_REASON_DATA, r->tag, 0);
|
|
|
|
|
scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_NO_SENSE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
BX_DEBUG(("data ready tag=0x%x len=%d", r->tag, r->buf_len));
|
|
|
|
|
|
|
|
|
|
completion(dev, SCSI_REASON_DATA, r->tag, r->buf_len);
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-23 19:09:24 +03:00
|
|
|
|
bx_bool scsi_device_t::scsi_read_data(Bit32u tag)
|
2012-01-14 16:36:32 +04:00
|
|
|
|
{
|
2014-01-22 21:56:20 +04:00
|
|
|
|
Bit32u i, n;
|
|
|
|
|
int ret = 0;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
|
|
|
|
|
SCSIRequest *r = scsi_find_request(tag);
|
|
|
|
|
if (!r) {
|
|
|
|
|
BX_ERROR(("bad read tag 0x%x", tag));
|
2015-08-23 19:09:24 +03:00
|
|
|
|
return 0;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
if (r->sector_count == (Bit32u)-1) {
|
|
|
|
|
BX_DEBUG(("read buf_len=%d", r->buf_len));
|
|
|
|
|
r->sector_count = 0;
|
|
|
|
|
completion(dev, SCSI_REASON_DATA, r->tag, r->buf_len);
|
2015-08-23 19:09:24 +03:00
|
|
|
|
return 0;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
BX_DEBUG(("read sector_count=%d", r->sector_count));
|
|
|
|
|
if (r->sector_count == 0) {
|
|
|
|
|
scsi_command_complete(r, STATUS_GOOD, SENSE_NO_SENSE);
|
2015-08-23 19:09:24 +03:00
|
|
|
|
return 0;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
n = r->sector_count;
|
|
|
|
|
if (n > (Bit32u)(SCSI_DMA_BUF_SIZE / (512 * cluster_size)))
|
|
|
|
|
n = SCSI_DMA_BUF_SIZE / (512 * cluster_size);
|
|
|
|
|
r->buf_len = n * 512 * cluster_size;
|
|
|
|
|
if (type == SCSIDEV_TYPE_CDROM) {
|
2015-09-13 23:04:41 +03:00
|
|
|
|
if (r->async_supported) {
|
|
|
|
|
if (!r->seek_active) start_seek(r);
|
|
|
|
|
return 1;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
} else {
|
2015-09-13 23:04:41 +03:00
|
|
|
|
i = 0;
|
|
|
|
|
do {
|
|
|
|
|
ret = (int)cdrom->read_block(r->dma_buf + (i * 2048), (Bit32u)(r->sector + i), 2048);
|
|
|
|
|
} while ((++i < n) && (ret == 1));
|
|
|
|
|
if (ret == 0) {
|
|
|
|
|
scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_MEDIUM_ERROR);
|
|
|
|
|
} else {
|
|
|
|
|
scsi_read_complete((void*)r, 0);
|
|
|
|
|
}
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
ret = (int)hdimage->lseek(r->sector * 512, SEEK_SET);
|
|
|
|
|
if (ret < 0) {
|
|
|
|
|
BX_ERROR(("could not lseek() hard drive image file"));
|
|
|
|
|
scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_HARDWARE_ERROR);
|
|
|
|
|
}
|
2014-01-25 21:07:10 +04:00
|
|
|
|
ret = (int)hdimage->read((bx_ptr_t)r->dma_buf, r->buf_len);
|
2012-01-14 16:36:32 +04:00
|
|
|
|
if (ret < r->buf_len) {
|
|
|
|
|
BX_ERROR(("could not read() hard drive image file"));
|
|
|
|
|
scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_HARDWARE_ERROR);
|
|
|
|
|
} else {
|
|
|
|
|
scsi_read_complete((void*)r, 0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
r->sector += n;
|
|
|
|
|
r->sector_count -= n;
|
2015-08-23 19:09:24 +03:00
|
|
|
|
return 1;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void scsi_device_t::scsi_write_complete(void *req, int ret)
|
|
|
|
|
{
|
|
|
|
|
SCSIRequest *r = (SCSIRequest *)req;
|
|
|
|
|
Bit32u len;
|
|
|
|
|
|
|
|
|
|
if (ret) {
|
|
|
|
|
BX_ERROR(("IO error"));
|
|
|
|
|
scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_HARDWARE_ERROR);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (r->sector_count == 0) {
|
|
|
|
|
scsi_command_complete(r, STATUS_GOOD, SENSE_NO_SENSE);
|
|
|
|
|
} else {
|
|
|
|
|
len = r->sector_count * 512;
|
|
|
|
|
if (len > SCSI_DMA_BUF_SIZE) {
|
|
|
|
|
len = SCSI_DMA_BUF_SIZE;
|
|
|
|
|
}
|
|
|
|
|
r->buf_len = len;
|
|
|
|
|
BX_DEBUG(("write complete tag=0x%x more=%d", r->tag, len));
|
|
|
|
|
completion(dev, SCSI_REASON_DATA, r->tag, len);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-23 19:09:24 +03:00
|
|
|
|
bx_bool scsi_device_t::scsi_write_data(Bit32u tag)
|
2012-01-14 16:36:32 +04:00
|
|
|
|
{
|
|
|
|
|
SCSIRequest *r;
|
|
|
|
|
Bit32u n;
|
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
|
|
BX_DEBUG(("write data tag=0x%x", tag));
|
|
|
|
|
r = scsi_find_request(tag);
|
|
|
|
|
if (!r) {
|
|
|
|
|
BX_ERROR(("bad write tag 0x%x", tag));
|
2015-08-23 19:09:24 +03:00
|
|
|
|
return 0;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
if (type == SCSIDEV_TYPE_DISK) {
|
|
|
|
|
n = r->buf_len / 512;
|
|
|
|
|
if (n) {
|
|
|
|
|
ret = (int)hdimage->lseek(r->sector * 512, SEEK_SET);
|
|
|
|
|
if (ret < 0) {
|
|
|
|
|
BX_ERROR(("could not lseek() hard drive image file"));
|
|
|
|
|
scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_HARDWARE_ERROR);
|
|
|
|
|
}
|
2014-01-25 21:07:10 +04:00
|
|
|
|
ret = (int)hdimage->write((bx_ptr_t)r->dma_buf, r->buf_len);
|
2012-01-14 16:36:32 +04:00
|
|
|
|
r->sector += n;
|
|
|
|
|
r->sector_count -= n;
|
|
|
|
|
if (ret < r->buf_len) {
|
|
|
|
|
BX_ERROR(("could not write() hard drive image file"));
|
|
|
|
|
scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_HARDWARE_ERROR);
|
2015-08-23 19:09:24 +03:00
|
|
|
|
return 0;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
} else {
|
|
|
|
|
scsi_write_complete((void*)r, 0);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
scsi_write_complete(r, 0);
|
2015-08-23 19:09:24 +03:00
|
|
|
|
return 0;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
BX_ERROR(("CD-ROM: write not supported"));
|
|
|
|
|
scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_HARDWARE_ERROR);
|
2015-08-23 19:09:24 +03:00
|
|
|
|
return 0;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
2015-08-23 19:09:24 +03:00
|
|
|
|
return 1;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Bit8u* scsi_device_t::scsi_get_buf(Bit32u tag)
|
|
|
|
|
{
|
|
|
|
|
SCSIRequest *r = scsi_find_request(tag);
|
|
|
|
|
if (!r) {
|
|
|
|
|
BX_ERROR(("bad buffer tag 0x%x", tag));
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
return r->dma_buf;
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-13 23:04:41 +03:00
|
|
|
|
Bit32s scsi_device_t::scsi_send_command(Bit32u tag, Bit8u *buf, int lun, bx_bool async)
|
2012-01-14 16:36:32 +04:00
|
|
|
|
{
|
|
|
|
|
Bit64u nb_sectors;
|
|
|
|
|
Bit64u lba;
|
|
|
|
|
Bit32s len;
|
2013-04-10 00:35:24 +04:00
|
|
|
|
//int cmdlen;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
int is_write;
|
|
|
|
|
Bit8u command;
|
|
|
|
|
Bit8u *outbuf;
|
|
|
|
|
SCSIRequest *r;
|
|
|
|
|
|
|
|
|
|
command = buf[0];
|
|
|
|
|
r = scsi_find_request(tag);
|
|
|
|
|
if (r) {
|
|
|
|
|
BX_ERROR(("tag 0x%x already in use", tag));
|
|
|
|
|
scsi_cancel_io(tag);
|
|
|
|
|
}
|
|
|
|
|
r = scsi_new_request(tag);
|
|
|
|
|
outbuf = r->dma_buf;
|
|
|
|
|
is_write = 0;
|
|
|
|
|
BX_DEBUG(("command: lun=%d tag=0x%x data=0x%02x", lun, tag, buf[0]));
|
|
|
|
|
switch (command >> 5) {
|
|
|
|
|
case 0:
|
|
|
|
|
lba = buf[3] | (buf[2] << 8) | ((buf[1] & 0x1f) << 16);
|
|
|
|
|
len = buf[4];
|
2013-04-10 00:35:24 +04:00
|
|
|
|
// cmdlen = 6;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
case 2:
|
|
|
|
|
lba = buf[5] | (buf[4] << 8) | (buf[3] << 16) | (buf[2] << 24);
|
|
|
|
|
len = buf[8] | (buf[7] << 8);
|
2013-04-10 00:35:24 +04:00
|
|
|
|
// cmdlen = 10;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
break;
|
|
|
|
|
case 4:
|
|
|
|
|
lba = buf[9] | (buf[8] << 8) | (buf[7] << 16) | (buf[6] << 24) |
|
|
|
|
|
((Bit64u)buf[5] << 32) | ((Bit64u)buf[4] << 40) |
|
|
|
|
|
((Bit64u)buf[3] << 48) | ((Bit64u)buf[2] << 56);
|
|
|
|
|
len = buf[13] | (buf[12] << 8) | (buf[11] << 16) | (buf[10] << 24);
|
2013-04-10 00:35:24 +04:00
|
|
|
|
// cmdlen = 16;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
break;
|
|
|
|
|
case 5:
|
|
|
|
|
lba = buf[5] | (buf[4] << 8) | (buf[3] << 16) | (buf[2] << 24);
|
|
|
|
|
len = buf[9] | (buf[8] << 8) | (buf[7] << 16) | (buf[6] << 24);
|
2013-04-10 00:35:24 +04:00
|
|
|
|
// cmdlen = 12;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
BX_ERROR(("Unsupported command length, command %x", command));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
if (lun || buf[1] >> 5) {
|
|
|
|
|
BX_ERROR(("unimplemented LUN %d", lun ? lun : buf[1] >> 5));
|
|
|
|
|
if ((command != 0x03) && (command != 0x12)) // REQUEST SENSE and INQUIRY
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
switch (command) {
|
|
|
|
|
case 0x0:
|
|
|
|
|
BX_DEBUG(("Test Unit Ready"));
|
|
|
|
|
if (!inserted)
|
|
|
|
|
goto notready;
|
|
|
|
|
break;
|
|
|
|
|
case 0x03:
|
|
|
|
|
BX_DEBUG(("request Sense (len %d)", len));
|
|
|
|
|
if (len < 4)
|
|
|
|
|
goto fail;
|
|
|
|
|
memset(outbuf, 0, 4);
|
|
|
|
|
r->buf_len = 4;
|
|
|
|
|
if ((sense == SENSE_NOT_READY) && (len >= 18)) {
|
|
|
|
|
memset(outbuf, 0, 18);
|
|
|
|
|
r->buf_len = 18;
|
|
|
|
|
outbuf[7] = 10;
|
|
|
|
|
/* asc 0x3a, ascq 0: Medium not present */
|
|
|
|
|
outbuf[12] = 0x3a;
|
|
|
|
|
outbuf[13] = 0;
|
|
|
|
|
}
|
|
|
|
|
outbuf[0] = 0xf0;
|
|
|
|
|
outbuf[1] = 0;
|
|
|
|
|
outbuf[2] = sense;
|
|
|
|
|
break;
|
|
|
|
|
case 0x12:
|
|
|
|
|
BX_DEBUG(("inquiry (len %d)", len));
|
|
|
|
|
if (buf[1] & 0x2) {
|
|
|
|
|
// Command support data - optional, not implemented
|
|
|
|
|
BX_ERROR(("optional INQUIRY command support request not implemented"));
|
|
|
|
|
goto fail;
|
|
|
|
|
} else if (buf[1] & 0x1) {
|
|
|
|
|
// Vital product data
|
|
|
|
|
Bit8u page_code = buf[2];
|
|
|
|
|
if (len < 4) {
|
|
|
|
|
BX_ERROR(("Error: Inquiry (EVPD[%02X]) buffer size %d is less than 4", page_code, len));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
switch (page_code) {
|
|
|
|
|
case 0x00:
|
|
|
|
|
// Supported page codes, mandatory
|
|
|
|
|
BX_DEBUG(("Inquiry EVPD[Supported pages] buffer size %d", len));
|
|
|
|
|
|
|
|
|
|
r->buf_len = 0;
|
|
|
|
|
|
|
|
|
|
if (type == SCSIDEV_TYPE_CDROM) {
|
|
|
|
|
outbuf[r->buf_len++] = 5;
|
|
|
|
|
} else {
|
|
|
|
|
outbuf[r->buf_len++] = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
outbuf[r->buf_len++] = 0x00; // this page
|
|
|
|
|
outbuf[r->buf_len++] = 0x00;
|
|
|
|
|
outbuf[r->buf_len++] = 3; // number of pages
|
|
|
|
|
outbuf[r->buf_len++] = 0x00; // list of supported pages (this page)
|
|
|
|
|
outbuf[r->buf_len++] = 0x80; // unit serial number
|
|
|
|
|
outbuf[r->buf_len++] = 0x83; // device identification
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 0x80:
|
|
|
|
|
{
|
|
|
|
|
int l;
|
|
|
|
|
|
|
|
|
|
// Device serial number, optional
|
|
|
|
|
if (len < 4) {
|
|
|
|
|
BX_ERROR(("Error: EVPD[Serial number] Inquiry buffer size %d too small, %d needed", len, 4));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
BX_DEBUG(("Inquiry EVPD[Serial number] buffer size %d\n", len));
|
|
|
|
|
l = BX_MIN(len, (int)strlen(drive_serial_str));
|
|
|
|
|
|
|
|
|
|
r->buf_len = 0;
|
|
|
|
|
|
|
|
|
|
// Supported page codes
|
|
|
|
|
if (type == SCSIDEV_TYPE_CDROM) {
|
|
|
|
|
outbuf[r->buf_len++] = 5;
|
|
|
|
|
} else {
|
|
|
|
|
outbuf[r->buf_len++] = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
outbuf[r->buf_len++] = 0x80; // this page
|
|
|
|
|
outbuf[r->buf_len++] = 0x00;
|
|
|
|
|
outbuf[r->buf_len++] = l;
|
|
|
|
|
memcpy(&outbuf[r->buf_len], drive_serial_str, l);
|
|
|
|
|
r->buf_len += l;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 0x83:
|
|
|
|
|
{
|
|
|
|
|
// Device identification page, mandatory
|
|
|
|
|
int max_len = 255 - 8;
|
|
|
|
|
int id_len = strlen(DEVICE_NAME);
|
|
|
|
|
if (id_len > max_len)
|
|
|
|
|
id_len = max_len;
|
|
|
|
|
|
|
|
|
|
BX_DEBUG(("Inquiry EVPD[Device identification] buffer size %d", len));
|
|
|
|
|
r->buf_len = 0;
|
|
|
|
|
if (type == SCSIDEV_TYPE_CDROM) {
|
|
|
|
|
outbuf[r->buf_len++] = 5;
|
|
|
|
|
} else {
|
|
|
|
|
outbuf[r->buf_len++] = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
outbuf[r->buf_len++] = 0x83; // this page
|
|
|
|
|
outbuf[r->buf_len++] = 0x00;
|
|
|
|
|
outbuf[r->buf_len++] = 3 + id_len;
|
|
|
|
|
|
|
|
|
|
outbuf[r->buf_len++] = 0x2; // ASCII
|
|
|
|
|
outbuf[r->buf_len++] = 0; // not officially assigned
|
|
|
|
|
outbuf[r->buf_len++] = 0; // reserved
|
|
|
|
|
outbuf[r->buf_len++] = id_len; // length of data following
|
|
|
|
|
|
|
|
|
|
memcpy(&outbuf[r->buf_len], DEVICE_NAME, id_len);
|
|
|
|
|
r->buf_len += id_len;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
BX_ERROR(("Error: unsupported Inquiry (EVPD[%02X]) buffer size %d", page_code, len));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
// done with EVPD
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
// Standard INQUIRY data
|
|
|
|
|
if (buf[2] != 0) {
|
|
|
|
|
BX_ERROR(("Error: Inquiry (STANDARD) page or code is non-zero [%02X]", buf[2]));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// PAGE CODE == 0
|
|
|
|
|
if (len < 5) {
|
|
|
|
|
BX_ERROR(("Error: Inquiry (STANDARD) buffer size %d is less than 5", len));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (len < 36) {
|
|
|
|
|
BX_ERROR(("Error: Inquiry (STANDARD) buffer size %d is less than 36 (TODO: only 5 required)", len));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(len > SCSI_MAX_INQUIRY_LEN)
|
|
|
|
|
len = SCSI_MAX_INQUIRY_LEN;
|
|
|
|
|
memset(outbuf, 0, len);
|
|
|
|
|
if (lun || buf[1] >> 5) {
|
|
|
|
|
outbuf[0] = 0x7f; // LUN not supported
|
|
|
|
|
} else if (type == SCSIDEV_TYPE_CDROM) {
|
|
|
|
|
outbuf[0] = 5;
|
|
|
|
|
outbuf[1] = 0x80;
|
|
|
|
|
memcpy(&outbuf[16], "BOCHS CD-ROM ", 16);
|
|
|
|
|
} else {
|
|
|
|
|
outbuf[0] = 0;
|
|
|
|
|
memcpy(&outbuf[16], "BOCHS HARDDISK ", 16);
|
|
|
|
|
}
|
|
|
|
|
memcpy(&outbuf[8], "BOCHS ", 8);
|
|
|
|
|
memcpy(&outbuf[32], "1.0", 4);
|
|
|
|
|
// Identify device as SCSI-3 rev 1.
|
|
|
|
|
// Some later commands are also implemented.
|
|
|
|
|
outbuf[2] = 3;
|
|
|
|
|
outbuf[3] = 2; // Format 2
|
|
|
|
|
outbuf[4] = len - 5; // Additional Length = (Len - 1) - 4
|
|
|
|
|
// Sync data transfer and TCQ.
|
|
|
|
|
outbuf[7] = 0x10 | (tcq ? 0x02 : 0);
|
|
|
|
|
r->buf_len = len;
|
|
|
|
|
break;
|
|
|
|
|
case 0x16:
|
|
|
|
|
BX_INFO(("Reserve(6)"));
|
|
|
|
|
if (buf[1] & 1)
|
|
|
|
|
goto fail;
|
|
|
|
|
break;
|
|
|
|
|
case 0x17:
|
|
|
|
|
BX_INFO(("Release(6)"));
|
|
|
|
|
if (buf[1] & 1)
|
|
|
|
|
goto fail;
|
|
|
|
|
break;
|
|
|
|
|
case 0x1a:
|
|
|
|
|
case 0x5a:
|
|
|
|
|
{
|
|
|
|
|
Bit8u *p;
|
|
|
|
|
int page;
|
|
|
|
|
|
|
|
|
|
page = buf[2] & 0x3f;
|
|
|
|
|
BX_DEBUG(("mode sense (page %d, len %d)", page, len));
|
|
|
|
|
p = outbuf;
|
|
|
|
|
memset(p, 0, 4);
|
|
|
|
|
outbuf[1] = 0; /* Default media type. */
|
|
|
|
|
outbuf[3] = 0; /* Block descriptor length. */
|
|
|
|
|
if (type == SCSIDEV_TYPE_CDROM) {
|
|
|
|
|
outbuf[2] = 0x80; /* Readonly. */
|
|
|
|
|
}
|
|
|
|
|
p += 4;
|
|
|
|
|
if ((page == 4) || (page == 5)) {
|
|
|
|
|
BX_ERROR(("mode sense: page %d not implemented", page));
|
|
|
|
|
}
|
|
|
|
|
if ((page == 8 || page == 0x3f)) {
|
|
|
|
|
/* Caching page. */
|
|
|
|
|
memset(p, 0, 20);
|
|
|
|
|
p[0] = 8;
|
|
|
|
|
p[1] = 0x12;
|
|
|
|
|
p[2] = 4; /* WCE */
|
|
|
|
|
p += 20;
|
|
|
|
|
}
|
|
|
|
|
if ((page == 0x3f || page == 0x2a)
|
|
|
|
|
&& (type == SCSIDEV_TYPE_CDROM)) {
|
|
|
|
|
/* CD Capabilities and Mechanical Status page. */
|
|
|
|
|
p[0] = 0x2a;
|
|
|
|
|
p[1] = 0x14;
|
|
|
|
|
p[2] = 3; // CD-R & CD-RW read
|
|
|
|
|
p[3] = 0; // Writing not supported
|
|
|
|
|
p[4] = 0x7f; /* Audio, composite, digital out,
|
|
|
|
|
mode 2 form 1&2, multi session */
|
|
|
|
|
p[5] = 0xff; /* CD DA, DA accurate, RW supported,
|
|
|
|
|
RW corrected, C2 errors, ISRC,
|
|
|
|
|
UPC, Bar code */
|
|
|
|
|
p[6] = 0x2d | (locked ? 2 : 0);
|
|
|
|
|
/* Locking supported, jumper present, eject, tray */
|
|
|
|
|
p[7] = 0; /* no volume & mute control, no changer */
|
|
|
|
|
p[8] = (50 * 176) >> 8; // 50x read speed
|
|
|
|
|
p[9] = (50 * 176) & 0xff;
|
|
|
|
|
p[10] = 0 >> 8; // No volume
|
|
|
|
|
p[11] = 0 & 0xff;
|
|
|
|
|
p[12] = 2048 >> 8; // 2M buffer
|
|
|
|
|
p[13] = 2048 & 0xff;
|
|
|
|
|
p[14] = (16 * 176) >> 8; // 16x read speed current
|
|
|
|
|
p[15] = (16 * 176) & 0xff;
|
|
|
|
|
p[18] = (16 * 176) >> 8; // 16x write speed
|
|
|
|
|
p[19] = (16 * 176) & 0xff;
|
|
|
|
|
p[20] = (16 * 176) >> 8; // 16x write speed current
|
|
|
|
|
p[21] = (16 * 176) & 0xff;
|
|
|
|
|
p += 22;
|
|
|
|
|
}
|
|
|
|
|
r->buf_len = p - outbuf;
|
|
|
|
|
outbuf[0] = r->buf_len - 4;
|
|
|
|
|
if (r->buf_len > (int)len)
|
|
|
|
|
r->buf_len = len;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0x1b:
|
|
|
|
|
BX_INFO(("Start Stop Unit"));
|
|
|
|
|
if (type == SCSIDEV_TYPE_CDROM && (buf[4] & 2)) {
|
|
|
|
|
if (!(buf[4] & 1)) {
|
|
|
|
|
// eject medium
|
|
|
|
|
cdrom->eject_cdrom();
|
|
|
|
|
inserted = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0x1e:
|
|
|
|
|
BX_INFO(("Prevent Allow Medium Removal (prevent = %d)", buf[4] & 3));
|
|
|
|
|
locked = buf[4] & 1;
|
|
|
|
|
break;
|
|
|
|
|
case 0x25:
|
|
|
|
|
BX_DEBUG(("Read Capacity"));
|
|
|
|
|
// The normal LEN field for this command is zero
|
|
|
|
|
memset(outbuf, 0, 8);
|
|
|
|
|
if (type == SCSIDEV_TYPE_CDROM) {
|
2014-01-22 00:56:50 +04:00
|
|
|
|
nb_sectors = max_lba;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
} else {
|
|
|
|
|
nb_sectors = hdimage->hd_size / 512;
|
2014-01-22 00:56:50 +04:00
|
|
|
|
nb_sectors--;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
/* Returned value is the address of the last sector. */
|
|
|
|
|
if (nb_sectors) {
|
|
|
|
|
outbuf[0] = (Bit8u)((nb_sectors >> 24) & 0xff);
|
|
|
|
|
outbuf[1] = (Bit8u)((nb_sectors >> 16) & 0xff);
|
|
|
|
|
outbuf[2] = (Bit8u)((nb_sectors >> 8) & 0xff);
|
|
|
|
|
outbuf[3] = (Bit8u)(nb_sectors & 0xff);
|
|
|
|
|
outbuf[4] = 0;
|
|
|
|
|
outbuf[5] = 0;
|
|
|
|
|
outbuf[6] = cluster_size * 2;
|
|
|
|
|
outbuf[7] = 0;
|
|
|
|
|
r->buf_len = 8;
|
|
|
|
|
} else {
|
|
|
|
|
notready:
|
|
|
|
|
scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_NOT_READY);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 0x08:
|
|
|
|
|
case 0x28:
|
|
|
|
|
case 0x88:
|
2014-12-01 20:06:00 +03:00
|
|
|
|
BX_DEBUG(("Read (sector " FMT_LL "d, count %d)", lba, len));
|
2015-08-23 19:09:24 +03:00
|
|
|
|
if (!inserted)
|
|
|
|
|
goto notready;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
if (lba > max_lba)
|
|
|
|
|
goto illegal_lba;
|
|
|
|
|
r->sector = lba;
|
|
|
|
|
r->sector_count = len;
|
2015-09-13 23:04:41 +03:00
|
|
|
|
r->async_supported = async;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
break;
|
|
|
|
|
case 0x0a:
|
|
|
|
|
case 0x2a:
|
|
|
|
|
case 0x8a:
|
2014-12-01 20:06:00 +03:00
|
|
|
|
BX_DEBUG(("Write (sector " FMT_LL "d, count %d)", lba, len));
|
2012-01-14 16:36:32 +04:00
|
|
|
|
if (lba > max_lba)
|
|
|
|
|
goto illegal_lba;
|
|
|
|
|
r->sector = lba;
|
|
|
|
|
r->sector_count = len;
|
|
|
|
|
is_write = 1;
|
|
|
|
|
break;
|
|
|
|
|
case 0x35:
|
2014-12-01 20:06:00 +03:00
|
|
|
|
BX_DEBUG(("Syncronise cache (sector " FMT_LL "d, count %d)", lba, len));
|
2012-01-14 16:36:32 +04:00
|
|
|
|
// TODO: flush cache
|
|
|
|
|
break;
|
|
|
|
|
case 0x43:
|
|
|
|
|
{
|
2013-11-05 22:06:39 +04:00
|
|
|
|
int start_track, format, msf, toclen = 0;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
|
|
|
|
|
if (type == SCSIDEV_TYPE_CDROM) {
|
2014-01-22 00:56:50 +04:00
|
|
|
|
if (!inserted)
|
|
|
|
|
goto notready;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
msf = buf[1] & 2;
|
|
|
|
|
format = buf[2] & 0xf;
|
|
|
|
|
start_track = buf[6];
|
|
|
|
|
BX_DEBUG(("Read TOC (track %d format %d msf %d)", start_track, format, msf >> 1));
|
|
|
|
|
cdrom->read_toc(outbuf, &toclen, msf, start_track, format);
|
|
|
|
|
if (toclen > 0) {
|
|
|
|
|
if (len > toclen)
|
|
|
|
|
len = toclen;
|
|
|
|
|
r->buf_len = len;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
BX_ERROR(("Read TOC error"));
|
|
|
|
|
goto fail;
|
|
|
|
|
} else {
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
case 0x46:
|
|
|
|
|
BX_DEBUG(("Get Configuration (rt %d, maxlen %d)", buf[1] & 3, len));
|
|
|
|
|
memset(outbuf, 0, 8);
|
|
|
|
|
/* ??? This shoud probably return much more information. For now
|
|
|
|
|
just return the basic header indicating the CD-ROM profile. */
|
|
|
|
|
outbuf[7] = 8; // CD-ROM
|
|
|
|
|
r->buf_len = 8;
|
|
|
|
|
break;
|
|
|
|
|
case 0x56:
|
|
|
|
|
BX_INFO(("Reserve(10)"));
|
|
|
|
|
if (buf[1] & 3)
|
|
|
|
|
goto fail;
|
|
|
|
|
break;
|
|
|
|
|
case 0x57:
|
|
|
|
|
BX_INFO(("Release(10)"));
|
|
|
|
|
if (buf[1] & 3)
|
|
|
|
|
goto fail;
|
|
|
|
|
break;
|
|
|
|
|
case 0xa0:
|
|
|
|
|
BX_INFO(("Report LUNs (len %d)", len));
|
|
|
|
|
if (len < 16)
|
|
|
|
|
goto fail;
|
|
|
|
|
memset(outbuf, 0, 16);
|
|
|
|
|
outbuf[3] = 8;
|
|
|
|
|
r->buf_len = 16;
|
|
|
|
|
break;
|
|
|
|
|
case 0x2f:
|
2015-07-26 09:48:55 +03:00
|
|
|
|
BX_DEBUG(("Verify (sector " FMT_LL "d, count %d)", lba, len));
|
|
|
|
|
if (lba > max_lba)
|
|
|
|
|
goto illegal_lba;
|
|
|
|
|
if (buf[1] & 2) {
|
|
|
|
|
BX_ERROR(("Verify with ByteChk not implemented yet"));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
2012-01-14 16:36:32 +04:00
|
|
|
|
break;
|
|
|
|
|
case 0x23: {
|
|
|
|
|
// USBMASS-UFI10.pdf rev 1.0 Section 4.10
|
|
|
|
|
BX_INFO(("READ FORMAT CAPACITIES (MMC)"));
|
|
|
|
|
|
|
|
|
|
unsigned len = (buf[7]<<8) | buf[8];
|
|
|
|
|
#define OUR_LEN 12
|
|
|
|
|
|
|
|
|
|
// Cap List Header
|
|
|
|
|
outbuf[0] = 0;
|
|
|
|
|
outbuf[1] = 0;
|
|
|
|
|
outbuf[2] = 0;
|
|
|
|
|
outbuf[3] = OUR_LEN;
|
|
|
|
|
|
|
|
|
|
// Current/Max Cap Header
|
|
|
|
|
if (type == SCSIDEV_TYPE_CDROM) {
|
2014-01-22 00:56:50 +04:00
|
|
|
|
nb_sectors = max_lba;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
} else {
|
|
|
|
|
nb_sectors = (hdimage->hd_size / 512);
|
2014-01-22 00:56:50 +04:00
|
|
|
|
nb_sectors--;
|
2012-01-14 16:36:32 +04:00
|
|
|
|
}
|
|
|
|
|
/* Returned value is the address of the last sector. */
|
|
|
|
|
outbuf[4] = (Bit8u)((nb_sectors >> 24) & 0xff);
|
|
|
|
|
outbuf[5] = (Bit8u)((nb_sectors >> 16) & 0xff);
|
|
|
|
|
outbuf[6] = (Bit8u)((nb_sectors >> 8) & 0xff);
|
|
|
|
|
outbuf[7] = (Bit8u)(nb_sectors & 0xff);
|
|
|
|
|
outbuf[8] = 2; // formatted (1 = unformatted)
|
|
|
|
|
outbuf[9] = 0;
|
|
|
|
|
outbuf[10] = cluster_size * 2;
|
|
|
|
|
outbuf[11] = 0;
|
|
|
|
|
|
|
|
|
|
r->buf_len = (len < OUR_LEN) ? len : OUR_LEN;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
BX_ERROR(("Unknown SCSI command (%2.2x)", buf[0]));
|
|
|
|
|
fail:
|
|
|
|
|
scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_ILLEGAL_REQUEST);
|
|
|
|
|
return 0;
|
|
|
|
|
illegal_lba:
|
|
|
|
|
scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_HARDWARE_ERROR);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
if (r->sector_count == 0 && r->buf_len == 0) {
|
|
|
|
|
scsi_command_complete(r, STATUS_GOOD, SENSE_NO_SENSE);
|
|
|
|
|
}
|
|
|
|
|
len = r->sector_count * 512 * cluster_size + r->buf_len;
|
|
|
|
|
if (is_write) {
|
|
|
|
|
return -len;
|
|
|
|
|
} else {
|
|
|
|
|
if (!r->sector_count)
|
|
|
|
|
r->sector_count = (Bit32u) -1;
|
|
|
|
|
return len;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-22 00:56:50 +04:00
|
|
|
|
void scsi_device_t::set_inserted(bx_bool value)
|
|
|
|
|
{
|
|
|
|
|
inserted = value;
|
|
|
|
|
if (inserted) {
|
|
|
|
|
max_lba = cdrom->capacity() - 1;
|
2015-09-13 23:04:41 +03:00
|
|
|
|
curr_lba = max_lba;
|
2014-01-22 00:56:50 +04:00
|
|
|
|
} else {
|
|
|
|
|
max_lba = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-13 23:04:41 +03:00
|
|
|
|
void scsi_device_t::start_seek(SCSIRequest *r)
|
|
|
|
|
{
|
|
|
|
|
Bit64s new_pos, prev_pos, max_pos;
|
|
|
|
|
Bit32u seek_time;
|
|
|
|
|
double fSeekBase, fSeekTime;
|
|
|
|
|
|
|
|
|
|
max_pos = max_lba;
|
|
|
|
|
prev_pos = curr_lba;
|
|
|
|
|
new_pos = r->sector + r->sector_count;
|
|
|
|
|
if (type == SCSIDEV_TYPE_CDROM) {
|
|
|
|
|
fSeekBase = 80000.0;
|
|
|
|
|
} else {
|
|
|
|
|
fSeekBase = 5000.0;
|
|
|
|
|
}
|
|
|
|
|
fSeekTime = fSeekBase * (double)abs((int)(new_pos - prev_pos + 1)) / (max_pos + 1);
|
|
|
|
|
seek_time = (fSeekTime > 10.0) ? (Bit32u)fSeekTime : 10;
|
|
|
|
|
bx_pc_system.activate_timer(seek_timer_index, seek_time, 0);
|
|
|
|
|
bx_pc_system.setTimerParam(seek_timer_index, r->tag);
|
|
|
|
|
r->seek_active = 1;
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-24 23:07:51 +04:00
|
|
|
|
void scsi_device_t::seek_timer_handler(void *this_ptr)
|
|
|
|
|
{
|
|
|
|
|
scsi_device_t *class_ptr = (scsi_device_t *) this_ptr;
|
|
|
|
|
class_ptr->seek_timer();
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-13 23:04:41 +03:00
|
|
|
|
void scsi_device_t::seek_timer()
|
2015-09-03 21:54:51 +03:00
|
|
|
|
{
|
2015-09-13 23:04:41 +03:00
|
|
|
|
Bit32u i, n, tag = bx_pc_system.triggeredTimerParam();
|
|
|
|
|
SCSIRequest *r = scsi_find_request(tag);
|
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
|
|
n = r->sector_count;
|
|
|
|
|
if (n > (Bit32u)(SCSI_DMA_BUF_SIZE / (512 * cluster_size)))
|
|
|
|
|
n = SCSI_DMA_BUF_SIZE / (512 * cluster_size);
|
|
|
|
|
if (type == SCSIDEV_TYPE_CDROM) {
|
|
|
|
|
i = 0;
|
|
|
|
|
curr_lba = r->sector;
|
|
|
|
|
do {
|
|
|
|
|
ret = (int)cdrom->read_block(r->dma_buf + (i * 2048), (Bit32u)(r->sector + i), 2048);
|
|
|
|
|
curr_lba++;
|
|
|
|
|
} while ((++i < n) && (ret == 1));
|
|
|
|
|
if (ret == 0) {
|
|
|
|
|
scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_MEDIUM_ERROR);
|
|
|
|
|
} else {
|
|
|
|
|
scsi_read_complete((void*)r, 0);
|
|
|
|
|
}
|
|
|
|
|
r->sector += n;
|
|
|
|
|
r->sector_count -= n;
|
|
|
|
|
r->seek_active = 0;
|
|
|
|
|
}
|
2015-09-03 21:54:51 +03:00
|
|
|
|
}
|
|
|
|
|
|
2015-09-13 23:04:41 +03:00
|
|
|
|
// Turn on BX_DEBUG messages at connection time
|
|
|
|
|
void scsi_device_t::set_debug_mode()
|
2014-01-24 23:07:51 +04:00
|
|
|
|
{
|
2015-09-13 23:04:41 +03:00
|
|
|
|
setonoff(LOGLEV_DEBUG, ACT_REPORT);
|
2014-01-24 23:07:51 +04:00
|
|
|
|
}
|
|
|
|
|
|
2012-01-14 16:36:32 +04:00
|
|
|
|
#endif // BX_SUPPORT_PCI && BX_SUPPORT_PCIUSB
|