@@ -162,6 +162,8 @@ typedef int HANDLE;
162
162
#endif
163
163
164
164
165
+ static struct audio_s * save_audio_config_p ; /* Save config information for later use. */
166
+
165
167
static int ptt_debug_level = 0 ;
166
168
167
169
void ptt_set_debug (int debug )
@@ -368,45 +370,45 @@ static void get_access_to_gpio (const char *path)
368
370
* Purpose: Tell the GPIO subsystem to export a GPIO line for
369
371
* us to use, and set the initial state of the GPIO.
370
372
*
371
- * Inputs: gpio - GPIO line number to export.
373
+ * Inputs: ch - Radio Channel.
374
+ * ot - Output type.
372
375
* invert: - Is the GPIO active low?
373
376
* direction: - 0 for input, 1 for output
374
377
*
375
- * Outputs: gpio_name - See below.
378
+ * Outputs: out_gpio_name - in the audio configuration structure.
379
+ * in_gpio_name
376
380
*
377
381
*------------------------------------------------------------------*/
378
382
379
383
#ifndef __WIN32__
380
384
381
- #define MAX_GPIO_NUM 100 // 76 would handle any case I've seen.
382
- #define MAX_GPIO_NAME_LEN 16 // 12 would handle any case I've seen.
383
-
384
- // Raspberry Pi was easy. GPIO 24 has the name gpio24.
385
- // Others, such as the Cubieboard, take a little more effort.
386
- // The name might be gpio24_ph11 meaning connector H, pin 11.
387
- // When we "export" GPIO number, we will store the corresponding device name
388
- // here for future use when we want to access it.
389
- // I never saw any references to anything above gpio75 so we will take the easy
390
- // way out and simply index into a fixed size array.
391
-
392
-
393
- static char gpio_name [MAX_GPIO_NUM ][MAX_GPIO_NAME_LEN ];
394
385
395
-
396
- void export_gpio (int gpio , int invert , int direction )
386
+ void export_gpio (int ch , int ot , int invert , int direction )
397
387
{
398
388
HANDLE fd ;
399
389
const char gpio_export_path [] = "/sys/class/gpio/export" ;
400
390
char gpio_direction_path [80 ];
391
+ char gpio_value_path [80 ];
401
392
char stemp [16 ];
393
+ int gpio_num ;
394
+ char * gpio_name ;
402
395
396
+ // Raspberry Pi was easy. GPIO 24 has the name gpio24.
397
+ // Others, such as the Cubieboard, take a little more effort.
398
+ // The name might be gpio24_ph11 meaning connector H, pin 11.
399
+ // When we "export" GPIO number, we will store the corresponding
400
+ // device name for future use when we want to access it.
403
401
404
- if (gpio < 0 || gpio >= MAX_GPIO_NUM ) {
405
- text_color_set (DW_COLOR_ERROR );
406
- dw_printf ("GPIO line number %d is invalid. Must be in range of 0 to %d.\n" , gpio , MAX_GPIO_NUM - 1 );
407
- exit (1 );
402
+ if (direction ) {
403
+ gpio_num = save_audio_config_p -> achan [ch ].octrl [ot ].out_gpio_num ;
404
+ gpio_name = save_audio_config_p -> achan [ch ].octrl [ot ].out_gpio_name ;
405
+ }
406
+ else {
407
+ gpio_num = save_audio_config_p -> achan [ch ].ictrl [ot ].in_gpio_num ;
408
+ gpio_name = save_audio_config_p -> achan [ch ].ictrl [ot ].in_gpio_name ;
408
409
}
409
410
411
+
410
412
get_access_to_gpio (gpio_export_path );
411
413
412
414
fd = open (gpio_export_path , O_WRONLY );
@@ -417,7 +419,7 @@ void export_gpio(int gpio, int invert, int direction)
417
419
exit (1 );
418
420
}
419
421
420
- snprintf (stemp , sizeof (stemp ), "%d" , gpio );
422
+ snprintf (stemp , sizeof (stemp ), "%d" , gpio_num );
421
423
if (write (fd , stemp , strlen (stemp )) != strlen (stemp )) {
422
424
int e = errno ;
423
425
/* Ignore EBUSY error which seems to mean */
@@ -473,7 +475,7 @@ void export_gpio(int gpio, int invert, int direction)
473
475
text_color_set (DW_COLOR_ERROR );
474
476
dw_printf ("ERROR! Could not get directory listing for /sys/class/gpio\n" );
475
477
476
- snprintf (gpio_name [ gpio ], sizeof ( gpio_name [ gpio ]) , "gpio%d" , gpio );
478
+ snprintf (gpio_name , MAX_GPIO_NAME_LEN , "gpio%d" , gpio_num );
477
479
num_files = 0 ;
478
480
ok = 1 ;
479
481
}
@@ -491,22 +493,22 @@ void export_gpio(int gpio, int invert, int direction)
491
493
// Look for exact name gpioNN
492
494
493
495
char lookfor [16 ];
494
- snprintf (lookfor , sizeof (lookfor ), "gpio%d" , gpio );
496
+ snprintf (lookfor , sizeof (lookfor ), "gpio%d" , gpio_num );
495
497
496
498
for (i = 0 ; i < num_files && ! ok ; i ++ ) {
497
499
if (strcmp (lookfor , file_list [i ]-> d_name ) == 0 ) {
498
- strlcpy (gpio_name [ gpio ] , file_list [i ]-> d_name , sizeof ( gpio_name [ gpio ]) );
500
+ strlcpy (gpio_name , file_list [i ]-> d_name , MAX_GPIO_NAME_LEN );
499
501
ok = 1 ;
500
502
}
501
503
}
502
504
503
505
// If not found, Look for gpioNN_*
504
506
505
- snprintf (lookfor , sizeof (lookfor ), "gpio%d_" , gpio );
507
+ snprintf (lookfor , sizeof (lookfor ), "gpio%d_" , gpio_num );
506
508
507
509
for (i = 0 ; i < num_files && ! ok ; i ++ ) {
508
510
if (strncmp (lookfor , file_list [i ]-> d_name , strlen (lookfor )) == 0 ) {
509
- strlcpy (gpio_name [ gpio ] , file_list [i ]-> d_name , sizeof ( gpio_name [ gpio ]) );
511
+ strlcpy (gpio_name , file_list [i ]-> d_name , MAX_GPIO_NAME_LEN );
510
512
ok = 1 ;
511
513
}
512
514
}
@@ -526,21 +528,21 @@ void export_gpio(int gpio, int invert, int direction)
526
528
527
529
if (ptt_debug_level >= 2 ) {
528
530
text_color_set (DW_COLOR_DEBUG );
529
- dw_printf ("Path for gpio number %d is /sys/class/gpio/%s\n" , gpio , gpio_name [ gpio ] );
531
+ dw_printf ("Path for gpio number %d is /sys/class/gpio/%s\n" , gpio_num , gpio_name );
530
532
}
531
533
}
532
534
else {
533
535
534
536
text_color_set (DW_COLOR_ERROR );
535
- dw_printf ("ERROR! Could not find Path for gpio number %d.n" , gpio );
537
+ dw_printf ("ERROR! Could not find Path for gpio number %d.n" , gpio_num );
536
538
exit (1 );
537
539
}
538
540
539
541
/*
540
542
* Set output direction and initial state
541
543
*/
542
544
543
- snprintf (gpio_direction_path , sizeof (gpio_direction_path ), "/sys/class/gpio/%s/direction" , gpio_name [ gpio ] );
545
+ snprintf (gpio_direction_path , sizeof (gpio_direction_path ), "/sys/class/gpio/%s/direction" , gpio_name );
544
546
get_access_to_gpio (gpio_direction_path );
545
547
546
548
fd = open (gpio_direction_path , O_WRONLY );
@@ -572,6 +574,14 @@ void export_gpio(int gpio, int invert, int direction)
572
574
exit (1 );
573
575
}
574
576
close (fd );
577
+
578
+ /*
579
+ * Make sure that we have access to 'value'.
580
+ * Do it once here, rather than each time we want to use it.
581
+ */
582
+
583
+ snprintf (gpio_value_path , sizeof (gpio_value_path ), "/sys/class/gpio/%s/value" , gpio_name );
584
+ get_access_to_gpio (gpio_value_path );
575
585
}
576
586
577
587
#endif /* not __WIN32__ */
@@ -600,7 +610,7 @@ void export_gpio(int gpio, int invert, int direction)
600
610
*
601
611
* ptt_line RTS or DTR when using serial port.
602
612
*
603
- * ptt_gpio GPIO number. Only used for Linux.
613
+ * out_gpio_num GPIO number. Only used for Linux.
604
614
* Valid only when ptt_method is PTT_METHOD_GPIO.
605
615
*
606
616
* ptt_lpt_bit Bit number for parallel printer port.
@@ -623,7 +633,6 @@ void export_gpio(int gpio, int invert, int direction)
623
633
624
634
625
635
626
- static struct audio_s * save_audio_config_p ; /* Save config information for later use. */
627
636
628
637
static HANDLE ptt_fd [MAX_CHANS ][NUM_OCTYPES ];
629
638
/* Serial port handle or fd. */
@@ -674,7 +683,7 @@ void ptt_init (struct audio_s *audio_config_p)
674
683
audio_config_p -> achan [ch ].octrl [ot ].ptt_method ,
675
684
audio_config_p -> achan [ch ].octrl [ot ].ptt_device ,
676
685
audio_config_p -> achan [ch ].octrl [ot ].ptt_line ,
677
- audio_config_p -> achan [ch ].octrl [ot ].ptt_gpio ,
686
+ audio_config_p -> achan [ch ].octrl [ot ].out_gpio_num ,
678
687
audio_config_p -> achan [ch ].octrl [ot ].ptt_lpt_bit ,
679
688
audio_config_p -> achan [ch ].octrl [ot ].ptt_invert );
680
689
}
@@ -818,8 +827,6 @@ void ptt_init (struct audio_s *audio_config_p)
818
827
819
828
if (using_gpio ) {
820
829
get_access_to_gpio ("/sys/class/gpio/export" );
821
-
822
-
823
830
}
824
831
825
832
/*
@@ -829,15 +836,18 @@ void ptt_init (struct audio_s *audio_config_p)
829
836
830
837
for (ch = 0 ; ch < MAX_CHANS ; ch ++ ) {
831
838
if (save_audio_config_p -> achan [ch ].valid ) {
832
- int ot ;
839
+
840
+ int ot ; // output control type, PTT, DCD, CON, ...
841
+ int it ; // input control type
842
+
833
843
for (ot = 0 ; ot < NUM_OCTYPES ; ot ++ ) {
834
844
if (audio_config_p -> achan [ch ].octrl [ot ].ptt_method == PTT_METHOD_GPIO ) {
835
- export_gpio (audio_config_p -> achan [ ch ]. octrl [ ot ]. ptt_gpio , audio_config_p -> achan [ch ].octrl [ot ].ptt_invert , 1 );
845
+ export_gpio (ch , ot , audio_config_p -> achan [ch ].octrl [ot ].ptt_invert , 1 );
836
846
}
837
847
}
838
- for (ot = 0 ; ot < NUM_ICTYPES ; ot ++ ) {
839
- if (audio_config_p -> achan [ch ].ictrl [ot ].method == PTT_METHOD_GPIO ) {
840
- export_gpio (audio_config_p -> achan [ ch ]. ictrl [ ot ]. gpio , audio_config_p -> achan [ch ].ictrl [ot ].invert , 0 );
848
+ for (it = 0 ; it < NUM_ICTYPES ; it ++ ) {
849
+ if (audio_config_p -> achan [ch ].ictrl [it ].method == PTT_METHOD_GPIO ) {
850
+ export_gpio (ch , it , audio_config_p -> achan [ch ].ictrl [it ].invert , 0 );
841
851
}
842
852
}
843
853
}
@@ -1125,9 +1135,7 @@ void ptt_set (int ot, int chan, int ptt_signal)
1125
1135
char gpio_value_path [80 ];
1126
1136
char stemp [16 ];
1127
1137
1128
- snprintf (gpio_value_path , sizeof (gpio_value_path ), "/sys/class/gpio/%s/value" , gpio_name [save_audio_config_p -> achan [chan ].octrl [ot ].ptt_gpio ]);
1129
-
1130
- get_access_to_gpio (gpio_value_path );
1138
+ snprintf (gpio_value_path , sizeof (gpio_value_path ), "/sys/class/gpio/%s/value" , save_audio_config_p -> achan [chan ].octrl [ot ].out_gpio_name );
1131
1139
1132
1140
fd = open (gpio_value_path , O_WRONLY );
1133
1141
if (fd < 0 ) {
@@ -1143,7 +1151,7 @@ void ptt_set (int ot, int chan, int ptt_signal)
1143
1151
if (write (fd , stemp , 1 ) != 1 ) {
1144
1152
int e = errno ;
1145
1153
text_color_set (DW_COLOR_ERROR );
1146
- dw_printf ("Error setting GPIO %d for %s\n" , save_audio_config_p -> achan [chan ].octrl [ot ].ptt_gpio , otnames [ot ]);
1154
+ dw_printf ("Error setting GPIO %d for %s\n" , save_audio_config_p -> achan [chan ].octrl [ot ].out_gpio_num , otnames [ot ]);
1147
1155
dw_printf ("%s\n" , strerror (e ));
1148
1156
}
1149
1157
close (fd );
@@ -1246,7 +1254,7 @@ int get_input (int it, int chan)
1246
1254
int fd ;
1247
1255
char gpio_value_path [80 ];
1248
1256
1249
- snprintf (gpio_value_path , sizeof (gpio_value_path ), "/sys/class/gpio/%s/value" , gpio_name [ save_audio_config_p -> achan [chan ].ictrl [it ].gpio ] );
1257
+ snprintf (gpio_value_path , sizeof (gpio_value_path ), "/sys/class/gpio/%s/value" , save_audio_config_p -> achan [chan ].ictrl [it ].in_gpio_name );
1250
1258
1251
1259
get_access_to_gpio (gpio_value_path );
1252
1260
@@ -1263,7 +1271,7 @@ int get_input (int it, int chan)
1263
1271
if (read (fd , vtemp , 1 ) != 1 ) {
1264
1272
int e = errno ;
1265
1273
text_color_set (DW_COLOR_ERROR );
1266
- dw_printf ("Error getting GPIO %d value\n" , save_audio_config_p -> achan [chan ].ictrl [it ].gpio );
1274
+ dw_printf ("Error getting GPIO %d value\n" , save_audio_config_p -> achan [chan ].ictrl [it ].in_gpio_num );
1267
1275
dw_printf ("%s\n" , strerror (e ));
1268
1276
}
1269
1277
close (fd );
@@ -1449,9 +1457,9 @@ main ()
1449
1457
my_audio_config .adev [0 ].num_channels = 1 ;
1450
1458
my_audio_config .valid [0 ] = 1 ;
1451
1459
my_audio_config .adev [0 ].octrl [OCTYPE_PTT ].ptt_method = PTT_METHOD_GPIO ;
1452
- my_audio_config .adev [0 ].octrl [OCTYPE_PTT ].ptt_gpio = 25 ;
1460
+ my_audio_config .adev [0 ].octrl [OCTYPE_PTT ].out_gpio_num = 25 ;
1453
1461
1454
- dw_printf ("Try GPIO %d a few times...\n" , my_audio_config .ptt_gpio [0 ]);
1462
+ dw_printf ("Try GPIO %d a few times...\n" , my_audio_config .out_gpio_num [0 ]);
1455
1463
1456
1464
ptt_init (& my_audio_config );
1457
1465
0 commit comments