hackens-org-configurations/machines/agb02/agb/agb.cpp

185 lines
5.8 KiB
C++
Raw Normal View History

2024-10-13 10:42:30 +02:00
#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
2024-10-12 00:40:55 +02:00
#include <gpiod.hpp>
#include <iostream>
2024-10-12 18:44:35 +02:00
#include <fstream>
2024-10-12 00:40:55 +02:00
#include <thread>
using namespace std::literals::chrono_literals;
constexpr std::chrono::microseconds debounce = 40ms;
2024-10-12 16:00:59 +02:00
constexpr std::chrono::microseconds poll_period = 5ms;
2024-10-12 18:44:35 +02:00
constexpr std::chrono::microseconds server_ratelimit = 50ms;
2024-10-13 10:42:30 +02:00
constexpr std::chrono::microseconds retry_timeout = 500ms;
2024-10-12 00:40:55 +02:00
constexpr std::pair<double, double> joystick_movement(1.0, 1.0);
const gpiod::line::offsets drive_down = { 21, 13, 6 };
2024-10-13 19:41:47 +02:00
const gpiod::line::offsets decoder = { 3, 4, 17, 27, 24, 23, 18, 15 }; // lsbf
const gpiod::line::offsets joystick = { 26, 19, 0, 5 }; // x+, y+, x-, y-
const gpiod::line::offset black_button = 20;
const gpiod::line::offset white_button = 16;
2024-10-12 00:40:55 +02:00
const gpiod::line_settings input_settings =
gpiod::line_settings()
.set_direction(gpiod::line::direction::INPUT)
.set_bias(gpiod::line::bias::PULL_UP)
2024-10-13 10:42:30 +02:00
.set_active_low(false)
2024-10-12 00:40:55 +02:00
.set_debounce_period(debounce);
2024-10-12 16:00:59 +02:00
constexpr std::array<uint8_t, 256> decoder_table =
#include "decoder_table.inl"
2024-10-12 00:40:55 +02:00
uint8_t read_decoder_realpos(gpiod::line_request& line_reader){
static gpiod::line::values decoder_read(8);
line_reader.get_values(decoder, decoder_read);
uint8_t graycode = 0;
for(uint8_t i = 0; i < 8; ++i) graycode |= uint8_t(decoder_read[i]) << i;
2024-10-12 16:00:59 +02:00
return decoder_table[graycode];
2024-10-12 00:40:55 +02:00
};
inline void clamp_decoder(uint8_t& decoder, int move){
decoder = uint8_t(std::clamp(decoder + move, 0, 255));
}
int main(const int argc, char const* const* const argv) {
2024-10-13 10:42:30 +02:00
if(argc < 2) {
std::cerr << "usage: agb gpiodevice" << std::endl;
2024-10-12 00:40:55 +02:00
return 1;
2024-10-12 18:44:35 +02:00
}
2024-10-12 00:40:55 +02:00
/// init gpio chip ///
gpiod::chip chip(argv[1]);
gpiod::line_request line_reader =
chip.prepare_request()
.set_consumer("AGB")
.add_line_settings(drive_down,
gpiod::line_settings()
.set_direction(gpiod::line::direction::OUTPUT)
.set_drive(gpiod::line::drive::OPEN_DRAIN)
.set_output_value(gpiod::line::value::INACTIVE)
)
.add_line_settings({ black_button, white_button }, input_settings)
2024-10-13 10:42:30 +02:00
.add_line_settings(joystick,
gpiod::line_settings(input_settings)
.set_active_low(true))
2024-10-12 16:00:59 +02:00
.add_line_settings(decoder,
gpiod::line_settings(input_settings)
.set_debounce_period(0ms))
2024-10-12 00:40:55 +02:00
.do_request();
// let the settings apply
std::this_thread::sleep_for(poll_period);
/// init server communication ///
2024-10-13 10:42:30 +02:00
int socket_file_desc = socket(AF_INET, SOCK_STREAM, 0);
2024-10-12 18:44:35 +02:00
{
2024-10-13 10:42:30 +02:00
sockaddr_in socket_addr = {
.sin_family = AF_INET,
.sin_port = htons(1235),
.sin_addr = { .s_addr = inet_addr("10.10.10.1") }
};
int con_ret = connect(socket_file_desc,
reinterpret_cast<const sockaddr*>(&socket_addr),
sizeof(socket_addr));
if(con_ret < 0) {
std::cerr << "Failed to open tcp socket." << std::endl;
return 1;
}
2024-10-12 18:44:35 +02:00
}
2024-10-12 00:40:55 +02:00
2024-10-12 18:44:35 +02:00
/// internal state and buffers ///
2024-10-12 00:40:55 +02:00
gpiod::line::values joystick_read(4);
std::pair<double, double> spot_pos(0.0, 0.0); //TODO: init from server
uint8_t decoder_pos = 0; //TODO: init from server
uint8_t decoder_realpos = read_decoder_realpos(line_reader);
bool white_pressed = false;
bool black_pressed = false;
2024-10-13 10:42:30 +02:00
bool has_changed = true;
2024-10-12 18:44:35 +02:00
std::chrono::time_point last_send = std::chrono::system_clock::now();
std::string postData;
2024-10-12 00:40:55 +02:00
for(;;){
std::this_thread::sleep_for(poll_period);
/// joystick ///
line_reader.get_values(joystick, joystick_read);
if(bool(joystick_read[0])) spot_pos.first += joystick_movement.first;
if(bool(joystick_read[1])) spot_pos.second += joystick_movement.second;
if(bool(joystick_read[2])) spot_pos.first -= joystick_movement.first;
if(bool(joystick_read[3])) spot_pos.second -= joystick_movement.second;
if (bool(joystick_read[0]) || bool(joystick_read[1])
|| bool(joystick_read[2]) || bool(joystick_read[3]))
2024-10-12 18:44:35 +02:00
has_changed = true;
2024-10-12 00:40:55 +02:00
/// Buttons ///
bool pressed = bool(line_reader.get_value(black_button));
if(pressed ^ black_pressed)
2024-10-12 18:44:35 +02:00
has_changed = true;
2024-10-12 00:40:55 +02:00
black_pressed = pressed;
pressed = bool(line_reader.get_value(white_button));
if(pressed ^ white_pressed)
2024-10-12 18:44:35 +02:00
has_changed = true;
2024-10-12 00:40:55 +02:00
white_pressed = pressed;
/// decoder ///
uint8_t new_realpos = read_decoder_realpos(line_reader);
uint8_t seen_travel = std::abs(int(new_realpos) - int(decoder_realpos));
2024-10-12 18:44:35 +02:00
// CCW
2024-10-12 00:40:55 +02:00
if(seen_travel < 50 && new_realpos < decoder_realpos)
2024-10-12 16:00:59 +02:00
clamp_decoder(decoder_pos, -seen_travel);
2024-10-12 00:40:55 +02:00
if(seen_travel >= 50 && new_realpos > decoder_realpos)
2024-10-12 16:00:59 +02:00
clamp_decoder(decoder_pos, seen_travel - 128);
2024-10-12 00:40:55 +02:00
2024-10-12 18:44:35 +02:00
// CW
2024-10-12 00:40:55 +02:00
if(seen_travel < 50 && new_realpos > decoder_realpos)
2024-10-12 16:00:59 +02:00
clamp_decoder(decoder_pos, seen_travel);
2024-10-12 00:40:55 +02:00
if(seen_travel >= 50 && new_realpos < decoder_realpos)
2024-10-12 16:00:59 +02:00
clamp_decoder(decoder_pos, 128 - seen_travel);
2024-10-12 00:40:55 +02:00
2024-10-12 16:00:59 +02:00
decoder_realpos = new_realpos;
2024-10-12 00:40:55 +02:00
if(seen_travel)
2024-10-12 18:44:35 +02:00
has_changed = true;
/// server notification
std::chrono::time_point now = std::chrono::system_clock::now();
if(has_changed && (now - last_send > server_ratelimit)){
postData.clear();
std::format_to(std::back_inserter(postData), "{{"
"\"pan\": {},"
"\"tilt\": {},"
"\"focus\": {},"
"\"whiteButton\": {},"
"\"blackButton\": {}"
2024-10-13 10:42:30 +02:00
"}}\n",
2024-10-12 18:44:35 +02:00
spot_pos.first,
spot_pos.second,
int(decoder_pos),
white_pressed,
black_pressed
);
2024-10-13 10:42:30 +02:00
int wrote = write(socket_file_desc, postData.data(), postData.size());
if(wrote < postData.size()){
std::cerr << "Failed to send data, retrying in " << retry_timeout << "." << std::endl;
last_send = now + retry_timeout - server_ratelimit;
} else {
has_changed = false;
last_send = now;
}
2024-10-12 18:44:35 +02:00
}
2024-10-12 00:40:55 +02:00
}
}