2019-08-13 23:09:11 -04:00
|
|
|
//
|
|
|
|
// SCSI.hpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 12/08/2019.
|
|
|
|
// Copyright © 2019 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef SCSI_hpp
|
|
|
|
#define SCSI_hpp
|
|
|
|
|
|
|
|
#include <limits>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
namespace SCSI {
|
|
|
|
|
|
|
|
typedef int BusState;
|
|
|
|
|
2019-08-15 23:14:40 -04:00
|
|
|
static const BusState DefaultBusState = 0;
|
2019-08-13 23:09:11 -04:00
|
|
|
|
|
|
|
/*!
|
|
|
|
SCSI bus state is encoded entirely within an int.
|
|
|
|
Bits correlate mostly but not exactly to the real SCSI bus.
|
|
|
|
|
|
|
|
TODO: validate levels below. The bus uses open collector logic,
|
|
|
|
so active low needs to be respected.
|
|
|
|
*/
|
|
|
|
enum Line: BusState {
|
|
|
|
/// Provides the value currently on the data lines.
|
|
|
|
Data = 0xff,
|
|
|
|
/// Parity of the data lines.
|
|
|
|
Parity = 1 << 8,
|
|
|
|
/// Set if the SEL line is currently selecting a target.
|
|
|
|
/// Reset if it is selecting an initiator.
|
|
|
|
SelectTarget = 1 << 9,
|
2019-08-15 23:14:40 -04:00
|
|
|
/// Set to indicate an attention condition. Reset otherwise.
|
2019-08-13 23:09:11 -04:00
|
|
|
Attention = 1 << 10,
|
|
|
|
/// Set if control is on the bus. Reset if data is on the bus.
|
|
|
|
Control = 1 << 11,
|
2019-08-15 23:14:40 -04:00
|
|
|
/// Set if the bus is busy. Reset otherwise.
|
2019-08-13 23:09:11 -04:00
|
|
|
Busy = 1 << 12,
|
2019-08-15 23:14:40 -04:00
|
|
|
/// Set if acknowledging a data transfer request. Reset otherwise.
|
2019-08-13 23:09:11 -04:00
|
|
|
Acknowledge = 1 << 13,
|
2019-08-15 23:14:40 -04:00
|
|
|
/// Set if a bus reset is being requested. Reset otherwise.
|
2019-08-13 23:09:11 -04:00
|
|
|
Reset = 1 << 14,
|
2019-08-15 23:14:40 -04:00
|
|
|
/// Set if data is currently an input to the initiator. Reset if it is an output.
|
2019-08-13 23:09:11 -04:00
|
|
|
Input = 1 << 15,
|
|
|
|
/// Set during the message phase. Reset otherwise.
|
2019-08-15 23:14:40 -04:00
|
|
|
Message = 1 << 16,
|
|
|
|
/// Set if requesting a data transfer. Reset otherwise.
|
|
|
|
Request = 1 << 17,
|
2019-08-13 23:09:11 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Bus {
|
|
|
|
public:
|
|
|
|
/*!
|
|
|
|
Adds a device to the bus, returning the index it should use
|
|
|
|
to refer to itself in subsequent calls to set_device_output.
|
|
|
|
*/
|
|
|
|
size_t add_device();
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Sets the current output for @c device.
|
|
|
|
*/
|
|
|
|
void set_device_output(size_t device, BusState output);
|
|
|
|
|
|
|
|
/*!
|
|
|
|
@returns the current state of the bus.
|
|
|
|
*/
|
|
|
|
BusState get_state();
|
|
|
|
|
2019-08-17 23:43:42 -04:00
|
|
|
struct Observer {
|
|
|
|
virtual void scsi_bus_did_change(Bus *, BusState new_state) = 0;
|
|
|
|
};
|
|
|
|
/*!
|
|
|
|
Adds an observer.
|
|
|
|
*/
|
|
|
|
void add_observer(Observer *);
|
|
|
|
|
2019-08-13 23:09:11 -04:00
|
|
|
private:
|
|
|
|
std::vector<BusState> device_states_;
|
|
|
|
BusState state_ = DefaultBusState;
|
2019-08-17 23:43:42 -04:00
|
|
|
std::vector<Observer *> observers_;
|
2019-08-13 23:09:11 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* SCSI_hpp */
|