tdc_gpx2/source/gpx2-raspi-readout-program/include/gpio.h
2023-08-11 14:09:43 +02:00

129 lines
3.1 KiB
C++

#ifndef GPIO_H
#define GPIO_H
#include <vector>
#include <mutex>
#include <shared_mutex>
#include <chrono>
#include <future>
#include <map>
#include <memory>
#include <linux/gpio.h>
#include <gpiod.h>
// TODO: check correctness of mutex'
// TODO: implement and check init, step, shutdown, write functions
enum class PinBias : std::uint8_t {
OpenDrain = 0x01,
OpenSource = 0x02,
ActiveLow = 0x04,
PullDown = 0x10,
PullUp = 0x20
};
class gpio
{
public:
constexpr static auto standard_timeout = std::chrono::seconds{10};
struct event
{
enum Type
{
Invalid,
Rising,
Falling
} type{ Invalid };
std::size_t pin{};
timespec ts;
inline operator bool()
{
return type != Invalid;
}
};
struct setting
{
[[nodiscard]] auto matches(const event& e) const -> bool;
std::vector<unsigned> gpio_pins;
};
class callback
{
friend class gpio;
public:
[[nodiscard]] auto wait_async(std::chrono::milliseconds timeout)->std::future<event>;
[[nodiscard]] auto wait(std::chrono::milliseconds timeout)->event;
[[nodiscard]] auto write_async(const event& e)->std::future<bool>;
[[nodiscard]] auto write(const event& e) -> bool;
[[nodiscard]] auto read(unsigned pin_num) -> int;
callback(setting s, gpio& handler);
private:
void notify(const event& e);
setting m_setting{};
event m_event {};
std::condition_variable m_wait{};
std::mutex m_wait_mutex{};
std::shared_mutex m_access_mutex{};
gpio& m_handler;
};
gpio(std::string consumer = "gpio", std::string chipname = "gpiochip0")
: m_consumer{std::move(consumer)}
, m_chipname{std::move(chipname)}
{};
virtual ~gpio();
void start();
void stop();
void join();
[[nodiscard]] auto list_callback(setting s)->std::shared_ptr<callback>;
void remove_callback(std::size_t id);
private:
[[nodiscard]] auto setup() -> int;
[[nodiscard]] auto step() -> int;
[[nodiscard]] auto shutdown() -> int;
[[nodiscard]] auto write(const event& e) -> bool;
[[nodiscard]] auto read(unsigned pin_num) -> int;
void notify_all(event e);
std::string m_consumer;
std::string m_chipname;
const timespec c_wait_timeout{1,0};
//std::shared_ptr<::gpiod_chip> chip{nullptr};
//std::shared_ptr<::gpiod_line_bulk> lines{nullptr};
gpiod_chip* chip{nullptr};
gpiod_line_bulk* lines{nullptr};
std::map< unsigned, gpiod_line* > other_lines{}; // map of other initialized lines which are requested but have no active event listening going on
std::vector<unsigned> pins_used_by_listeners{};
//inline static std::size_t global_id_counter{ 0 };
std::size_t global_id_counter{ 0 };
std::map < std::size_t, std::shared_ptr<callback> > m_callback{};
std::future<int> m_result{};
std::chrono::microseconds m_timeout{ 10 };
std::atomic<bool> m_run{ true };
std::vector<setting> m_settings{};
};
#endif // GPIO_H