Skip to content

Commit

Permalink
libticables: Introduce a cable for using GPIO pins on RPi SBCs
Browse files Browse the repository at this point in the history
With this cable, a calculator can be connected to pins 16 (Tip/Red) and 18
(Ring/White) ideally through a ~1k resistor. Transmission happens by switching
the lines between tristate input and 0V output states.

In case the voltage of the calculator exceeds 3.3V, diodes should be used
in addition to the resistors to limit the line voltage to 3V3 to avoid
damage.
  • Loading branch information
Vogtinator committed Feb 20, 2022
1 parent d1780dc commit df23c90
Show file tree
Hide file tree
Showing 5 changed files with 363 additions and 0 deletions.
11 changes: 11 additions & 0 deletions libticables/trunk/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,8 @@ set(PUBLIC_HEADERS

# external deps lookup
pkg_check_modules(DEPS REQUIRED glib-2.0 libusb-1.0>=1.0.16)
# 1.5 introduced support for SET_CONFIG
pkg_check_modules(LIBGPIOD libgpiod>=1.5 IMPORTED_TARGET)

# check includes for parallel and serial support
include(CheckIncludeFile)
Expand All @@ -48,6 +50,10 @@ CHECK_INCLUDE_FILE(linux/serial.h HAVE_LINUX_SERIAL_H)

try_static_libs_if_needed()

if(LIBGPIOD_FOUND)
set(SRC_FILES ${SRC_FILES} src/link_gpio.cc)
endif()

# Needed for the .pc files configured in the function below.
set(TICABLES_LIBUSB_REQUIRES_PRIVATE "libusb-1.0")

Expand All @@ -62,6 +68,11 @@ if(ENABLE_NLS)
add_dependencies(ticables2_objlib potfiles_2)
endif()

if(LIBGPIOD_FOUND)
target_compile_definitions(ticables2_objlib PUBLIC HAVE_LIBGPIOD=1)
target_link_libraries(ticables2_shared PkgConfig::LIBGPIOD)
endif()

# additional internal defines
target_compile_definitions(ticables2_objlib PUBLIC HAVE_LIBUSB_1_0=1 HAVE_LIBUSB10_STRERROR=1 HAVE_TERMIOS_H=1)

Expand Down
347 changes: 347 additions & 0 deletions libticables/trunk/src/link_gpio.cc
Original file line number Diff line number Diff line change
@@ -0,0 +1,347 @@
/* Hey EMACS -*- linux-c -*- */
/* $Id$ */

/* libticables2 - link cable library, a part of the TiLP project
* Copyright (C) 1999-2005 Romain Lievin
* Copyright (C) 2022 Fabian Vogt
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/

/* Cable using Pins 16 (GPIO 23) and 18 (GPIO 24) of Raspberry Pi SBCs.
*
* Both GPIOs can be configured as input (tristate/Hi-Z) and output (3V3/0V).
* When the GPIO is set as input, the line voltage is not affected and it can
* be sensed whether it is pulled low by the other side. To pull the line low,
* it is configured as 0V output. It is never set as high (3V3) output. */

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include <stdio.h>
#include <stdint.h>
#include <gpiod.h>

#include "ticables.h"
#include "error.h"
#include "logging.h"
#include "gettext.h"
#include "internal.h"

enum {
BIT_RED = 0b01,
BIT_WHITE = 0b10,
};

/* Syscalls are costly, so it would be beneficial to query and set both GPIO
* lines at once. With gpiod_line_bulk, getting the values of both lines is
* possible, but it doesn't support changing the direction of them
* independently. This should be possible with the GPIO_V2 IOCTL API in the
* future and libgpiod 2.x might expose it at some point.
*
* It's not possible to use gpiod_line_bulk for querying and individual
* gpiod_line instances for setting either, as a line can only be in use
* once. So for the time being, just use a gpiod_line for each. */
struct GPIOPriv {
struct gpiod_chip *chip;
struct gpiod_line *line_red, *line_white;
};

static int cable_gpio_close(CableHandle *h);

static int cable_gpio_open(CableHandle *h)
{
GPIOPriv *priv = (GPIOPriv*) calloc(1, sizeof(GPIOPriv));
h->priv = priv;

if (!priv) {
ticables_warning(_("Memory allocation failed."));
goto err_cleanup;
}

priv->chip = gpiod_chip_open("/dev/gpiochip0");
if (!priv->chip) {
ticables_warning(_("Failed to open /dev/gpiochip0."));
goto err_cleanup;
}

if (strcmp(gpiod_chip_label(priv->chip), "pinctrl-bcm2835") != 0) {
ticables_warning(_("/dev/gpiochip0 isn't the expected device."));
goto err_cleanup;
}

priv->line_red = gpiod_chip_get_line(priv->chip, 23),
priv->line_white = gpiod_chip_get_line(priv->chip, 24);

if(!priv->line_red || !priv->line_white
|| gpiod_line_request_input(priv->line_red, "libticables") != 0
|| gpiod_line_request_input(priv->line_white, "libticables") != 0) {
ticables_warning(_("Failed to request GPIO lines."));
goto err_cleanup;
}

return ERR_NO_ERROR;

err_cleanup:
cable_gpio_close(h);
return ERR_TTY_OPEN;
}

static int cable_gpio_close(CableHandle *h)
{
GPIOPriv *priv = (GPIOPriv*) h->priv;
if (!priv)
return 0;

// It does not hurt to release a line which wasn't requested
if (priv->line_red)
gpiod_line_release(priv->line_red);

if (priv->line_white)
gpiod_line_release(priv->line_white);

if (priv->chip)
gpiod_chip_close(priv->chip);

free(h->priv);
h->priv = NULL;
return 0;
}

/* Unfortunately gpiod_line_set_config internally does GPIO_GET_LINEINFO_IOCTL
* after changing the config, which halves the speed of this function. */
static int set_line(struct gpiod_line *line, bool b)
{
int ret;
if(b)
ret = gpiod_line_set_config(line, GPIOD_LINE_REQUEST_DIRECTION_INPUT, 0, 0);
else
ret = gpiod_line_set_config(line, GPIOD_LINE_REQUEST_DIRECTION_OUTPUT, 0, 0);

return ret == 0;
}

static int cable_gpio_set_red_wire(CableHandle *h, int b)
{
GPIOPriv *priv = (GPIOPriv*) h->priv;
set_line(priv->line_red, b);
return 0;
}

static int cable_gpio_set_white_wire(CableHandle *h, int b)
{
GPIOPriv *priv = (GPIOPriv*) h->priv;
set_line(priv->line_white, b);
return 0;
}

static int cable_gpio_set_raw(CableHandle *h, int state)
{
GPIOPriv *priv = (GPIOPriv*) h->priv;
set_line(priv->line_white, state & BIT_WHITE);
set_line(priv->line_red, state & BIT_RED);
return 0;
}

/* Get the current level of the GPIO lines given in mask, store in *state.
* The mask is used to avoid unnecessary ioctl calls. */
static int cable_gpio_get_raw_masked(CableHandle *h, int *state, int mask)
{
GPIOPriv *priv = (GPIOPriv*) h->priv;

*state = 0;
if ((mask & BIT_WHITE) && gpiod_line_get_value(priv->line_white) == 1)
*state |= BIT_WHITE;
if ((mask & BIT_RED) && gpiod_line_get_value(priv->line_red) == 1)
*state |= BIT_RED;

return 0;
}

static int cable_gpio_get_raw(CableHandle *h, int *state)
{
return cable_gpio_get_raw_masked(h, state, 0b11);
}

static int cable_gpio_get_red_wire(CableHandle *h)
{
int state = 0;
cable_gpio_get_raw_masked(h, &state, BIT_RED);
return !!(state & BIT_RED);
}

static int cable_gpio_get_white_wire(CableHandle *h)
{
int state = 0;
cable_gpio_get_raw_masked(h, &state, BIT_WHITE);
return !!(state & BIT_WHITE);
}

// Wait until cable_gpio_get_raw() & mask == want.
static int wait_for_line_state(CableHandle *h, int mask, int want)
{
tiTIME clk;
bool to_started = false;

while (true) {
int state;
int ret = cable_gpio_get_raw_masked(h, &state, mask);
if (ret != 0)
return ret;

if ((state & mask) == want)
return 0;

// Syscalls are expensive, so avoid unnecessary calls to
// TO_START/TO_ELAPSED.
if (!to_started) {
TO_START(clk);
to_started = true;
} else if (TO_ELAPSED(clk, h->timeout))
return ERR_READ_TIMEOUT; // TODO: What if called from put?
}
}

static int cable_gpio_probe(CableHandle *h)
{
int ret;

// Stop pulling lines low
ret = cable_gpio_set_raw(h, 0b11);
if (ret != 0)
return ret;

// Wait for them to read high
return wait_for_line_state(h, 0b11, 0b11);
}

static int cable_gpio_reset(CableHandle *h)
{
return cable_gpio_probe(h);
}

static int cable_gpio_put(CableHandle *h, uint8_t *data, uint32_t len)
{
for(unsigned int off = 0; off < len; off++) {
const uint8_t byte = data[off];
int ret;

for (int bit = 0; bit < 8; bit++) {
if (byte & (1 << bit)) {
cable_gpio_set_white_wire(h, 0);
ret = wait_for_line_state(h, BIT_RED, 0);
if (ret)
return ret;
cable_gpio_set_white_wire(h, 1);
ret = wait_for_line_state(h, BIT_RED, BIT_RED);
if (ret)
return ret;
} else {
cable_gpio_set_red_wire(h, 0);
ret = wait_for_line_state(h, BIT_WHITE, 0);
if (ret)
return ret;
cable_gpio_set_red_wire(h, 1);
ret = wait_for_line_state(h, BIT_WHITE, BIT_WHITE);
if (ret)
return ret;
}
}
}

return 0;
}

static int cable_gpio_get(CableHandle *h, uint8_t *data, uint32_t len)
{
for(unsigned int off = 0; off < len; off++) {
uint8_t byte = 0;

for (int bit = 0; bit < 8; bit++) {
tiTIME clk;
bool to_started = false;
int state;
int ret;

// Wait until a line is low
while (true) {
ret = cable_gpio_get_raw(h, &state);
if (ret != 0)
return ret;

if (state != 0b11)
break;

if (!to_started) {
TO_START(clk);
to_started = true;
} else if (TO_ELAPSED(clk, h->timeout))
return ERR_READ_TIMEOUT;
}

byte >>= 1;

if (state == BIT_RED) {
// White line low -> 1 bit
byte |= 0x80;
cable_gpio_set_red_wire(h, 0);
ret = wait_for_line_state(h, BIT_WHITE, BIT_WHITE);
if (ret)
return ret;
cable_gpio_set_red_wire(h, 1);
} else {
cable_gpio_set_white_wire(h, 0);
ret = wait_for_line_state(h, BIT_RED, BIT_RED);
if (ret)
return ret;
cable_gpio_set_white_wire(h, 1);
}

/* There should be a delay according to h->delay here,
* but gpiod calls are slow enough already that it works reliably,
* and adding explicit waiting would only slow it down further. */
}

data[off] = byte;
}

return 0;
}

static int cable_gpio_check(CableHandle *h, int *status)
{
int state = 0;
cable_gpio_get_raw(h, &state);
*status = state == 0b11 ? STATUS_NONE : (STATUS_RX | STATUS_TX);
return 0;
}

extern const CableFncts cable_gpio =
{
CABLE_GPIO,
"GPIO",
N_("RPi GPIO Link"),
N_("Cable using RPi GPIO pins 16 and 18"),
1,
NULL,
&cable_gpio_open, &cable_gpio_close, &cable_gpio_reset, &cable_gpio_probe, NULL,
&cable_gpio_put, &cable_gpio_get, &cable_gpio_check,
&cable_gpio_set_red_wire, &cable_gpio_set_white_wire,
&cable_gpio_get_red_wire, &cable_gpio_get_white_wire,
&cable_gpio_set_raw, &cable_gpio_get_raw,
NULL,
NULL
};
1 change: 1 addition & 0 deletions libticables/trunk/src/link_xxx.h
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ extern const CableFncts cable_raw;
extern CableFncts cable_ilp;
extern const CableFncts cable_tcpc;
extern const CableFncts cable_tcps;
extern const CableFncts cable_gpio;

#endif

Expand Down
3 changes: 3 additions & 0 deletions libticables/trunk/src/ticables.cc
Original file line number Diff line number Diff line change
Expand Up @@ -86,6 +86,9 @@ static CableFncts const *const cables[] =
#endif
#if !defined(NO_CABLE_TCPS) && !defined(__WIN32__)
&cable_tcps,
#endif
#if defined(HAVE_LIBGPIOD)
&cable_gpio,
#endif
NULL
};
Expand Down
Loading

0 comments on commit df23c90

Please sign in to comment.