Skip to content

Commit a3f6820

Browse files
committed
Merge branch 'ew1abz-gpiod' into dev
2 parents 6192661 + c87534f commit a3f6820

9 files changed

+147
-8
lines changed

.gitignore

+1-1
Original file line numberDiff line numberDiff line change
@@ -109,5 +109,5 @@ $RECYCLE.BIN/
109109
*.dSYM
110110

111111
# cmake
112-
build/
112+
build*/
113113
tmp/

CMakeLists.txt

+8
Original file line numberDiff line numberDiff line change
@@ -320,6 +320,14 @@ else()
320320
set(HAMLIB_LIBRARIES "")
321321
endif()
322322

323+
find_package(gpiod)
324+
if(GPIOD_FOUND)
325+
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DUSE_GPIOD")
326+
else()
327+
set(GPIOD_INCLUDE_DIRS "")
328+
set(GPIOD_LIBRARIES "")
329+
endif()
330+
323331
if(LINUX)
324332
find_package(ALSA REQUIRED)
325333
if(ALSA_FOUND)

cmake/modules/FindCompiler.cmake

+2-2
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,9 @@ elseif(NOT DEFINED C_GCC AND CMAKE_CXX_COMPILER_ID MATCHES "GNU")
55
set(C_GCC 1)
66
elseif(NOT DEFINED C_MSVC AND CMAKE_CXX_COMPILER_ID MATCHES "MSVC")
77
set(C_MSVC 1)
8-
if(MSVC_VERSION GREATER_EQUAL 1920 AND MSVC_VERSION LESS_EQUAL 1929)
8+
if(MSVC_VERSION GREATER 1919 AND MSVC_VERSION LESS 1926)
99
set(VS2019 ON)
10-
elseif(MSVC_VERSION GREATER_EQUAL 1910 AND MSVC_VERSION LESS_EQUAL 1919)
10+
elseif(MSVC_VERSION GREATER 1910 AND MSVC_VERSION LESS 1919)
1111
set(VS2017 ON)
1212
elseif(MSVC_VERSION GREATER 1899 AND MSVC_VERSION LESS 1910)
1313
set(VS2015 ON)

cmake/modules/Findgpiod.cmake

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
# - Try to find libgpiod
2+
# Once done this will define
3+
# GPIOD_FOUND - System has libgpiod
4+
# GPIOD_INCLUDE_DIRS - The libgpiod include directories
5+
# GPIOD_LIBRARIES - The libraries needed to use libgpiod
6+
# GPIOD_DEFINITIONS - Compiler switches required for using libgpiod
7+
8+
find_package(PkgConfig)
9+
pkg_check_modules(PC_GPIOD QUIET gpiod)
10+
11+
find_path(GPIOD_INCLUDE_DIR gpiod.h)
12+
find_library(GPIOD_LIBRARY NAMES gpiod)
13+
14+
include(FindPackageHandleStandardArgs)
15+
# handle the QUIETLY and REQUIRED arguments and set GPIOD_FOUND to TRUE
16+
# if all listed variables are TRUE
17+
find_package_handle_standard_args(gpiod DEFAULT_MSG
18+
GPIOD_LIBRARY GPIOD_INCLUDE_DIR)
19+
20+
mark_as_advanced(GPIOD_INCLUDE_DIR GPIOD_LIBRARY)
21+
22+
set(GPIOD_LIBRARIES ${GPIOD_LIBRARY})
23+
set(GPIOD_INCLUDE_DIRS ${GPIOD_INCLUDE_DIR})

src/CMakeLists.txt

+2
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ include_directories(
1010
${PORTAUDIO_INCLUDE_DIRS}
1111
${SNDIO_INCLUDE_DIRS}
1212
${CUSTOM_GEOTRANZ_DIR}
13+
${GPIOD_INCLUDE_DIRS}
1314
${CUSTOM_HIDAPI_DIR}
1415
)
1516

@@ -155,6 +156,7 @@ target_link_libraries(direwolf
155156
${ALSA_LIBRARIES}
156157
${UDEV_LIBRARIES}
157158
${PORTAUDIO_LIBRARIES}
159+
${GPIOD_LIBRARIES}
158160
${SNDIO_LIBRARIES}
159161
${AVAHI_LIBRARIES}
160162
)

src/audio.h

