mirror of
https://github.com/akuker/RASCSI.git
synced 2025-01-10 17:30:47 +00:00
157172b566
* Use ioexception for file open error handling * Use standard string comparison functions * Error message update * Improved CD-ROM handling error messages
185 lines
5.6 KiB
C++
185 lines
5.6 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
|
||
void Open(const Filepath& path, BOOL attn = TRUE);
|
||
// Capture packets
|
||
|
||
// commands
|
||
int Inquiry(const DWORD *cdb, BYTE *buffer, DWORD major, DWORD minor);
|
||
// INQUIRY command
|
||
BOOL TestUnitReady(const DWORD *cdb);
|
||
// TEST UNIT READY command
|
||
int Read(const DWORD *cdb, BYTE *buf, DWORD block) override;
|
||
// READ command
|
||
BOOL Write(const DWORD *cdb, const BYTE *buf, DWORD block) override;
|
||
// WRITE command
|
||
int WriteCheck(DWORD block) override;
|
||
// WRITE check
|
||
|
||
int RetrieveStats(const DWORD *cdb, BYTE *buffer);
|
||
// Retrieve DaynaPort statistics
|
||
BOOL EnableInterface(const DWORD *cdb);
|
||
// Enable/Disable Interface command
|
||
|
||
void SetMacAddr(const DWORD *cdb, BYTE *buffer);
|
||
// Set MAC address
|
||
void SetMode(const DWORD *cdb, BYTE *buffer);
|
||
// Set the mode: whether broadcast traffic is enabled or not
|
||
int 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;
|
||
|
||
// When we're reading the Linux tap device, most of the messages will not be for us, so we
|
||
// need to filter through those. However, we don't want to keep re-reading the packets
|
||
// indefinitely. So, we'll pick a large-ish number that will cause the emulated DaynaPort
|
||
// to respond with "no data" after MAX_READ_RETRIES tries.
|
||
static const int MAX_READ_RETRIES = 50;
|
||
|
||
// The READ response has a header which consists of:
|
||
// 2 bytes - payload size
|
||
// 4 bytes - status flags
|
||
static const DWORD DAYNAPORT_READ_HEADER_SZ = 2 + 4;
|
||
|
||
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] ... Processor Device
|
||
// buf[1] ... Not 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_daynaport_inquiry_response[44] = {
|
||
0x03, 0x00, 0x01, 0x00, // 4 bytes
|
||
0x1E, 0x00, 0x00, 0x00, // 4 bytes
|
||
// Vendor ID (8 Bytes)
|
||
'D','a','y','n','a',' ',' ',' ',
|
||
// 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
|
||
};
|
||
|
||
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];
|
||
};
|