2022-10-08 17:26:04 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// SCSI Target Emulator RaSCSI Reloaded
|
|
|
|
// for Raspberry Pi
|
|
|
|
//
|
|
|
|
// Copyright (C) 2022 Uwe Seimet
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
#include "mocks.h"
|
|
|
|
#include "controllers/controller_manager.h"
|
|
|
|
#include "rascsi_version.h"
|
|
|
|
#include "test_shared.h"
|
2022-10-23 19:51:39 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <vector>
|
2022-10-08 17:26:04 +00:00
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
2022-10-23 19:51:39 +00:00
|
|
|
shared_ptr<PrimaryDevice> CreateDevice(PbDeviceType type, MockAbstractController& controller, const string& extension)
|
2022-10-08 17:26:04 +00:00
|
|
|
{
|
|
|
|
DeviceFactory device_factory;
|
|
|
|
auto bus = make_shared<MockBus>();
|
|
|
|
auto controller_manager = make_shared<ControllerManager>(*bus);
|
|
|
|
|
2022-10-23 19:51:39 +00:00
|
|
|
auto device = device_factory.CreateDevice(*controller_manager, type, 0, extension);
|
2022-10-08 17:26:04 +00:00
|
|
|
|
|
|
|
controller.AddDevice(device);
|
|
|
|
|
|
|
|
return device;
|
|
|
|
}
|
|
|
|
|
2022-10-23 19:51:39 +00:00
|
|
|
void TestInquiry(PbDeviceType type, device_type t, scsi_level l, const string& ident, int additional_length,
|
|
|
|
bool removable, const string& extension)
|
2022-10-08 17:26:04 +00:00
|
|
|
{
|
|
|
|
NiceMock<MockAbstractController> controller(0);
|
2022-10-23 19:51:39 +00:00
|
|
|
auto device = CreateDevice(type, controller, extension);
|
2022-10-08 17:26:04 +00:00
|
|
|
|
2022-10-23 19:51:39 +00:00
|
|
|
vector<int>& cmd = controller.GetCmd();
|
2022-10-08 17:26:04 +00:00
|
|
|
|
|
|
|
// ALLOCATION LENGTH
|
|
|
|
cmd[4] = 255;
|
2022-10-23 19:51:39 +00:00
|
|
|
EXPECT_CALL(controller, DataIn());
|
2022-10-08 17:26:04 +00:00
|
|
|
EXPECT_TRUE(device->Dispatch(scsi_command::eCmdInquiry));
|
|
|
|
const vector<BYTE>& buffer = controller.GetBuffer();
|
|
|
|
EXPECT_EQ((int)t, buffer[0]);
|
|
|
|
EXPECT_EQ(removable ? 0x80: 0x00, buffer[1]);
|
|
|
|
EXPECT_EQ((int)l, buffer[2]);
|
2022-10-23 19:51:39 +00:00
|
|
|
EXPECT_EQ((int)l > (int)scsi_level::SCSI_2 ? (int)scsi_level::SCSI_2 : (int)l, buffer[3]);
|
2022-10-08 17:26:04 +00:00
|
|
|
EXPECT_EQ(additional_length, buffer[4]);
|
|
|
|
string product_data;
|
|
|
|
if (ident.size() == 24) {
|
|
|
|
ostringstream s;
|
|
|
|
s << ident << setw(2) << setfill('0') << rascsi_major_version << rascsi_minor_version;
|
|
|
|
product_data = s.str();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
product_data = ident;
|
|
|
|
}
|
|
|
|
EXPECT_TRUE(!memcmp(product_data.c_str(), &buffer[8], 28));
|
|
|
|
}
|
2022-10-23 19:51:39 +00:00
|
|
|
|
|
|
|
void TestDispatch(PbDeviceType type)
|
|
|
|
{
|
|
|
|
NiceMock<MockAbstractController> controller(0);
|
|
|
|
auto device = CreateDevice(type, controller);
|
|
|
|
|
|
|
|
EXPECT_FALSE(device->Dispatch(scsi_command::eCmdIcd)) << "Command is not supported by this class";
|
|
|
|
}
|
|
|
|
|
|
|
|
int OpenTempFile(string& file)
|
|
|
|
{
|
|
|
|
char filename[] = "/tmp/rascsi_test-XXXXXX"; //NOSONAR mkstemp() requires a modifiable string
|
|
|
|
|
|
|
|
const int fd = mkstemp(filename);
|
|
|
|
EXPECT_NE(-1, fd) << "Couldn't create temporary file '" << filename << "'";
|
|
|
|
|
|
|
|
file = filename;
|
|
|
|
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
string CreateTempFile(int size)
|
|
|
|
{
|
|
|
|
char filename[] = "/tmp/rascsi_test-XXXXXX"; //NOSONAR mkstemp() requires a modifiable string
|
|
|
|
vector<char> data(size);
|
|
|
|
|
|
|
|
const int fd = mkstemp(filename);
|
|
|
|
|
|
|
|
const size_t count = write(fd, data.data(), data.size());
|
|
|
|
close(fd);
|
|
|
|
EXPECT_EQ(count, data.size()) << "Couldn't create temporary file '" << string(filename) << "'";
|
|
|
|
|
|
|
|
return filename;
|
|
|
|
}
|
|
|
|
|
|
|
|
int GetInt16(const vector<byte>& buf, int offset)
|
|
|
|
{
|
|
|
|
assert(buf.size() > (size_t)offset + 1);
|
|
|
|
|
|
|
|
return ((int)buf[offset] << 8) | (int)buf[offset + 1];
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t GetInt32(const vector<byte>& buf, int offset)
|
|
|
|
{
|
|
|
|
assert(buf.size() > (size_t)offset + 3);
|
|
|
|
|
|
|
|
return ((uint32_t)buf[offset] << 24) | ((uint32_t)buf[offset + 1] << 16) |
|
|
|
|
((uint32_t)buf[offset + 2] << 8) | (uint32_t)buf[offset + 3];
|
|
|
|
}
|
|
|
|
|