1
0
mirror of https://github.com/TomHarte/CLK.git synced 2024-06-17 06:29:28 +00:00

Enhance and better-document I2C states.

This commit is contained in:
Thomas Harte 2024-03-26 21:52:29 -04:00
parent a3931674dc
commit 8b04d0e3ef
3 changed files with 58 additions and 14 deletions

View File

@ -37,6 +37,7 @@ bool Bus::clock() {
} }
void Bus::set_clock_data(bool clock_pulled, bool data_pulled) { void Bus::set_clock_data(bool clock_pulled, bool data_pulled) {
// Proceed only if changes are evidenced.
if(clock_pulled == clock_ && data_pulled == data_) { if(clock_pulled == clock_ && data_pulled == data_) {
return; return;
} }
@ -46,6 +47,8 @@ void Bus::set_clock_data(bool clock_pulled, bool data_pulled) {
clock_ = clock_pulled; clock_ = clock_pulled;
data_ = data_pulled; data_ = data_pulled;
// If currently serialising from a peripheral then shift onwards on
// every clock trailing edge.
if(peripheral_bits_) { if(peripheral_bits_) {
// Trailing edge of clock => bit has been consumed. // Trailing edge of clock => bit has been consumed.
if(!prior_clock && clock_) { if(!prior_clock && clock_) {
@ -60,8 +63,9 @@ void Bus::set_clock_data(bool clock_pulled, bool data_pulled) {
return; return;
} }
// Not currently serialising implies listening.
if(!clock_ && prior_data != data_) { if(!clock_ && prior_data != data_) {
// Data transition outside of a clock cycle => start or stop. // A data transition outside of a clock cycle implies a start or stop.
in_bit_ = false; in_bit_ = false;
if(data_) { if(data_) {
logger.info().append("S"); logger.info().append("S");
@ -76,7 +80,8 @@ void Bus::set_clock_data(bool clock_pulled, bool data_pulled) {
// Rising edge: clock period begins. // Rising edge: clock period begins.
in_bit_ = true; in_bit_ = true;
} else if(in_bit_) { } else if(in_bit_) {
// Falling edge: clock period ends (assuming it began; otherwise this is prep, post-start bit). // Falling edge: clock period ends (assuming it began; otherwise this is a preparatory
// clock transition only, immediately after a start bit).
in_bit_ = false; in_bit_ = false;
if(data_) { if(data_) {
@ -118,6 +123,11 @@ void Bus::signal(Event event) {
} }
}; };
const auto stop = [&]() {
set_state(State::AwaitingAddress);
active_peripheral_ = nullptr;
};
// Allow start and stop conditions at any time. // Allow start and stop conditions at any time.
if(event == Event::Start) { if(event == Event::Start) {
set_state(State::CollectingAddress); set_state(State::CollectingAddress);
@ -126,17 +136,21 @@ void Bus::signal(Event event) {
} }
if(event == Event::Stop) { if(event == Event::Stop) {
set_state(State::AwaitingAddress);
if(active_peripheral_) { if(active_peripheral_) {
active_peripheral_->stop(); active_peripheral_->stop();
} }
active_peripheral_ = nullptr; stop();
return; return;
} }
switch(state_) { switch(state_) {
// While waiting for an address, don't respond to anything other than a
// start bit, which is actually dealt with above.
case State::AwaitingAddress: break; case State::AwaitingAddress: break;
// To collect an address: shift in eight bits, and if there's a device
// at that address then acknowledge the address and segue into a read
// or write loop.
case State::CollectingAddress: case State::CollectingAddress:
capture_bit(); capture_bit();
if(input_count_ == 8) { if(input_count_ == 8) {
@ -147,7 +161,7 @@ void Bus::signal(Event event) {
if(input_&1) { if(input_&1) {
acknowledge(); acknowledge();
set_state(State::CompletingWriteAcknowledge); set_state(State::CompletingReadAcknowledge);
} else { } else {
acknowledge(); acknowledge();
set_state(State::ReceivingByte); set_state(State::ReceivingByte);
@ -158,32 +172,46 @@ void Bus::signal(Event event) {
} }
break; break;
// Receiving byte: wait until a scheduled acknowledgment has
// happened, then collect eight bits, then see whether the
// active peripheral will accept them. If so, acknowledge and repeat.
// Otherwise fall silent.
case State::ReceivingByte: case State::ReceivingByte:
if(event == Event::FinishedOutput) { if(event == Event::FinishedOutput) {
return; return;
} }
capture_bit(); capture_bit();
if(input_count_ == 8) { if(input_count_ == 8) {
active_peripheral_->write(static_cast<uint8_t>(input_)); if(active_peripheral_->write(static_cast<uint8_t>(input_))) {
acknowledge(); acknowledge();
set_state(State::ReceivingByte); set_state(State::ReceivingByte);
} else {
stop();
}
} }
break; break;
case State::CompletingWriteAcknowledge: // The initial state immediately after a peripheral has been started
// in read mode and the address-select acknowledgement is still
// being serialised.
//
// Once that is completed, enqueues the first byte from the peripheral.
case State::CompletingReadAcknowledge:
if(event != Event::FinishedOutput) { if(event != Event::FinishedOutput) {
break; break;
} }
enqueue(active_peripheral_->read()); enqueue(active_peripheral_->read());
break; break;
// Repeating state during reading; waits until the previous byte has
// been fully serialised, and if the host acknowledged it then posts
// the next. If the host didn't acknowledge, stops the connection.
case State::AwaitingByteAcknowledge: case State::AwaitingByteAcknowledge:
if(event == Event::FinishedOutput) { if(event == Event::FinishedOutput) {
break; break;
} }
if(event != Event::Zero) { if(event != Event::Zero) {
logger.info().append("Peripheral byte not acknowledged"); stop();
state_ = State::AwaitingAddress;
break; break;
} }

View File

@ -17,10 +17,25 @@ namespace I2C {
/// Provides the virtual interface for an I2C peripheral; attaching this to a bus /// Provides the virtual interface for an I2C peripheral; attaching this to a bus
/// provides automatic protocol handling. /// provides automatic protocol handling.
struct Peripheral { struct Peripheral {
/// Indicates that the host signalled the start condition and addressed this
/// peripheral, along with whether it indicated a read or write.
virtual void start([[maybe_unused]] bool is_read) {} virtual void start([[maybe_unused]] bool is_read) {}
/// Indicates that the host signalled a stop.
virtual void stop() {} virtual void stop() {}
/// Requests the next byte to serialise onto the I2C bus after this peripheral has
/// been started in read mode.
///
/// @returns A byte to serialise or std::nullopt if the peripheral declines to
/// continue to communicate.
virtual std::optional<uint8_t> read() { return std::nullopt; } virtual std::optional<uint8_t> read() { return std::nullopt; }
virtual void write(uint8_t) {}
/// Provides a byte received from the bus after this peripheral has been started
/// in write mode.
///
/// @returns @c true if the write should be acknowledged; @c false otherwise.
virtual bool write(uint8_t) { return false; }
}; };
class Bus { class Bus {
@ -57,7 +72,7 @@ private:
AwaitingAddress, AwaitingAddress,
CollectingAddress, CollectingAddress,
CompletingWriteAcknowledge, CompletingReadAcknowledge,
AwaitingByteAcknowledge, AwaitingByteAcknowledge,
ReceivingByte, ReceivingByte,

View File

@ -46,7 +46,7 @@ struct CMOSRAM: public I2C::Peripheral {
return result; return result;
} }
void write(uint8_t value) override { bool write(uint8_t value) override {
if(expecting_address_) { if(expecting_address_) {
address_ = value; address_ = value;
expecting_address_ = false; expecting_address_ = false;
@ -54,6 +54,7 @@ struct CMOSRAM: public I2C::Peripheral {
++address_; ++address_;
// TODO: write. // TODO: write.
} }
return true;
} }
private: private: