Merge pull request #1539 from aschuhardt/libgpiod

Refactoring GPIO push-to-talk logic to use libgpiod rather than the d…
pull/1551/head
Michael Black 2024-05-12 11:41:31 -05:00 zatwierdzone przez GitHub
commit c06255d364
Nie znaleziono w bazie danych klucza dla tego podpisu
ID klucza GPG: B5690EEEBB952194
3 zmienionych plików z 49 dodań i 95 usunięć

Wyświetl plik

@ -529,6 +529,9 @@ AS_IF([test x"${cf_with_xml_support}" = "xyes"], [
AC_SUBST([LIBXML2_LIBS]) AC_SUBST([LIBXML2_LIBS])
AC_SUBST([LIBXML2_CFLAGS]) AC_SUBST([LIBXML2_CFLAGS])
# TODO: I don't know how to use autoconf
LIBS="$LIBS -lgpiod"
## ----------------- ## ## ----------------- ##
## Language bindings ## ## Language bindings ##

Wyświetl plik

@ -2340,6 +2340,7 @@ typedef struct hamlib_port {
int fd; /*!< File descriptor */ int fd; /*!< File descriptor */
void *handle; /*!< handle for USB */ void *handle; /*!< handle for USB */
void *gpio; /*!< handle for GPIO */
int write_delay; /*!< Delay between each byte sent out, in mS */ int write_delay; /*!< Delay between each byte sent out, in mS */
int post_write_delay; /*!< Delay between each commands send out, in mS */ int post_write_delay; /*!< Delay between each commands send out, in mS */

Wyświetl plik

@ -19,135 +19,91 @@
* *
*/ */
#include <string.h>
#include <errno.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include "gpio.h" #include "gpio.h"
#include <errno.h>
#include <fcntl.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <gpiod.h>
#ifndef GPIOD_PATH
// This is what's used on the Raspberry Pi 4; I'm not sure about others
#define GPIO_CHIP_NAME "gpiochip0"
#endif
#define GPIO_CHIP_CONSUMER "Hamlib"
int gpio_open(hamlib_port_t *port, int output, int on_value) int gpio_open(hamlib_port_t *port, int output, int on_value)
{ {
char pathname[HAMLIB_FILPATHLEN * 2]; struct gpiod_chip *chip;
FILE *fexp, *fdir; struct gpiod_line *line;
int fd;
char *dir;
port->parm.gpio.on_value = on_value; port->parm.gpio.on_value = on_value;
SNPRINTF(pathname, HAMLIB_FILPATHLEN, "/sys/class/gpio/export"); chip = gpiod_chip_open_by_name(GPIO_CHIP_NAME);
fexp = fopen(pathname, "w");
if (!fexp) if (!chip)
{ {
rig_debug(RIG_DEBUG_ERR, rig_debug(RIG_DEBUG_ERR, "Failed to open GPIO chip %s: %s\n", GPIO_CHIP_NAME, strerror(errno));
"Export GPIO%s (using %s): %s\n",
port->pathname,
pathname,
strerror(errno));
return -RIG_EIO; return -RIG_EIO;
} }
fprintf(fexp, "%s\n", port->pathname); line = gpiod_chip_get_line(chip, atoi(port->pathname));
fclose(fexp); if (!line)
SNPRINTF(pathname,
sizeof(pathname),
"/sys/class/gpio/gpio%s/direction",
port->pathname);
fdir = fopen(pathname, "w");
if (!fdir)
{ {
rig_debug(RIG_DEBUG_ERR, rig_debug(RIG_DEBUG_ERR, "Failed to acquire GPIO%s: %s\n", port->pathname, strerror(errno));
"GPIO%s direction (using %s): %s\n", gpiod_chip_close(chip);
port->pathname,
pathname,
strerror(errno));
return -RIG_EIO; return -RIG_EIO;
} }
dir = output ? "out" : "in"; if ((output && gpiod_line_request_output(line, GPIO_CHIP_CONSUMER, 0) < 0) ||
rig_debug(RIG_DEBUG_VERBOSE, "Setting direction of GPIO%s to %s\n", (!output && gpiod_line_request_input(line, GPIO_CHIP_CONSUMER) < 0))
port->pathname, dir);
fprintf(fdir, "%s\n", dir);
fclose(fdir);
SNPRINTF(pathname,
sizeof(pathname),
"/sys/class/gpio/gpio%s/value",
port->pathname);
fd = open(pathname, O_RDWR);
if (fd < 0)
{ {
rig_debug(RIG_DEBUG_ERR, rig_debug(RIG_DEBUG_ERR, "Failed to set GPIO%s to %s mode: %s\n",
"GPIO%s opening value file %s: %s\n", port->pathname, (output ? "OUTPUT" : "INPUT"), strerror(errno));
port->pathname, gpiod_line_release(line);
pathname, gpiod_chip_close(chip);
strerror(errno));
return -RIG_EIO; return -RIG_EIO;
} }
port->fd = fd; port->gpio = line;
return fd;
return RIG_OK;
} }
int gpio_close(hamlib_port_t *port) int gpio_close(hamlib_port_t *port)
{ {
int retval; gpiod_line_close_chip((struct gpiod_line*)port->gpio);
char pathname[HAMLIB_FILPATHLEN * 2]; return RIG_OK;
FILE *fexp;
retval = close(port->fd);
SNPRINTF(pathname, HAMLIB_FILPATHLEN, "/sys/class/gpio/unexport");
fexp = fopen(pathname, "w");
if (!fexp)
{
rig_debug(RIG_DEBUG_ERR,
"Export GPIO%s (using %s): %s\n",
port->pathname,
pathname,
strerror(errno));
return -RIG_EIO;
}
fprintf(fexp, "%s\n", port->pathname);
fclose(fexp);
return retval;
} }
int gpio_ptt_set(hamlib_port_t *port, ptt_t pttx) int gpio_ptt_set(hamlib_port_t *port, ptt_t pttx)
{ {
char *val; int result = 0;
port->parm.gpio.value = pttx != RIG_PTT_OFF; port->parm.gpio.value = pttx != RIG_PTT_OFF;
if ((port->parm.gpio.value && port->parm.gpio.on_value) if ((port->parm.gpio.value && port->parm.gpio.on_value) ||
|| (!port->parm.gpio.value && !port->parm.gpio.on_value)) (!port->parm.gpio.value && !port->parm.gpio.on_value))
{ {
val = "1\n"; result = gpiod_line_set_value((struct gpiod_line*)port->gpio, 1);
} }
else else
{ {
val = "0\n"; result = gpiod_line_set_value((struct gpiod_line*)port->gpio, 0);
} }
if (write(port->fd, val, strlen(val)) <= 0) if (result)
{ {
rig_debug(RIG_DEBUG_ERR, "Failed to set the value of GPIO%s: %s\n", port->pathname, strerror(errno));
return -RIG_EIO; return -RIG_EIO;
} }
return RIG_OK; return RIG_OK;
} }
int gpio_ptt_get(hamlib_port_t *port, ptt_t *pttx) int gpio_ptt_get(hamlib_port_t *port, ptt_t *pttx)
{ {
if (port->parm.gpio.value) if (port->parm.gpio.value)
@ -164,21 +120,15 @@ int gpio_ptt_get(hamlib_port_t *port, ptt_t *pttx)
int gpio_dcd_get(hamlib_port_t *port, dcd_t *dcdx) int gpio_dcd_get(hamlib_port_t *port, dcd_t *dcdx)
{ {
char val; int val = gpiod_line_get_value((struct gpiod_line*)port->gpio);
int port_value; if (val < 0)
lseek(port->fd, 0, SEEK_SET);
if (read(port->fd, &val, sizeof(val)) <= 0)
{ {
return -RIG_EIO; rig_debug(RIG_DEBUG_ERR, "Failed to read the value of GPIO%s: %s\n", port->pathname, strerror(errno));
} }
rig_debug(RIG_DEBUG_VERBOSE, "DCD GPIO pin value: %c\n", val); rig_debug(RIG_DEBUG_VERBOSE, "DCD GPIO pin value: %c\n", val);
port_value = val - '0'; if (val == port->parm.gpio.on_value)
if (port_value == port->parm.gpio.on_value)
{ {
*dcdx = RIG_DCD_ON; *dcdx = RIG_DCD_ON;
} }