+3-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,8 @@
2828
enum ptt_method_e {
2929
PTT_METHOD_NONE, /* VOX or no transmit. */
3030
PTT_METHOD_SERIAL, /* Serial port RTS or DTR. */
31-
PTT_METHOD_GPIO, /* General purpose I/O, Linux only. */
31+
PTT_METHOD_GPIO, /* General purpose I/O using sysfs, deprecated after 2020, Linux only. */
32+
PTT_METHOD_GPIOD, /* General purpose I/O, using libgpiod, Linux only. */
3233
PTT_METHOD_LPT, /* Parallel printer port, Linux only. */
3334
PTT_METHOD_HAMLIB, /* HAMLib, Linux only. */
3435
PTT_METHOD_CM108 }; /* GPIO pin of CM108/CM119/etc. Linux only. */
@@ -311,6 +312,7 @@ struct audio_s {
311312
/* the case for CubieBoard where it was longer. */
312313
/* This is filled in by ptt_init so we don't have to */
313314
/* recalculate it each time we access it. */
315+
/* Also GPIO chip name for GPIOD method. Looks like 'gpiochip4' */
314316

315317
/* This could probably be collapsed into ptt_device instead of being separate. */
316318

src/config.c

+37
Original file line numberDiff line numberDiff line change
@@ -1842,6 +1842,43 @@ void config_init (char *fname, struct audio_s *p_audio_config,
18421842
}
18431843
p_audio_config->achan[channel].octrl[ot].ptt_method = PTT_METHOD_GPIO;
18441844
#endif
1845+
}
1846+
else if (strcasecmp(t, "GPIOD") == 0) {
1847+
#if __WIN32__
1848+
text_color_set(DW_COLOR_ERROR);
1849+
dw_printf ("Config file line %d: %s with GPIOD is only available on Linux.\n", line, otname);
1850+
#else
1851+
#if defined(USE_GPIOD)
1852+
t = split(NULL,0);
1853+
if (t == NULL) {
1854+
text_color_set(DW_COLOR_ERROR);
1855+
dw_printf ("Config file line %d: Missing GPIO chip for %s.\n", line, otname);
1856+
continue;
1857+
}
1858+
strlcpy(p_audio_config->achan[channel].octrl[ot].out_gpio_name, t,
1859+
sizeof(p_audio_config->achan[channel].octrl[ot].out_gpio_name));
1860+
1861+
t = split(NULL,0);
1862+
if (t == NULL) {
1863+
text_color_set(DW_COLOR_ERROR);
1864+
dw_printf("Config file line %d: Missing GPIO number for %s.\n", line, otname);
1865+
continue;
1866+
}
1867+
1868+
if (*t == '-') {
1869+
p_audio_config->achan[channel].octrl[ot].out_gpio_num = atoi(t+1);
1870+
p_audio_config->achan[channel].octrl[ot].ptt_invert = 1;
1871+
}
1872+
else {
1873+
p_audio_config->achan[channel].octrl[ot].out_gpio_num = atoi(t);
1874+
p_audio_config->achan[channel].octrl[ot].ptt_invert = 0;
1875+
}
1876+
p_audio_config->achan[channel].octrl[ot].ptt_method = PTT_METHOD_GPIOD;
1877+
#else
1878+
text_color_set(DW_COLOR_ERROR);
1879+
dw_printf ("GPIOD is not supported.\n");
1880+
#endif /* USE_GPIOD*/
1881+
#endif /* __WIN32__ */
18451882
}
18461883
else if (strcasecmp(t, "LPT") == 0) {
18471884

src/direwolf.c

+4-1
Original file line numberDiff line numberDiff line change
@@ -306,7 +306,7 @@ int main (int argc, char *argv[])
306306
//dw_printf ("Dire Wolf version %d.%d\n", MAJOR_VERSION, MINOR_VERSION);
307307

308308

309-
#if defined(ENABLE_GPSD) || defined(USE_HAMLIB) || defined(USE_CM108) || USE_AVAHI_CLIENT || USE_MACOS_DNSSD
309+
#if defined(ENABLE_GPSD) || defined(USE_HAMLIB) || defined(USE_CM108) || USE_AVAHI_CLIENT || USE_MACOS_DNSSD || USE_GPIOD
310310
dw_printf ("Includes optional support for: ");
311311
#if defined(ENABLE_GPSD)
312312
dw_printf (" gpsd");
@@ -317,6 +317,9 @@ int main (int argc, char *argv[])
317317
#if defined(USE_CM108)
318318
dw_printf (" cm108-ptt");
319319
#endif
320+
#if defined(USE_GPIOD)
321+
dw_printf (" libgpiod");
322+
#endif
320323
#if (USE_AVAHI_CLIENT|USE_MACOS_DNSSD)
321324
dw_printf (" dns-sd");
322325
#endif

src/ptt.c

+67-3
Original file line numberDiff line numberDiff line change
@@ -162,6 +162,10 @@
162162
#include <hamlib/rig.h>
163163
#endif
164164

165+
#ifdef USE_GPIOD
166+
#include <gpiod.h>
167+
#endif
168+
165169
/* So we can have more common code for fd. */
166170
typedef int HANDLE;
167171
#define INVALID_HANDLE_VALUE (-1)
@@ -634,6 +638,31 @@ void export_gpio(int ch, int ot, int invert, int direction)
634638
get_access_to_gpio (gpio_value_path);
635639
}
636640

