SATN fixes (Blue Swirl).
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@1947 c046a42c-6fe2-441c-8c8c-71466251a162
This commit is contained in:
parent
7d8406be69
commit
9f149aa9c1
77
hw/esp.c
77
hw/esp.c
@ -59,6 +59,9 @@ struct ESPState {
|
|||||||
int dma;
|
int dma;
|
||||||
SCSIDevice *scsi_dev[MAX_DISKS];
|
SCSIDevice *scsi_dev[MAX_DISKS];
|
||||||
SCSIDevice *current_dev;
|
SCSIDevice *current_dev;
|
||||||
|
uint8_t cmdbuf[TI_BUFSZ];
|
||||||
|
int cmdlen;
|
||||||
|
int do_cmd;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define STAT_DO 0x00
|
#define STAT_DO 0x00
|
||||||
@ -79,17 +82,14 @@ struct ESPState {
|
|||||||
#define SEQ_0 0x0
|
#define SEQ_0 0x0
|
||||||
#define SEQ_CD 0x4
|
#define SEQ_CD 0x4
|
||||||
|
|
||||||
static void handle_satn(ESPState *s)
|
static int get_cmd(ESPState *s, uint8_t *buf)
|
||||||
{
|
{
|
||||||
uint8_t buf[32];
|
|
||||||
uint32_t dmaptr, dmalen;
|
uint32_t dmaptr, dmalen;
|
||||||
int target;
|
int target;
|
||||||
int32_t datalen;
|
|
||||||
int lun;
|
|
||||||
|
|
||||||
dmalen = s->wregs[0] | (s->wregs[1] << 8);
|
dmalen = s->wregs[0] | (s->wregs[1] << 8);
|
||||||
target = s->wregs[4] & 7;
|
target = s->wregs[4] & 7;
|
||||||
DPRINTF("Select with ATN len %d target %d\n", dmalen, target);
|
DPRINTF("get_cmd: len %d target %d\n", dmalen, target);
|
||||||
if (s->dma) {
|
if (s->dma) {
|
||||||
dmaptr = iommu_translate(s->espdmaregs[1]);
|
dmaptr = iommu_translate(s->espdmaregs[1]);
|
||||||
DPRINTF("DMA Direction: %c, addr 0x%8.8x\n",
|
DPRINTF("DMA Direction: %c, addr 0x%8.8x\n",
|
||||||
@ -100,8 +100,6 @@ static void handle_satn(ESPState *s)
|
|||||||
memcpy(&buf[1], s->ti_buf, dmalen);
|
memcpy(&buf[1], s->ti_buf, dmalen);
|
||||||
dmalen++;
|
dmalen++;
|
||||||
}
|
}
|
||||||
DPRINTF("busid 0x%x\n", buf[0]);
|
|
||||||
lun = buf[0] & 7;
|
|
||||||
|
|
||||||
s->ti_size = 0;
|
s->ti_size = 0;
|
||||||
s->ti_rptr = 0;
|
s->ti_rptr = 0;
|
||||||
@ -114,9 +112,19 @@ static void handle_satn(ESPState *s)
|
|||||||
s->rregs[6] = SEQ_0;
|
s->rregs[6] = SEQ_0;
|
||||||
s->espdmaregs[0] |= DMA_INTR;
|
s->espdmaregs[0] |= DMA_INTR;
|
||||||
pic_set_irq(s->irq, 1);
|
pic_set_irq(s->irq, 1);
|
||||||
return;
|
return 0;
|
||||||
}
|
}
|
||||||
s->current_dev = s->scsi_dev[target];
|
s->current_dev = s->scsi_dev[target];
|
||||||
|
return dmalen;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void do_cmd(ESPState *s, uint8_t *buf)
|
||||||
|
{
|
||||||
|
int32_t datalen;
|
||||||
|
int lun;
|
||||||
|
|
||||||
|
DPRINTF("do_cmd: busid 0x%x\n", buf[0]);
|
||||||
|
lun = buf[0] & 7;
|
||||||
datalen = scsi_send_command(s->current_dev, 0, &buf[1], lun);
|
datalen = scsi_send_command(s->current_dev, 0, &buf[1], lun);
|
||||||
if (datalen == 0) {
|
if (datalen == 0) {
|
||||||
s->ti_size = 0;
|
s->ti_size = 0;
|
||||||
@ -136,6 +144,31 @@ static void handle_satn(ESPState *s)
|
|||||||
pic_set_irq(s->irq, 1);
|
pic_set_irq(s->irq, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void handle_satn(ESPState *s)
|
||||||
|
{
|
||||||
|
uint8_t buf[32];
|
||||||
|
int len;
|
||||||
|
|
||||||
|
len = get_cmd(s, buf);
|
||||||
|
if (len)
|
||||||
|
do_cmd(s, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_satn_stop(ESPState *s)
|
||||||
|
{
|
||||||
|
s->cmdlen = get_cmd(s, s->cmdbuf);
|
||||||
|
if (s->cmdlen) {
|
||||||
|
DPRINTF("Set ATN & Stop: cmdlen %d\n", s->cmdlen);
|
||||||
|
s->do_cmd = 1;
|
||||||
|
s->espdmaregs[1] += s->cmdlen;
|
||||||
|
s->rregs[4] = STAT_IN | STAT_TC | STAT_CD;
|
||||||
|
s->rregs[5] = INTR_BS | INTR_FC;
|
||||||
|
s->rregs[6] = SEQ_CD;
|
||||||
|
s->espdmaregs[0] |= DMA_INTR;
|
||||||
|
pic_set_irq(s->irq, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void write_response(ESPState *s)
|
static void write_response(ESPState *s)
|
||||||
{
|
{
|
||||||
uint32_t dmaptr;
|
uint32_t dmaptr;
|
||||||
@ -188,6 +221,9 @@ static void handle_ti(ESPState *s)
|
|||||||
dmalen=0x10000;
|
dmalen=0x10000;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (s->do_cmd)
|
||||||
|
minlen = (dmalen < 32) ? dmalen : 32;
|
||||||
|
else
|
||||||
minlen = (dmalen < s->ti_size) ? dmalen : s->ti_size;
|
minlen = (dmalen < s->ti_size) ? dmalen : s->ti_size;
|
||||||
DPRINTF("Transfer Information len %d\n", minlen);
|
DPRINTF("Transfer Information len %d\n", minlen);
|
||||||
if (s->dma) {
|
if (s->dma) {
|
||||||
@ -207,6 +243,15 @@ static void handle_ti(ESPState *s)
|
|||||||
}
|
}
|
||||||
DPRINTF("DMA address p %08x v %08x len %08x, from %08x, to %08x\n", dmaptr, s->espdmaregs[1] + i, len, from, to);
|
DPRINTF("DMA address p %08x v %08x len %08x, from %08x, to %08x\n", dmaptr, s->espdmaregs[1] + i, len, from, to);
|
||||||
s->ti_size -= len;
|
s->ti_size -= len;
|
||||||
|
if (s->do_cmd) {
|
||||||
|
DPRINTF("command len %d + %d\n", s->cmdlen, len);
|
||||||
|
cpu_physical_memory_read(dmaptr, &s->cmdbuf[s->cmdlen], len);
|
||||||
|
s->ti_size = 0;
|
||||||
|
s->cmdlen = 0;
|
||||||
|
s->do_cmd = 0;
|
||||||
|
do_cmd(s, s->cmdbuf);
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
if (to_device) {
|
if (to_device) {
|
||||||
cpu_physical_memory_read(dmaptr, buf, len);
|
cpu_physical_memory_read(dmaptr, buf, len);
|
||||||
scsi_write_data(s->current_dev, buf, len);
|
scsi_write_data(s->current_dev, buf, len);
|
||||||
@ -215,6 +260,7 @@ static void handle_ti(ESPState *s)
|
|||||||
cpu_physical_memory_write(dmaptr, buf, len);
|
cpu_physical_memory_write(dmaptr, buf, len);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
if (s->ti_size) {
|
if (s->ti_size) {
|
||||||
s->rregs[4] = STAT_IN | STAT_TC | (to_device ? STAT_DO : STAT_DI);
|
s->rregs[4] = STAT_IN | STAT_TC | (to_device ? STAT_DO : STAT_DI);
|
||||||
}
|
}
|
||||||
@ -222,6 +268,13 @@ static void handle_ti(ESPState *s)
|
|||||||
s->rregs[6] = 0;
|
s->rregs[6] = 0;
|
||||||
s->rregs[7] = 0;
|
s->rregs[7] = 0;
|
||||||
s->espdmaregs[0] |= DMA_INTR;
|
s->espdmaregs[0] |= DMA_INTR;
|
||||||
|
} else if (s->do_cmd) {
|
||||||
|
DPRINTF("command len %d\n", s->cmdlen);
|
||||||
|
s->ti_size = 0;
|
||||||
|
s->cmdlen = 0;
|
||||||
|
s->do_cmd = 0;
|
||||||
|
do_cmd(s, s->cmdbuf);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
pic_set_irq(s->irq, 1);
|
pic_set_irq(s->irq, 1);
|
||||||
}
|
}
|
||||||
@ -237,6 +290,7 @@ static void esp_reset(void *opaque)
|
|||||||
s->ti_rptr = 0;
|
s->ti_rptr = 0;
|
||||||
s->ti_wptr = 0;
|
s->ti_wptr = 0;
|
||||||
s->dma = 0;
|
s->dma = 0;
|
||||||
|
s->do_cmd = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static uint32_t esp_mem_readb(void *opaque, target_phys_addr_t addr)
|
static uint32_t esp_mem_readb(void *opaque, target_phys_addr_t addr)
|
||||||
@ -291,7 +345,9 @@ static void esp_mem_writeb(void *opaque, target_phys_addr_t addr, uint32_t val)
|
|||||||
break;
|
break;
|
||||||
case 2:
|
case 2:
|
||||||
// FIFO
|
// FIFO
|
||||||
if ((s->rregs[4] & 6) == 0) {
|
if (s->do_cmd) {
|
||||||
|
s->cmdbuf[s->cmdlen++] = val & 0xff;
|
||||||
|
} else if ((s->rregs[4] & 6) == 0) {
|
||||||
uint8_t buf;
|
uint8_t buf;
|
||||||
buf = val & 0xff;
|
buf = val & 0xff;
|
||||||
s->ti_size--;
|
s->ti_size--;
|
||||||
@ -348,11 +404,12 @@ static void esp_mem_writeb(void *opaque, target_phys_addr_t addr, uint32_t val)
|
|||||||
DPRINTF("Set ATN (%2.2x)\n", val);
|
DPRINTF("Set ATN (%2.2x)\n", val);
|
||||||
break;
|
break;
|
||||||
case 0x42:
|
case 0x42:
|
||||||
|
DPRINTF("Set ATN (%2.2x)\n", val);
|
||||||
handle_satn(s);
|
handle_satn(s);
|
||||||
break;
|
break;
|
||||||
case 0x43:
|
case 0x43:
|
||||||
DPRINTF("Set ATN & stop (%2.2x)\n", val);
|
DPRINTF("Set ATN & stop (%2.2x)\n", val);
|
||||||
handle_satn(s);
|
handle_satn_stop(s);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
DPRINTF("Unhandled ESP command (%2.2x)\n", val);
|
DPRINTF("Unhandled ESP command (%2.2x)\n", val);
|
||||||
|
Loading…
Reference in New Issue
Block a user