775 lines
22 KiB
C
775 lines
22 KiB
C
/*
|
|
* OpenBIOS LSI driver
|
|
*
|
|
* Copyright (C) 2018 Mark Cave-Ayland <mark.cave-ayland@ilande.co.uk>
|
|
*
|
|
* Based upon drivers/esp.c
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* version 2
|
|
*
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "libc/byteorder.h"
|
|
#include "libc/vsprintf.h"
|
|
#include "libopenbios/bindings.h"
|
|
#include "drivers/drivers.h"
|
|
#include "scsi.h"
|
|
|
|
typedef struct sd_private sd_private_t;
|
|
typedef struct lsi_table lsi_table_t;
|
|
typedef struct lsi_private lsi_private_t;
|
|
|
|
struct sd_private {
|
|
unsigned int bs;
|
|
const char *media_str[2];
|
|
uint32_t sectors;
|
|
uint8_t media;
|
|
uint8_t id;
|
|
uint8_t present;
|
|
char model[40];
|
|
lsi_private_t *lsi;
|
|
};
|
|
|
|
struct lsi_table {
|
|
uint32_t id;
|
|
uint32_t id_addr;
|
|
uint32_t msg_out_len;
|
|
uint32_t msg_out_ptr;
|
|
uint32_t cmd_len;
|
|
uint32_t cmd_ptr;
|
|
uint32_t data_in_len;
|
|
uint32_t data_in_ptr;
|
|
uint32_t status_len;
|
|
uint32_t status_ptr;
|
|
uint32_t msg_in_len;
|
|
uint32_t msg_in_ptr;
|
|
};
|
|
|
|
struct lsi_private {
|
|
volatile uint8_t *mmio;
|
|
uint32_t *scripts;
|
|
uint32_t *scripts_iova;
|
|
lsi_table_t *table;
|
|
lsi_table_t *table_iova;
|
|
volatile uint8_t *buffer;
|
|
volatile uint8_t *buffer_iova;
|
|
sd_private_t sd[8];
|
|
};
|
|
|
|
#ifdef CONFIG_DEBUG_LSI
|
|
#define DPRINTF(fmt, args...) \
|
|
do { printk(fmt , ##args); } while (0)
|
|
#else
|
|
#define DPRINTF(fmt, args...)
|
|
#endif
|
|
|
|
/* DECLARE data structures for the nodes. */
|
|
DECLARE_UNNAMED_NODE(ob_sd, INSTALL_OPEN, sizeof(sd_private_t *));
|
|
DECLARE_UNNAMED_NODE(ob_lsi, INSTALL_OPEN, sizeof(lsi_private_t **));
|
|
|
|
#ifdef CONFIG_DEBUG_LSI
|
|
static void dump_drive(sd_private_t *drive)
|
|
{
|
|
printk("SCSI DRIVE @%lx:\n", (unsigned long)drive);
|
|
printk("id: %d\n", drive->id);
|
|
printk("media: %s\n", drive->media_str[0]);
|
|
printk("media: %s\n", drive->media_str[1]);
|
|
printk("model: %s\n", drive->model);
|
|
printk("sectors: %d\n", drive->sectors);
|
|
printk("present: %d\n", drive->present);
|
|
printk("bs: %d\n", drive->bs);
|
|
}
|
|
#endif
|
|
|
|
#define PHASE_DO 0
|
|
#define PHASE_DI 1
|
|
#define PHASE_CMD 2
|
|
#define PHASE_ST 3
|
|
#define PHASE_MO 6
|
|
#define PHASE_MI 7
|
|
|
|
#define LSI_DSTAT 0x0c
|
|
#define LSI_DSA 0x10
|
|
#define LSI_ISTAT0 0x14
|
|
#define LSI_DSP 0x2c
|
|
#define LSI_SIST0 0x42
|
|
#define LSI_SIST1 0x43
|
|
|
|
#define LSI_ISTAT0_DIP 0x01
|
|
#define LSI_ISTAT0_SIP 0x02
|
|
|
|
/* Indirection table */
|
|
#define LSI_TABLE_OFFSET(x) (((uintptr_t)&(x)) - ((uintptr_t)lsi->table))
|
|
|
|
#define LSI_TABLE_MSG_OUT_OFFSET 0x0
|
|
#define LSI_TABLE_CMD_OFFSET 0x2
|
|
#define LSI_TABLE_DATA_OFFSET 0x20
|
|
#define LSI_TABLE_STATUS_OFFSET 0x10
|
|
#define LSI_TABLE_MSG_IN_OFFSET 0x12
|
|
|
|
static void
|
|
init_scripts(lsi_private_t *lsi)
|
|
{
|
|
/* Initialise SCRIPTS for the commands we are interested in */
|
|
|
|
/* 1 - INQUIRY / READ CAPACITY */
|
|
|
|
/* 1.0 Select with ATN */
|
|
lsi->scripts[0x0] = __cpu_to_le32(0x47000000);
|
|
lsi->scripts[0x1] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->id));
|
|
|
|
/* 1.1 Select LUN */
|
|
lsi->scripts[0x2] = __cpu_to_le32(0x10000000 | (PHASE_MO << 24));
|
|
lsi->scripts[0x3] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->msg_out_len));
|
|
|
|
/* 1.2 Send command */
|
|
lsi->scripts[0x4] = __cpu_to_le32(0x10000000 | (PHASE_CMD << 24));
|
|
lsi->scripts[0x5] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->cmd_len));
|
|
|
|
/* 1.3 Data in */
|
|
lsi->scripts[0x6] = __cpu_to_le32(0x10000000 | (PHASE_DI << 24));
|
|
lsi->scripts[0x7] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->data_in_len));
|
|
|
|
/* 1.4 Status */
|
|
lsi->scripts[0x8] = __cpu_to_le32(0x10000000 | (PHASE_ST << 24));
|
|
lsi->scripts[0x9] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->status_len));
|
|
|
|
/* 1.5 Message in */
|
|
lsi->scripts[0xa] = __cpu_to_le32(0x10000000 | (PHASE_MI << 24));
|
|
lsi->scripts[0xb] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->msg_in_len));
|
|
|
|
/* 1.6 Wait disconnect */
|
|
lsi->scripts[0xc] = __cpu_to_le32(0x48000000);
|
|
lsi->scripts[0xd] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->id));
|
|
|
|
/* 1.7 Interrupt */
|
|
lsi->scripts[0xe] = __cpu_to_le32(0x98080000);
|
|
lsi->scripts[0xf] = 0x0;
|
|
|
|
|
|
/* 2 - TEST UNIT READY */
|
|
|
|
/* 2.0 Select with ATN */
|
|
lsi->scripts[0x10] = __cpu_to_le32(0x47000000);
|
|
lsi->scripts[0x11] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->id));
|
|
|
|
/* 2.1 Select LUN */
|
|
lsi->scripts[0x12] = __cpu_to_le32(0x10000000 | (PHASE_MO << 24));
|
|
lsi->scripts[0x13] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->msg_out_len));
|
|
|
|
/* 2.2 Send command */
|
|
lsi->scripts[0x14] = __cpu_to_le32(0x10000000 | (PHASE_CMD << 24));
|
|
lsi->scripts[0x15] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->cmd_len));
|
|
|
|
/* 2.3 Status */
|
|
lsi->scripts[0x16] = __cpu_to_le32(0x10000000 | (PHASE_ST << 24));
|
|
lsi->scripts[0x17] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->status_len));
|
|
|
|
/* 2.4 Message in */
|
|
lsi->scripts[0x18] = __cpu_to_le32(0x10000000 | (PHASE_MI << 24));
|
|
lsi->scripts[0x19] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->msg_in_len));
|
|
|
|
/* 2.5 Wait disconnect */
|
|
lsi->scripts[0x1a] = __cpu_to_le32(0x48000000);
|
|
lsi->scripts[0x1b] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->id));
|
|
|
|
/* 2.6 Interrupt */
|
|
lsi->scripts[0x1c] = __cpu_to_le32(0x98080000);
|
|
lsi->scripts[0x1d] = 0x0;
|
|
|
|
|
|
/* 3 - READ 10 */
|
|
|
|
/* 3.0 Select with ATN */
|
|
lsi->scripts[0x20] = __cpu_to_le32(0x47000000);
|
|
lsi->scripts[0x21] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->id));
|
|
|
|
/* 3.1 Select LUN */
|
|
lsi->scripts[0x22] = __cpu_to_le32(0x10000000 | (PHASE_MO << 24));
|
|
lsi->scripts[0x23] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->msg_out_len));
|
|
|
|
/* 3.2 Send command */
|
|
lsi->scripts[0x24] = __cpu_to_le32(0x10000000 | (PHASE_CMD << 24));
|
|
lsi->scripts[0x25] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->cmd_len));
|
|
|
|
/* 3.3 Message in */
|
|
lsi->scripts[0x26] = __cpu_to_le32(0x10000000 | (PHASE_MI << 24));
|
|
lsi->scripts[0x27] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->msg_in_len));
|
|
|
|
/* 3.6 Interrupt */
|
|
lsi->scripts[0x28] = __cpu_to_le32(0x98080000);
|
|
lsi->scripts[0x29] = 0x0;
|
|
|
|
/* 3.7 Wait reselect */
|
|
lsi->scripts[0x2a] = __cpu_to_le32(0x50000000);
|
|
lsi->scripts[0x2b] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->id));
|
|
|
|
/* 3.8 Message in */
|
|
lsi->scripts[0x2c] = __cpu_to_le32(0x10000000 | (PHASE_MI << 24));
|
|
lsi->scripts[0x2d] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->msg_in_len));
|
|
|
|
/* 3.9 Data in */
|
|
lsi->scripts[0x2e] = __cpu_to_le32(0x10000000 | (PHASE_DI << 24));
|
|
lsi->scripts[0x2f] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->data_in_len));
|
|
|
|
/* 3.10 Wait disconnect */
|
|
lsi->scripts[0x30] = __cpu_to_le32(0x48000000);
|
|
lsi->scripts[0x31] = __cpu_to_le32(LSI_TABLE_OFFSET(lsi->table->id));
|
|
|
|
/* 3.11 Interrupt */
|
|
lsi->scripts[0x32] = __cpu_to_le32(0x98080000);
|
|
lsi->scripts[0x33] = 0x0;
|
|
}
|
|
|
|
static void
|
|
init_table(lsi_private_t *lsi)
|
|
{
|
|
uint32_t dsa;
|
|
|
|
/* Initialise indirect table */
|
|
lsi->table->msg_out_ptr = __cpu_to_le32((uintptr_t)&lsi->buffer_iova[LSI_TABLE_MSG_OUT_OFFSET]);
|
|
lsi->table->cmd_ptr = __cpu_to_le32((uintptr_t)&lsi->buffer_iova[LSI_TABLE_CMD_OFFSET]);
|
|
lsi->table->data_in_ptr = __cpu_to_le32((uintptr_t)&lsi->buffer_iova[LSI_TABLE_DATA_OFFSET]);
|
|
lsi->table->status_ptr = __cpu_to_le32((uintptr_t)&lsi->buffer_iova[LSI_TABLE_STATUS_OFFSET]);
|
|
lsi->table->msg_in_ptr = __cpu_to_le32((uintptr_t)&lsi->buffer_iova[LSI_TABLE_MSG_IN_OFFSET]);
|
|
|
|
/* Set the DSA to point to the base of our data table */
|
|
dsa = (uintptr_t)lsi->table_iova;
|
|
lsi->mmio[LSI_DSA] = dsa & 0xff;
|
|
lsi->mmio[LSI_DSA + 1] = (dsa >> 8) & 0xff;
|
|
lsi->mmio[LSI_DSA + 2] = (dsa >> 16) & 0xff;
|
|
lsi->mmio[LSI_DSA + 3] = (dsa >> 24) & 0xff;
|
|
}
|
|
|
|
static unsigned int
|
|
lsi_interrupt_status(lsi_private_t *lsi)
|
|
{
|
|
uint32_t istat, sist0, sist1, dstat;
|
|
|
|
/* Wait for interrupt status */
|
|
while ((istat = lsi->mmio[LSI_ISTAT0]) == 0);
|
|
|
|
if (istat & LSI_ISTAT0_SIP) {
|
|
/* If SCSI interrupt, clear SCSI interrupt registers */
|
|
sist0 = lsi->mmio[LSI_SIST0];
|
|
sist1 = lsi->mmio[LSI_SIST1];
|
|
|
|
if (sist0 != 0 || sist1 != 0) {
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
if (istat & LSI_ISTAT0_DIP) {
|
|
/* If DMA interrupt, clear DMA interrupt register */
|
|
dstat = lsi->mmio[LSI_DSTAT];
|
|
|
|
if ((dstat & 0x7f) != 0x4) {
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int
|
|
inquiry(lsi_private_t *lsi, sd_private_t *sd)
|
|
{
|
|
const char *media[2] = { "UNKNOWN", "UNKNOWN"};
|
|
uint8_t *buffer;
|
|
|
|
// Setup command = Inquiry
|
|
memset((uint8_t *)&lsi->buffer[LSI_TABLE_CMD_OFFSET], 0, 7);
|
|
lsi->buffer[LSI_TABLE_MSG_OUT_OFFSET] = 0x80;
|
|
lsi->table->msg_out_len = __cpu_to_le32(0x1);
|
|
|
|
lsi->buffer[LSI_TABLE_CMD_OFFSET] = INQUIRY;
|
|
lsi->table->cmd_len = __cpu_to_le32(0x6);
|
|
|
|
lsi->buffer[LSI_TABLE_CMD_OFFSET + 4] = 36;
|
|
lsi->table->data_in_len = __cpu_to_le32(36);
|
|
|
|
lsi->table->status_len = __cpu_to_le32(0x1);
|
|
lsi->table->msg_in_len = __cpu_to_le32(0x1);
|
|
|
|
lsi->table->id = __cpu_to_le32((sd->id << 16));
|
|
lsi->table->id_addr = __cpu_to_le32(&lsi->scripts_iova[0x2]);
|
|
|
|
/* Write DSP to start DMA engine */
|
|
uint32_t dsp = (uintptr_t)lsi->scripts_iova;
|
|
lsi->mmio[LSI_DSP] = dsp & 0xff;
|
|
lsi->mmio[LSI_DSP + 1] = (dsp >> 8) & 0xff;
|
|
lsi->mmio[LSI_DSP + 2] = (dsp >> 16) & 0xff;
|
|
lsi->mmio[LSI_DSP + 3] = (dsp >> 24) & 0xff;
|
|
|
|
if (lsi_interrupt_status(lsi)) {
|
|
sd->present = 0;
|
|
sd->media = -1;
|
|
return 0;
|
|
}
|
|
|
|
buffer = (uint8_t *)&lsi->buffer[LSI_TABLE_DATA_OFFSET];
|
|
sd->present = 1;
|
|
sd->media = buffer[0];
|
|
|
|
switch (sd->media) {
|
|
case TYPE_DISK:
|
|
media[0] = "disk";
|
|
media[1] = "hd";
|
|
break;
|
|
case TYPE_ROM:
|
|
media[0] = "cdrom";
|
|
media[1] = "cd";
|
|
break;
|
|
}
|
|
sd->media_str[0] = media[0];
|
|
sd->media_str[1] = media[1];
|
|
memcpy(sd->model, &buffer[16], 16);
|
|
sd->model[17] = '\0';
|
|
|
|
return 1;
|
|
}
|
|
|
|
static unsigned int
|
|
read_capacity(lsi_private_t *lsi, sd_private_t *sd)
|
|
{
|
|
uint8_t *buffer;
|
|
|
|
// Setup command = Read Capacity
|
|
memset((uint8_t *)&lsi->buffer[LSI_TABLE_CMD_OFFSET], 0, 11);
|
|
lsi->buffer[LSI_TABLE_MSG_OUT_OFFSET] = 0x80;
|
|
lsi->table->msg_out_len = __cpu_to_le32(0x1);
|
|
|
|
lsi->buffer[LSI_TABLE_CMD_OFFSET] = READ_CAPACITY;
|
|
lsi->table->cmd_len = __cpu_to_le32(0x11);
|
|
|
|
lsi->table->data_in_len = __cpu_to_le32(0x8);
|
|
|
|
lsi->table->status_len = __cpu_to_le32(0x1);
|
|
lsi->table->msg_in_len = __cpu_to_le32(0x1);
|
|
|
|
lsi->table->id = __cpu_to_le32((sd->id << 16));
|
|
lsi->table->id_addr = __cpu_to_le32(&lsi->scripts_iova[0x2]);
|
|
|
|
/* Write DSP to start DMA engine */
|
|
uint32_t dsp = (uintptr_t)lsi->scripts_iova;
|
|
lsi->mmio[LSI_DSP] = dsp & 0xff;
|
|
lsi->mmio[LSI_DSP + 1] = (dsp >> 8) & 0xff;
|
|
lsi->mmio[LSI_DSP + 2] = (dsp >> 16) & 0xff;
|
|
lsi->mmio[LSI_DSP + 3] = (dsp >> 24) & 0xff;
|
|
|
|
if (lsi_interrupt_status(lsi)) {
|
|
sd->sectors = 0;
|
|
sd->bs = 0;
|
|
DPRINTF("read_capacity id %d failed\n", sd->id);
|
|
return 0;
|
|
}
|
|
|
|
buffer = (uint8_t *)&lsi->buffer[LSI_TABLE_DATA_OFFSET];
|
|
sd->bs = (buffer[4] << 24) | (buffer[5] << 16) | (buffer[6] << 8) | buffer[7];
|
|
sd->sectors = ((buffer[0] << 24) | (buffer[1] << 16) | (buffer[2] << 8) | buffer[3]) * (sd->bs / 512);
|
|
|
|
DPRINTF("read_capacity id %d bs %d sectors %d\n", sd->id, sd->bs,
|
|
sd->sectors);
|
|
return 1;
|
|
}
|
|
|
|
static unsigned int
|
|
test_unit_ready(lsi_private_t *lsi, sd_private_t *sd)
|
|
{
|
|
/* Setup command = Test Unit Ready */
|
|
memset((uint8_t *)&lsi->buffer[LSI_TABLE_CMD_OFFSET], 0, 7);
|
|
lsi->buffer[LSI_TABLE_MSG_OUT_OFFSET] = 0x80;
|
|
lsi->table->msg_out_len = __cpu_to_le32(0x1);
|
|
|
|
lsi->buffer[LSI_TABLE_CMD_OFFSET] = TEST_UNIT_READY;
|
|
lsi->table->cmd_len = __cpu_to_le32(0x6);
|
|
|
|
lsi->table->status_len = __cpu_to_le32(0x1);
|
|
lsi->table->msg_in_len = __cpu_to_le32(0x1);
|
|
|
|
lsi->table->id = __cpu_to_le32((sd->id << 16));
|
|
lsi->table->id_addr = __cpu_to_le32(&lsi->scripts_iova[0x12]);
|
|
|
|
/* Write DSP to start DMA engine */
|
|
uint32_t dsp = (uintptr_t)&lsi->scripts_iova[0x10];
|
|
lsi->mmio[LSI_DSP] = dsp & 0xff;
|
|
lsi->mmio[LSI_DSP + 1] = (dsp >> 8) & 0xff;
|
|
lsi->mmio[LSI_DSP + 2] = (dsp >> 16) & 0xff;
|
|
lsi->mmio[LSI_DSP + 3] = (dsp >> 24) & 0xff;
|
|
|
|
if (lsi_interrupt_status(lsi)) {
|
|
DPRINTF("test_unit_ready id %d failed\n", sd->id);
|
|
return 0;
|
|
}
|
|
|
|
DPRINTF("test_unit_ready id %d success\n", sd->id);
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
ob_lsi_dma_alloc(__attribute__((unused)) lsi_private_t **lsi)
|
|
{
|
|
call_parent_method("dma-alloc");
|
|
}
|
|
|
|
static void
|
|
ob_lsi_dma_free(__attribute__((unused)) lsi_private_t **lsi)
|
|
{
|
|
call_parent_method("dma-free");
|
|
}
|
|
|
|
static void
|
|
ob_lsi_dma_map_in(__attribute__((unused)) lsi_private_t **lsi)
|
|
{
|
|
call_parent_method("dma-map-in");
|
|
}
|
|
|
|
static void
|
|
ob_lsi_dma_map_out(__attribute__((unused)) lsi_private_t **lsi)
|
|
{
|
|
call_parent_method("dma-map-out");
|
|
}
|
|
|
|
static void
|
|
ob_lsi_dma_sync(__attribute__((unused)) lsi_private_t **lsi)
|
|
{
|
|
call_parent_method("dma-sync");
|
|
}
|
|
|
|
// offset is in sectors
|
|
static int
|
|
ob_sd_read_sector(lsi_private_t *lsi, sd_private_t *sd, int offset)
|
|
{
|
|
uint32_t dsp;
|
|
|
|
DPRINTF("ob_sd_read_sector id %d sector=%d\n",
|
|
sd->id, offset);
|
|
|
|
// Setup command = Read(10)
|
|
memset((uint8_t *)&lsi->buffer[LSI_TABLE_CMD_OFFSET], 0, 10);
|
|
lsi->buffer[LSI_TABLE_MSG_OUT_OFFSET] = 0x80;
|
|
lsi->table->msg_out_len = __cpu_to_le32(0x1);
|
|
|
|
lsi->buffer[LSI_TABLE_CMD_OFFSET] = READ_10;
|
|
lsi->buffer[LSI_TABLE_CMD_OFFSET + 2] = (offset >> 24) & 0xff;
|
|
lsi->buffer[LSI_TABLE_CMD_OFFSET + 3] = (offset >> 16) & 0xff;;
|
|
lsi->buffer[LSI_TABLE_CMD_OFFSET + 4] = (offset >> 8) & 0xff;
|
|
lsi->buffer[LSI_TABLE_CMD_OFFSET + 5] = offset & 0xff;
|
|
lsi->buffer[LSI_TABLE_CMD_OFFSET + 7] = 0;
|
|
lsi->buffer[LSI_TABLE_CMD_OFFSET + 8] = 1;
|
|
lsi->table->cmd_len = __cpu_to_le32(0xa);
|
|
|
|
lsi->table->data_in_len = __cpu_to_le32(sd->bs);
|
|
|
|
lsi->table->status_len = __cpu_to_le32(0x1);
|
|
lsi->table->msg_in_len = __cpu_to_le32(0x2);
|
|
|
|
lsi->table->id = __cpu_to_le32((sd->id << 16));
|
|
lsi->table->id_addr = __cpu_to_le32(&lsi->scripts_iova[0x22]);
|
|
|
|
/* Write DSP to start DMA engine */
|
|
dsp = (uintptr_t)&lsi->scripts_iova[0x20];
|
|
lsi->mmio[LSI_DSP] = dsp & 0xff;
|
|
lsi->mmio[LSI_DSP + 1] = (dsp >> 8) & 0xff;
|
|
lsi->mmio[LSI_DSP + 2] = (dsp >> 16) & 0xff;
|
|
lsi->mmio[LSI_DSP + 3] = (dsp >> 24) & 0xff;
|
|
|
|
if (lsi_interrupt_status(lsi)) {
|
|
return 1;
|
|
}
|
|
|
|
// Reslect and data transfer
|
|
lsi->table->msg_in_len = __cpu_to_le32(0x1);
|
|
|
|
lsi->table->data_in_len = __cpu_to_le32(sd->bs);
|
|
|
|
/* Write DSP to start DMA engine */
|
|
dsp = (uintptr_t)&lsi->scripts_iova[0x2a];
|
|
lsi->mmio[LSI_DSP] = dsp & 0xff;
|
|
lsi->mmio[LSI_DSP + 1] = (dsp >> 8) & 0xff;
|
|
lsi->mmio[LSI_DSP + 2] = (dsp >> 16) & 0xff;
|
|
lsi->mmio[LSI_DSP + 3] = (dsp >> 24) & 0xff;
|
|
|
|
if (lsi_interrupt_status(lsi)) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
ob_sd_read_blocks(sd_private_t **sd)
|
|
{
|
|
cell n = POP(), cnt = n;
|
|
ucell blk = POP();
|
|
char *dest = (char*)POP();
|
|
int pos, spb, sect_offset;
|
|
lsi_private_t *lsi = (*sd)->lsi;
|
|
|
|
DPRINTF("ob_sd_read_blocks id %d %lx block=%d n=%d\n", (*sd)->id, (unsigned long)dest, blk, n );
|
|
|
|
if ((*sd)->bs == 0) {
|
|
PUSH(0);
|
|
return;
|
|
}
|
|
spb = (*sd)->bs / 512;
|
|
while (n) {
|
|
sect_offset = blk / spb;
|
|
pos = (blk - sect_offset * spb) * 512;
|
|
|
|
if (ob_sd_read_sector(lsi, *sd, sect_offset)) {
|
|
DPRINTF("ob_sd_read_blocks: error\n");
|
|
RET(0);
|
|
}
|
|
while (n && pos < spb * 512) {
|
|
memcpy(dest, (uint8_t *)&lsi->buffer[LSI_TABLE_DATA_OFFSET] + pos, 512);
|
|
pos += 512;
|
|
dest += 512;
|
|
n--;
|
|
blk++;
|
|
}
|
|
}
|
|
PUSH(cnt);
|
|
}
|
|
|
|
static void
|
|
ob_sd_block_size(__attribute__((unused))sd_private_t **sd)
|
|
{
|
|
PUSH(512);
|
|
}
|
|
|
|
static void
|
|
ob_sd_open(__attribute__((unused))sd_private_t **sd)
|
|
{
|
|
int ret = 1;
|
|
phandle_t ph;
|
|
|
|
PUSH(find_ih_method("sd-private", my_self()));
|
|
fword("execute");
|
|
*sd = cell2pointer(POP());
|
|
|
|
#ifdef CONFIG_DEBUG_LSI
|
|
{
|
|
char *args;
|
|
|
|
fword("my-args");
|
|
args = pop_fstr_copy();
|
|
DPRINTF("opening drive args %s\n", args);
|
|
free(args);
|
|
}
|
|
#endif
|
|
|
|
selfword("open-deblocker");
|
|
|
|
/* interpose disk-label */
|
|
ph = find_dev("/packages/disk-label");
|
|
fword("my-args");
|
|
PUSH_ph( ph );
|
|
fword("interpose");
|
|
|
|
RET ( -ret );
|
|
}
|
|
|
|
static void
|
|
ob_sd_close(__attribute__((unused)) sd_private_t **sd)
|
|
{
|
|
selfword("close-deblocker");
|
|
}
|
|
|
|
NODE_METHODS(ob_sd) = {
|
|
{ "open", ob_sd_open },
|
|
{ "close", ob_sd_close },
|
|
{ "read-blocks", ob_sd_read_blocks },
|
|
{ "block-size", ob_sd_block_size },
|
|
};
|
|
|
|
static void
|
|
ob_lsi_decodeunit(__attribute__((unused)) lsi_private_t **lsi_p)
|
|
{
|
|
/* ( str len -- id ) */
|
|
fword("parse-hex");
|
|
}
|
|
|
|
static void
|
|
ob_lsi_encodeunit(__attribute__((unused)) lsi_private_t **lsi_p)
|
|
{
|
|
/* ( id -- str len ) */
|
|
fword("pocket");
|
|
fword("tohexstr");
|
|
}
|
|
|
|
static void
|
|
ob_lsi_open(__attribute__((unused)) lsi_private_t **lsi_p)
|
|
{
|
|
PUSH(-1);
|
|
}
|
|
|
|
static void
|
|
ob_lsi_close(__attribute__((unused)) lsi_private_t **lsi_p)
|
|
{
|
|
return;
|
|
}
|
|
|
|
NODE_METHODS(ob_lsi) = {
|
|
{ "open" , ob_lsi_open },
|
|
{ "close" , ob_lsi_close },
|
|
{ "decode-unit", ob_lsi_decodeunit },
|
|
{ "encode-unit", ob_lsi_encodeunit },
|
|
{ "dma-alloc", ob_lsi_dma_alloc },
|
|
{ "dma-free", ob_lsi_dma_free },
|
|
{ "dma-map-in", ob_lsi_dma_map_in },
|
|
{ "dma-map-out", ob_lsi_dma_map_out },
|
|
{ "dma-sync", ob_lsi_dma_sync },
|
|
};
|
|
|
|
static void
|
|
add_alias(const char *device, const char *alias)
|
|
{
|
|
phandle_t aliases;
|
|
|
|
DPRINTF("add_alias dev \"%s\" = alias \"%s\"\n", device, alias);
|
|
|
|
aliases = find_dev("/aliases");
|
|
set_property(aliases, alias, device, strlen(device) + 1);
|
|
}
|
|
|
|
int
|
|
ob_lsi_init(const char *path, uint64_t mmio, uint64_t ram)
|
|
{
|
|
int id, diskcount = 0, cdcount = 0, *counter_ptr;
|
|
char nodebuff[256], aliasbuff[256];
|
|
phandle_t ph = get_cur_dev();
|
|
lsi_private_t *lsi;
|
|
int i;
|
|
ucell addr;
|
|
|
|
BIND_NODE_METHODS(ph, ob_lsi);
|
|
|
|
lsi = malloc(sizeof(lsi_private_t));
|
|
if (!lsi) {
|
|
DPRINTF("Can't allocate LSI private structure\n");
|
|
return -1;
|
|
}
|
|
|
|
/* Buffer for commands */
|
|
PUSH(0x1000);
|
|
feval("dma-alloc");
|
|
addr = POP();
|
|
lsi->buffer = cell2pointer(addr);
|
|
|
|
PUSH(addr);
|
|
PUSH(0x1000);
|
|
PUSH(0);
|
|
feval("dma-map-in");
|
|
addr = POP();
|
|
lsi->buffer_iova = cell2pointer(addr);
|
|
|
|
PUSH(0x40 * sizeof(uint32_t));
|
|
feval("dma-alloc");
|
|
addr = POP();
|
|
lsi->scripts = cell2pointer(addr);
|
|
|
|
PUSH(addr);
|
|
PUSH(0x40 * sizeof(uint32_t));
|
|
PUSH(0);
|
|
feval("dma-map-in");
|
|
addr = POP();
|
|
lsi->scripts_iova = cell2pointer(addr);
|
|
|
|
PUSH(sizeof(lsi_table_t));
|
|
feval("dma-alloc");
|
|
addr = POP();
|
|
lsi->table = cell2pointer(addr);
|
|
|
|
PUSH(addr);
|
|
PUSH(sizeof(lsi_table_t));
|
|
PUSH(0);
|
|
feval("dma-map-in");
|
|
addr = POP();
|
|
lsi->table_iova = cell2pointer(addr);
|
|
|
|
set_int_property(ph, "#address-cells", 1);
|
|
set_int_property(ph, "#size-cells", 0);
|
|
|
|
/* Initialise SCRIPTS */
|
|
lsi->mmio = (uint8_t *)(uint32_t)mmio;
|
|
init_scripts(lsi);
|
|
init_table(lsi);
|
|
|
|
/* Scan the SCSI bus */
|
|
for (id = 0; id < 8; id++) {
|
|
lsi->sd[id].id = id;
|
|
if (!inquiry(lsi, &lsi->sd[id])) {
|
|
DPRINTF("Unit %d not present\n", id);
|
|
continue;
|
|
}
|
|
|
|
/* Clear Unit Attention condition from reset */
|
|
for (i = 0; i < 5; i++) {
|
|
if (test_unit_ready(lsi, &lsi->sd[id])) {
|
|
break;
|
|
}
|
|
}
|
|
if (i == 5) {
|
|
DPRINTF("Unit %d present but won't become ready\n", id);
|
|
continue;
|
|
}
|
|
DPRINTF("Unit %d present\n", id);
|
|
read_capacity(lsi, &lsi->sd[id]);
|
|
|
|
#ifdef CONFIG_DEBUG_LSI
|
|
dump_drive(&lsi->sd[id]);
|
|
#endif
|
|
}
|
|
|
|
for (id = 0; id < 8; id++) {
|
|
if (!lsi->sd[id].present)
|
|
continue;
|
|
|
|
lsi->sd[id].lsi = lsi;
|
|
|
|
fword("new-device");
|
|
push_str("sd");
|
|
fword("device-name");
|
|
push_str("block");
|
|
fword("device-type");
|
|
fword("is-deblocker");
|
|
PUSH(id);
|
|
fword("encode-int");
|
|
PUSH(0);
|
|
fword("encode-int");
|
|
fword("encode+");
|
|
push_str("reg");
|
|
fword("property");
|
|
|
|
PUSH(pointer2cell(&lsi->sd[id]));
|
|
feval("value sd-private");
|
|
|
|
BIND_NODE_METHODS(get_cur_dev(), ob_sd);
|
|
fword("finish-device");
|
|
|
|
snprintf(nodebuff, sizeof(nodebuff), "%s/sd@%d",
|
|
get_path_from_ph(ph), id);
|
|
|
|
if (lsi->sd[id].media == TYPE_ROM) {
|
|
counter_ptr = &cdcount;
|
|
} else {
|
|
counter_ptr = &diskcount;
|
|
}
|
|
if (*counter_ptr == 0) {
|
|
add_alias(nodebuff, lsi->sd[id].media_str[0]);
|
|
add_alias(nodebuff, lsi->sd[id].media_str[1]);
|
|
}
|
|
snprintf(aliasbuff, sizeof(aliasbuff), "%s%d",
|
|
lsi->sd[id].media_str[0], *counter_ptr);
|
|
add_alias(nodebuff, aliasbuff);
|
|
snprintf(aliasbuff, sizeof(aliasbuff), "%s%d",
|
|
lsi->sd[id].media_str[1], *counter_ptr);
|
|
add_alias(nodebuff, aliasbuff);
|
|
}
|
|
|
|
return 0;
|
|
}
|