641+
#if defined(USE_GPIOD)
642+
int gpiod_probe(const char *chip_name, int line_number)
643+
{
644+
struct gpiod_chip *chip;
645+
chip = gpiod_chip_open_by_name(chip_name);
646+
if (chip == NULL) {
647+
text_color_set(DW_COLOR_ERROR);
648+
dw_printf ("Can't open GPIOD chip %s.\n", chip_name);
649+
return -1;
650+
}
651+
652+
struct gpiod_line *line;
653+
line = gpiod_chip_get_line(chip, line_number);
654+
if (line == NULL) {
655+
text_color_set(DW_COLOR_ERROR);
656+
dw_printf ("Can't get GPIOD line %d.\n", line_number);
657+
return -1;
658+
}
659+
if (ptt_debug_level >= 2) {
660+
text_color_set(DW_COLOR_DEBUG);
661+
dw_printf("GPIOD probe OK. Chip: %s line: %d\n", chip_name, line_number);
662+
}
663+
return 0;
664+
}
665+
#endif /* USE_GPIOD */
637666
#endif /* not __WIN32__ */
638667

639668

@@ -650,7 +679,8 @@ void export_gpio(int ch, int ot, int invert, int direction)
650679
* ptt_method Method for PTT signal.
651680
* PTT_METHOD_NONE - not configured. Could be using VOX.
652681
* PTT_METHOD_SERIAL - serial (com) port.
653-
* PTT_METHOD_GPIO - general purpose I/O.
682+
* PTT_METHOD_GPIO - general purpose I/O (sysfs).
683+
* PTT_METHOD_GPIOD - general purpose I/O (libgpiod).
654684
* PTT_METHOD_LPT - Parallel printer port.
655685
* PTT_METHOD_HAMLIB - HAMLib rig control.
656686
* PTT_METHOD_CM108 - GPIO pins of CM108 etc. USB Audio.
@@ -729,12 +759,13 @@ void ptt_init (struct audio_s *audio_config_p)
729759
if (ptt_debug_level >= 2) {
730760

731761
text_color_set(DW_COLOR_DEBUG);
732-
dw_printf ("ch=%d, %s method=%d, device=%s, line=%d, gpio=%d, lpt_bit=%d, invert=%d\n",
762+
dw_printf ("ch=%d, %s method=%d, device=%s, line=%d, name=%s, gpio=%d, lpt_bit=%d, invert=%d\n",
733763
ch,
734764
otnames[ot],
735765
audio_config_p->achan[ch].octrl[ot].ptt_method,
736766
audio_config_p->achan[ch].octrl[ot].ptt_device,
737767
audio_config_p->achan[ch].octrl[ot].ptt_line,
768+
audio_config_p->achan[ch].octrl[ot].out_gpio_name,
738769
audio_config_p->achan[ch].octrl[ot].out_gpio_num,
739770
audio_config_p->achan[ch].octrl[ot].ptt_lpt_bit,
740771
audio_config_p->achan[ch].octrl[ot].ptt_invert);
@@ -880,7 +911,28 @@ void ptt_init (struct audio_s *audio_config_p)
880911
if (using_gpio) {
881912
get_access_to_gpio ("/sys/class/gpio/export");
882913
}
883-
914+
#if defined(USE_GPIOD)
915+
// GPIOD
916+
for (ch = 0; ch < MAX_CHANS; ch++) {
917+
if (save_audio_config_p->achan[ch].medium == MEDIUM_RADIO) {
918+
for (int ot = 0; ot < NUM_OCTYPES; ot++) {
919+
if (audio_config_p->achan[ch].octrl[ot].ptt_method == PTT_METHOD_GPIOD) {
920+
const char *chip_name = audio_config_p->achan[ch].octrl[ot].out_gpio_name;
921+
int line_number = audio_config_p->achan[ch].octrl[ot].out_gpio_num;
922+
int rc = gpiod_probe(chip_name, line_number);
923+
if (rc < 0) {
924+
text_color_set(DW_COLOR_ERROR);
925+
dw_printf ("Disable PTT for channel %d\n", ch);
926+
audio_config_p->achan[ch].octrl[ot].ptt_method = PTT_METHOD_NONE;
927+
} else {
928+
// Set initial state off ptt_set will invert output signal if appropriate.
929+
ptt_set (ot, ch, 0);
930+
}
931+
}
932+
}
933+
}
934+
}
935+
#endif /* USE_GPIOD */
884936
/*
885937
* We should now be able to create the device nodes for
886938
* the pins we want to use.
@@ -1298,6 +1350,18 @@ void ptt_set (int ot, int chan, int ptt_signal)
12981350
close (fd);
12991351

13001352
}
1353+
1354+
#if defined(USE_GPIOD)
1355+
if (save_audio_config_p->achan[chan].octrl[ot].ptt_method == PTT_METHOD_GPIOD) {
1356+
const char *chip = save_audio_config_p->achan[chan].octrl[ot].out_gpio_name;
1357+
int line = save_audio_config_p->achan[chan].octrl[ot].out_gpio_num;
1358+
int rc = gpiod_ctxless_set_value(chip, line, ptt, false, "direwolf", NULL, NULL);
1359+
if (ptt_debug_level >= 1) {
1360+
text_color_set(DW_COLOR_DEBUG);
1361+
dw_printf("PTT_METHOD_GPIOD chip: %s line: %d ptt: %d rc: %d\n", chip, line, ptt, rc);
1362+
}
1363+
}
1364+
#endif /* USE_GPIOD */
13011365
#endif
13021366

13031367
/*

0 commit comments

Comments
 (0)