mirror of
https://github.com/akuker/RASCSI.git
synced 2024-06-30 03:29:32 +00:00
200 lines
6.8 KiB
C
200 lines
6.8 KiB
C
|
//---------------------------------------------------------------------------
|
|||
|
//
|
|||
|
// SCSI Target Emulator RaSCSI (*^..^*)
|
|||
|
// for Raspberry Pi
|
|||
|
//
|
|||
|
// Copyright (C) 2020 akuker
|
|||
|
// Copyright (C) 2014-2020 GIMONS
|
|||
|
// Copyright (C) 2001-2006 PI.(ytanaka@ipc-tokai.or.jp)
|
|||
|
//
|
|||
|
// Licensed under the BSD 3-Clause License.
|
|||
|
// See LICENSE file in the project root folder.
|
|||
|
//
|
|||
|
// [ Emulation of the DaynaPort SCSI Link Ethernet interface ]
|
|||
|
//
|
|||
|
// This design is derived from the SLINKCMD.TXT file, as well as David Kuder's
|
|||
|
// Tiny SCSI Emulator
|
|||
|
// - SLINKCMD: http://www.bitsavers.org/pdf/apple/scsi/dayna/daynaPORT/SLINKCMD.TXT
|
|||
|
// - Tiny SCSI : https://hackaday.io/project/18974-tiny-scsi-emulator
|
|||
|
//
|
|||
|
// Special thanks to @PotatoFi for loaning me his Farallon EtherMac for
|
|||
|
// this development. (Farallon's EtherMac is a re-branded DaynaPort
|
|||
|
// SCSI/Link-T).
|
|||
|
//
|
|||
|
// This does NOT include the file system functionality that is present
|
|||
|
// in the Sharp X68000 host bridge.
|
|||
|
//
|
|||
|
// Note: This requires the DaynaPort SCSI Link driver.
|
|||
|
//---------------------------------------------------------------------------
|
|||
|
#pragma once
|
|||
|
|
|||
|
#include "xm6.h"
|
|||
|
#include "os.h"
|
|||
|
#include "disk.h"
|
|||
|
#include "ctapdriver.h"
|
|||
|
|
|||
|
//===========================================================================
|
|||
|
//
|
|||
|
// DaynaPort SCSI Link
|
|||
|
//
|
|||
|
//===========================================================================
|
|||
|
class SCSIDaynaPort: public Disk
|
|||
|
{
|
|||
|
public:
|
|||
|
// Basic Functions
|
|||
|
SCSIDaynaPort();
|
|||
|
// Constructor
|
|||
|
virtual ~SCSIDaynaPort();
|
|||
|
// Destructor
|
|||
|
|
|||
|
// commands
|
|||
|
int FASTCALL Inquiry(const DWORD *cdb, BYTE *buffer, DWORD major, DWORD minor);
|
|||
|
// INQUIRY command
|
|||
|
BOOL FASTCALL TestUnitReady(const DWORD *cdb);
|
|||
|
// TEST UNIT READY command
|
|||
|
int FASTCALL Read(const DWORD *cdb, BYTE *buf, DWORD block) override;
|
|||
|
// READ command
|
|||
|
BOOL FASTCALL Write(const DWORD *cdb, const BYTE *buf, DWORD block) override;
|
|||
|
// WRITE command
|
|||
|
int FASTCALL WriteCheck(DWORD block) override;
|
|||
|
// WRITE check
|
|||
|
|
|||
|
int FASTCALL RetrieveStats(const DWORD *cdb, BYTE *buffer);
|
|||
|
// Retrieve DaynaPort statistics
|
|||
|
BOOL FASTCALL EnableInterface(const DWORD *cdb);
|
|||
|
// Enable/Disable Interface command
|
|||
|
|
|||
|
void FASTCALL SetMacAddr(const DWORD *cdb, BYTE *buffer);
|
|||
|
// Set MAC address
|
|||
|
void FASTCALL SetMode(const DWORD *cdb, BYTE *buffer);
|
|||
|
// Set the mode: whether broadcast traffic is enabled or not
|
|||
|
int FASTCALL RequestSense(const DWORD *cdb, BYTE *buf) override;
|
|||
|
|
|||
|
static const BYTE CMD_SCSILINK_STATS = 0x09;
|
|||
|
static const BYTE CMD_SCSILINK_ENABLE = 0x0E;
|
|||
|
static const BYTE CMD_SCSILINK_SET = 0x0C;
|
|||
|
static const BYTE CMD_SCSILINK_SETMODE = 0x80;
|
|||
|
static const BYTE CMD_SCSILINK_SETMAC = 0x40;
|
|||
|
|
|||
|
private:
|
|||
|
typedef struct __attribute__((packed)) {
|
|||
|
BYTE operation_code;
|
|||
|
BYTE reserved;
|
|||
|
WORD pad;
|
|||
|
BYTE transfer_length;
|
|||
|
BYTE control;
|
|||
|
} scsi_cmd_config_multicast_t;
|
|||
|
|
|||
|
typedef struct __attribute__((packed)) {
|
|||
|
BYTE operation_code;
|
|||
|
BYTE reserved;
|
|||
|
BYTE pad2;
|
|||
|
BYTE pad3;
|
|||
|
BYTE pad4;
|
|||
|
BYTE control;
|
|||
|
} scsi_cmd_enable_disable_iface_t;
|
|||
|
|
|||
|
|
|||
|
typedef struct __attribute__((packed)) {
|
|||
|
BYTE operation_code;
|
|||
|
BYTE misc_cdb_information;
|
|||
|
BYTE logical_block_address;
|
|||
|
WORD length;
|
|||
|
BYTE format;
|
|||
|
} scsi_cmd_daynaport_write_t;
|
|||
|
|
|||
|
|
|||
|
enum read_data_flags_t : DWORD {
|
|||
|
e_no_more_data = 0x00000000,
|
|||
|
e_more_data_available = 0x00000001,
|
|||
|
e_dropped_packets = 0xFFFFFFFF,
|
|||
|
};
|
|||
|
|
|||
|
typedef struct __attribute__((packed)) {
|
|||
|
WORD length;
|
|||
|
read_data_flags_t flags;
|
|||
|
BYTE pad;
|
|||
|
BYTE data[ETH_FRAME_LEN + sizeof(DWORD)]; // Frame length + 4 byte CRC
|
|||
|
} scsi_resp_read_t;
|
|||
|
|
|||
|
typedef struct __attribute__((packed)) {
|
|||
|
BYTE mac_address[6];
|
|||
|
DWORD frame_alignment_errors;
|
|||
|
DWORD crc_errors;
|
|||
|
DWORD frames_lost;
|
|||
|
} scsi_resp_link_stats_t;
|
|||
|
|
|||
|
static const char* m_vendor_name;
|
|||
|
static const char* m_device_name;
|
|||
|
static const char* m_revision;
|
|||
|
static const char* m_firmware_version;
|
|||
|
|
|||
|
scsi_resp_link_stats_t m_scsi_link_stats = {
|
|||
|
.mac_address = { 0x00, 0x80, 0x19, 0x10, 0x98, 0xE3 },//MAC address of @PotatoFi's DayanPort
|
|||
|
.frame_alignment_errors = 0,
|
|||
|
.crc_errors = 0,
|
|||
|
.frames_lost = 0,
|
|||
|
};
|
|||
|
|
|||
|
const BYTE m_daynacom_mac_prefix[3] = {0x00,0x80,0x19};
|
|||
|
|
|||
|
|
|||
|
// Basic data
|
|||
|
// buf[0] ... CD-ROM Device
|
|||
|
// buf[1] ... Removable
|
|||
|
// buf[2] ... SCSI-2 compliant command system
|
|||
|
// buf[3] ... SCSI-2 compliant Inquiry response
|
|||
|
// buf[4] ... Inquiry additional data
|
|||
|
//http://www.bitsavers.org/pdf/apple/scsi/dayna/daynaPORT/pocket_scsiLINK/pocketscsilink_inq.png
|
|||
|
const uint8_t m_target_ethernet_inquiry_response[255] = {
|
|||
|
0x03, 0x00, 0x01, 0x00, // 4 bytes
|
|||
|
0x1E, 0x00, 0x00, 0x00, // 4 bytes
|
|||
|
// Vendor ID (8 Bytes)
|
|||
|
'D','a','y','n','a',' ',' ',' ',
|
|||
|
//'D','A','Y','N','A','T','R','N',
|
|||
|
// Product ID (16 Bytes)
|
|||
|
'S','C','S','I','/','L','i','n',
|
|||
|
'k',' ',' ',' ',' ',' ',' ',' ',
|
|||
|
// Revision Number (4 Bytes)
|
|||
|
'1','.','4','a',
|
|||
|
// Firmware Version (8 Bytes)
|
|||
|
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
|
|||
|
// Data
|
|||
|
0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, //16 bytes
|
|||
|
0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, //16 bytes
|
|||
|
0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, //16 bytes
|
|||
|
0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, //16 bytes
|
|||
|
0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, //16 bytes
|
|||
|
0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, //16 bytes
|
|||
|
0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x80,0x80,0xBA, //16 bytes
|
|||
|
0x00,0x00,0xC0,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, //16 bytes
|
|||
|
0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, //16 bytes
|
|||
|
0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, //16 bytes
|
|||
|
0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x81, 0x00,0x00,0x00,0x00, //16 bytes
|
|||
|
0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, //16 bytes
|
|||
|
0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, //16 bytes
|
|||
|
0x00,0x00,0x00 //3 bytes
|
|||
|
};
|
|||
|
|
|||
|
|
|||
|
#if defined(RASCSI) && !defined(BAREMETAL)
|
|||
|
|
|||
|
CTapDriver *m_tap;
|
|||
|
// TAP driver
|
|||
|
BOOL m_bTapEnable;
|
|||
|
// TAP valid flag
|
|||
|
BYTE m_mac_addr[6];
|
|||
|
// MAC Address
|
|||
|
static const BYTE m_bcast_addr[6];
|
|||
|
static const BYTE m_apple_talk_addr[6];
|
|||
|
|
|||
|
// The READ response has a header which consists of:
|
|||
|
// 2 bytes - payload size
|
|||
|
// 4 bytes - status flags
|
|||
|
// 1 byte - magic pad bit, that I don't know why it works.....
|
|||
|
const DWORD m_read_header_size = 2 + 4 + 1;
|
|||
|
|
|||
|
#endif // RASCSI && !BAREMETAL
|
|||
|
|
|||
|
};
|