2017-07-25 20:20:55 -04:00
|
|
|
//
|
|
|
|
// ClockReceiver.hpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 22/07/2017.
|
2018-05-13 15:19:52 -04:00
|
|
|
// Copyright 2017 Thomas Harte. All rights reserved.
|
2017-07-25 20:20:55 -04:00
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef ClockReceiver_hpp
|
|
|
|
#define ClockReceiver_hpp
|
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
#include "ForceInline.hpp"
|
2019-10-29 22:36:29 -04:00
|
|
|
#include <cstdint>
|
2019-07-07 14:12:20 -04:00
|
|
|
|
2017-07-27 21:38:50 -04:00
|
|
|
/*
|
|
|
|
Informal pattern for all classes that run from a clock cycle:
|
|
|
|
|
|
|
|
Each will implement either or both of run_for(Cycles) and run_for(HalfCycles), as
|
|
|
|
is appropriate.
|
|
|
|
|
|
|
|
Callers that are accumulating HalfCycles but want to talk to receivers that implement
|
|
|
|
only run_for(Cycles) can use HalfCycle.flush_cycles if they have appropriate storage, or
|
|
|
|
can wrap the receiver in HalfClockReceiver in order automatically to bind half-cycle
|
|
|
|
storage to it.
|
|
|
|
|
|
|
|
Alignment rule:
|
|
|
|
|
|
|
|
run_for(Cycles) may be called only after an even number of half cycles. E.g. the following
|
|
|
|
sequence will have undefined results:
|
|
|
|
|
|
|
|
run_for(HalfCycles(1))
|
|
|
|
run_for(Cycles(1))
|
|
|
|
|
|
|
|
An easy way to ensure this as a caller is to pick only one of run_for(Cycles) and
|
|
|
|
run_for(HalfCycles) to use.
|
|
|
|
|
|
|
|
Reasoning:
|
|
|
|
|
|
|
|
Users of this template may with to implement run_for(Cycles) and run_for(HalfCycles)
|
|
|
|
where there is a need to implement at half-cycle precision but a faster execution
|
|
|
|
path can be offered for full-cycle precision. Those users are permitted to assume
|
|
|
|
phase in run_for(Cycles) and should do so to be compatible with callers that use
|
|
|
|
only run_for(Cycles).
|
|
|
|
|
|
|
|
Corollary:
|
|
|
|
|
|
|
|
Starting from nothing, the first run_for(HalfCycles(1)) will do the **first** half
|
|
|
|
of a full cycle. The second will do the second half. Etc.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2017-07-25 20:20:55 -04:00
|
|
|
/*!
|
|
|
|
Provides a class that wraps a plain int, providing most of the basic arithmetic and
|
|
|
|
Boolean operators, but forcing callers and receivers to be explicit as to usage.
|
|
|
|
*/
|
|
|
|
template <class T> class WrappedInt {
|
|
|
|
public:
|
2019-10-29 22:36:29 -04:00
|
|
|
using IntType = int64_t;
|
|
|
|
|
|
|
|
forceinline constexpr WrappedInt(IntType l) noexcept : length_(l) {}
|
2019-07-26 22:18:40 -04:00
|
|
|
forceinline constexpr WrappedInt() noexcept : length_(0) {}
|
2017-07-25 20:20:55 -04:00
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline T &operator =(const T &rhs) {
|
2017-07-25 20:20:55 -04:00
|
|
|
length_ = rhs.length_;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline T &operator +=(const T &rhs) {
|
2017-07-25 20:20:55 -04:00
|
|
|
length_ += rhs.length_;
|
|
|
|
return *static_cast<T *>(this);
|
|
|
|
}
|
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline T &operator -=(const T &rhs) {
|
2017-07-25 20:20:55 -04:00
|
|
|
length_ -= rhs.length_;
|
|
|
|
return *static_cast<T *>(this);
|
|
|
|
}
|
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline T &operator ++() {
|
2017-07-25 20:20:55 -04:00
|
|
|
++ length_;
|
|
|
|
return *static_cast<T *>(this);
|
|
|
|
}
|
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline T &operator ++(int) {
|
2017-07-25 20:20:55 -04:00
|
|
|
length_ ++;
|
|
|
|
return *static_cast<T *>(this);
|
|
|
|
}
|
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline T &operator --() {
|
2017-07-25 20:20:55 -04:00
|
|
|
-- length_;
|
|
|
|
return *static_cast<T *>(this);
|
|
|
|
}
|
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline T &operator --(int) {
|
2017-07-25 20:20:55 -04:00
|
|
|
length_ --;
|
|
|
|
return *static_cast<T *>(this);
|
|
|
|
}
|
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline T &operator *=(const T &rhs) {
|
2019-05-04 22:27:58 -04:00
|
|
|
length_ *= rhs.length_;
|
|
|
|
return *static_cast<T *>(this);
|
|
|
|
}
|
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline T &operator /=(const T &rhs) {
|
2019-05-04 22:27:58 -04:00
|
|
|
length_ /= rhs.length_;
|
|
|
|
return *static_cast<T *>(this);
|
|
|
|
}
|
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline T &operator %=(const T &rhs) {
|
2017-07-25 20:20:55 -04:00
|
|
|
length_ %= rhs.length_;
|
|
|
|
return *static_cast<T *>(this);
|
|
|
|
}
|
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline T &operator &=(const T &rhs) {
|
2017-07-31 07:29:50 -04:00
|
|
|
length_ &= rhs.length_;
|
|
|
|
return *static_cast<T *>(this);
|
|
|
|
}
|
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline constexpr T operator +(const T &rhs) const { return T(length_ + rhs.length_); }
|
|
|
|
forceinline constexpr T operator -(const T &rhs) const { return T(length_ - rhs.length_); }
|
2017-07-25 20:20:55 -04:00
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline constexpr T operator *(const T &rhs) const { return T(length_ * rhs.length_); }
|
|
|
|
forceinline constexpr T operator /(const T &rhs) const { return T(length_ / rhs.length_); }
|
2019-05-04 22:27:58 -04:00
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline constexpr T operator %(const T &rhs) const { return T(length_ % rhs.length_); }
|
|
|
|
forceinline constexpr T operator &(const T &rhs) const { return T(length_ & rhs.length_); }
|
2017-07-31 07:29:50 -04:00
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline constexpr T operator -() const { return T(- length_); }
|
2017-07-27 21:53:45 -04:00
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline constexpr bool operator <(const T &rhs) const { return length_ < rhs.length_; }
|
|
|
|
forceinline constexpr bool operator >(const T &rhs) const { return length_ > rhs.length_; }
|
|
|
|
forceinline constexpr bool operator <=(const T &rhs) const { return length_ <= rhs.length_; }
|
|
|
|
forceinline constexpr bool operator >=(const T &rhs) const { return length_ >= rhs.length_; }
|
|
|
|
forceinline constexpr bool operator ==(const T &rhs) const { return length_ == rhs.length_; }
|
|
|
|
forceinline constexpr bool operator !=(const T &rhs) const { return length_ != rhs.length_; }
|
2017-07-25 20:20:55 -04:00
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline constexpr bool operator !() const { return !length_; }
|
2017-07-27 21:38:50 -04:00
|
|
|
// bool operator () is not supported because it offers an implicit cast to int, which is prone silently to permit misuse
|
2017-07-25 20:20:55 -04:00
|
|
|
|
2019-10-29 22:36:29 -04:00
|
|
|
/// @returns The underlying int, cast to an integral type of your choosing.
|
2019-11-18 22:11:52 -05:00
|
|
|
template<typename Type = IntType> forceinline constexpr Type as() const { return Type(length_); }
|
2019-10-29 22:36:29 -04:00
|
|
|
|
|
|
|
/// @returns The underlying int, in its native form.
|
|
|
|
forceinline constexpr IntType as_integral() const { return length_; }
|
2017-07-25 20:20:55 -04:00
|
|
|
|
|
|
|
/*!
|
2018-05-13 15:34:31 -04:00
|
|
|
Severs from @c this the effect of dividing by @c divisor; @c this will end up with
|
2017-07-25 20:20:55 -04:00
|
|
|
the value of @c this modulo @c divisor and @c divided by @c divisor is returned.
|
|
|
|
*/
|
2019-10-13 18:19:39 -04:00
|
|
|
template <typename Result = T> forceinline Result divide(const T &divisor) {
|
|
|
|
Result r;
|
|
|
|
static_cast<T *>(this)->fill(r, divisor);
|
|
|
|
return r;
|
2017-07-25 20:20:55 -04:00
|
|
|
}
|
|
|
|
|
2019-07-29 23:04:02 -04:00
|
|
|
/*!
|
|
|
|
Flushes the value in @c this. The current value is returned, and the internal value
|
|
|
|
is reset to zero.
|
|
|
|
*/
|
2019-07-29 23:07:02 -04:00
|
|
|
template <typename Result> Result flush() {
|
2019-07-29 23:04:02 -04:00
|
|
|
// Jiggery pokery here; switching to function overloading avoids
|
|
|
|
// the namespace-level requirement for template specialisation.
|
|
|
|
Result r;
|
2019-07-30 10:54:56 -04:00
|
|
|
static_cast<T *>(this)->fill(r);
|
2019-07-29 23:04:02 -04:00
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2017-07-25 20:20:55 -04:00
|
|
|
// operator int() is deliberately not provided, to avoid accidental subtitution of
|
|
|
|
// classes that use this template.
|
|
|
|
|
|
|
|
protected:
|
2019-10-29 22:36:29 -04:00
|
|
|
IntType length_;
|
2017-07-25 20:20:55 -04:00
|
|
|
};
|
|
|
|
|
2018-05-13 15:34:31 -04:00
|
|
|
/// Describes an integer number of whole cycles: pairs of clock signal transitions.
|
2017-07-25 20:20:55 -04:00
|
|
|
class Cycles: public WrappedInt<Cycles> {
|
|
|
|
public:
|
2019-10-29 22:36:29 -04:00
|
|
|
forceinline constexpr Cycles(IntType l) noexcept : WrappedInt<Cycles>(l) {}
|
2019-07-26 22:18:40 -04:00
|
|
|
forceinline constexpr Cycles() noexcept : WrappedInt<Cycles>() {}
|
2019-07-29 22:49:02 -04:00
|
|
|
|
2019-07-29 23:04:02 -04:00
|
|
|
private:
|
|
|
|
friend WrappedInt;
|
2019-07-30 10:54:56 -04:00
|
|
|
void fill(Cycles &result) {
|
2019-07-29 23:04:02 -04:00
|
|
|
result.length_ = length_;
|
2019-07-29 22:49:02 -04:00
|
|
|
length_ = 0;
|
|
|
|
}
|
2019-10-13 18:19:39 -04:00
|
|
|
|
|
|
|
void fill(Cycles &result, const Cycles &divisor) {
|
|
|
|
result.length_ = length_ / divisor.length_;
|
|
|
|
length_ %= divisor.length_;
|
|
|
|
}
|
2017-07-25 20:20:55 -04:00
|
|
|
};
|
|
|
|
|
2018-05-13 15:34:31 -04:00
|
|
|
/// Describes an integer number of half cycles: single clock signal transitions.
|
2017-07-25 20:20:55 -04:00
|
|
|
class HalfCycles: public WrappedInt<HalfCycles> {
|
|
|
|
public:
|
2019-10-29 22:36:29 -04:00
|
|
|
forceinline constexpr HalfCycles(IntType l) noexcept : WrappedInt<HalfCycles>(l) {}
|
2019-07-26 22:18:40 -04:00
|
|
|
forceinline constexpr HalfCycles() noexcept : WrappedInt<HalfCycles>() {}
|
2017-07-25 20:20:55 -04:00
|
|
|
|
2019-10-29 22:36:29 -04:00
|
|
|
forceinline constexpr HalfCycles(const Cycles &cycles) noexcept : WrappedInt<HalfCycles>(cycles.as_integral() * 2) {}
|
2017-07-25 20:20:55 -04:00
|
|
|
|
2017-07-27 21:38:50 -04:00
|
|
|
/// @returns The number of whole cycles completely covered by this span of half cycles.
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline constexpr Cycles cycles() const {
|
2017-07-27 21:38:50 -04:00
|
|
|
return Cycles(length_ >> 1);
|
|
|
|
}
|
2017-07-25 20:20:55 -04:00
|
|
|
|
2017-08-02 07:21:21 -04:00
|
|
|
/*!
|
2018-05-13 15:34:31 -04:00
|
|
|
Severs from @c this the effect of dividing by @c divisor; @c this will end up with
|
2017-08-02 07:21:21 -04:00
|
|
|
the value of @c this modulo @c divisor and @c divided by @c divisor is returned.
|
|
|
|
*/
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline Cycles divide_cycles(const Cycles &divisor) {
|
|
|
|
const HalfCycles half_divisor = HalfCycles(divisor);
|
|
|
|
const Cycles result(length_ / half_divisor.length_);
|
2017-08-02 07:21:21 -04:00
|
|
|
length_ %= half_divisor.length_;
|
|
|
|
return result;
|
|
|
|
}
|
2019-07-29 23:04:02 -04:00
|
|
|
|
|
|
|
private:
|
|
|
|
friend WrappedInt;
|
2019-07-30 10:54:56 -04:00
|
|
|
void fill(Cycles &result) {
|
2019-07-29 23:04:02 -04:00
|
|
|
result = Cycles(length_ >> 1);
|
|
|
|
length_ &= 1;
|
|
|
|
}
|
|
|
|
|
2019-07-30 10:54:56 -04:00
|
|
|
void fill(HalfCycles &result) {
|
2019-07-29 23:04:02 -04:00
|
|
|
result.length_ = length_;
|
|
|
|
length_ = 0;
|
|
|
|
}
|
2019-10-13 18:19:39 -04:00
|
|
|
|
|
|
|
void fill(Cycles &result, const HalfCycles &divisor) {
|
|
|
|
result = Cycles(length_ / (divisor.length_ << 1));
|
|
|
|
length_ %= (divisor.length_ << 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void fill(HalfCycles &result, const HalfCycles &divisor) {
|
|
|
|
result.length_ = length_ / divisor.length_;
|
|
|
|
length_ %= divisor.length_;
|
|
|
|
}
|
2017-07-27 21:38:50 -04:00
|
|
|
};
|
2017-07-25 20:20:55 -04:00
|
|
|
|
2019-07-29 21:23:37 -04:00
|
|
|
// Create a specialisation of WrappedInt::flush for converting HalfCycles to Cycles
|
|
|
|
// without losing the fractional part.
|
|
|
|
|
2017-07-27 07:40:02 -04:00
|
|
|
/*!
|
|
|
|
If a component implements only run_for(Cycles), an owner can wrap it in HalfClockReceiver
|
|
|
|
automatically to gain run_for(HalfCycles).
|
|
|
|
*/
|
|
|
|
template <class T> class HalfClockReceiver: public T {
|
|
|
|
public:
|
|
|
|
using T::T;
|
2017-07-25 20:20:55 -04:00
|
|
|
|
2019-07-07 14:12:20 -04:00
|
|
|
forceinline void run_for(const HalfCycles half_cycles) {
|
2017-07-27 21:38:50 -04:00
|
|
|
half_cycles_ += half_cycles;
|
2019-07-28 21:49:54 -04:00
|
|
|
T::run_for(half_cycles_.flush<Cycles>());
|
2017-07-25 20:20:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2017-07-27 21:38:50 -04:00
|
|
|
HalfCycles half_cycles_;
|
2017-07-25 20:20:55 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif /* ClockReceiver_hpp */
|