mirror of
				https://github.com/saitohirga/WSJT-X.git
				synced 2025-10-25 10:00:23 -04:00 
			
		
		
		
	git-svn-id: svn+ssh://svn.code.sf.net/p/wsjt/wsjt/branches/wsjtx@2738 ab8295b8-cf94-4d9e-aec4-7959e3be5d79
		
			
				
	
	
		
			394 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			394 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * WSJT is Copyright (c) 2001-2006 by Joseph H. Taylor, Jr., K1JT, 
 | |
|  * and is licensed under the GNU General Public License (GPL).
 | |
|  *
 | |
|  * Code used from cwdaemon for parallel port ptt only.
 | |
|  *
 | |
|  * cwdaemon - morse sounding daemon for the parallel or serial port
 | |
|  * Copyright (C) 2002 -2005 Joop Stakenborg <pg4i@amsat.org>
 | |
|  *                       and many authors, see the AUTHORS file.
 | |
|  *
 | |
|  * 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 Library 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 | |
|  */
 | |
| # if HAVE_STDIO_H
 | |
| # include <stdio.h>
 | |
| #endif
 | |
| #if STDC_HEADERS
 | |
| # include <stdlib.h>
 | |
| # include <stddef.h>
 | |
| #else
 | |
| # if HAVE_STDLIB_H
 | |
| #  include <stdlib.h>
 | |
| # endif
 | |
| #endif
 | |
| #if HAVE_UNISTD_H
 | |
| # include <unistd.h>
 | |
| #endif
 | |
| #if HAVE_SYS_IOCTL_H
 | |
| # include <sys/ioctl.h>
 | |
| #endif
 | |
| //#if HAVE_FCNTL_H
 | |
| # include <fcntl.h>
 | |
| //#endif
 | |
| #include <stdio.h>
 | |
| #include <sys/ioctl.h>
 | |
| 
 | |
| #ifdef HAVE_LINUX_PPDEV_H
 | |
| # include <linux/ppdev.h>
 | |
| # include <linux/parport.h>
 | |
| #endif
 | |
| #ifdef HAVE_DEV_PPBUS_PPI_H
 | |
| # include <dev/ppbus/ppi.h>
 | |
| # include <dev/ppbus/ppbconf.h>
 | |
| #endif
 | |
| 
 | |
| int lp_reset (int fd);
 | |
| int lp_ptt (int fd, int onoff);
 | |
| 
 | |
| #ifdef HAVE_SYS_STAT_H
 | |
| # include <sys/stat.h>
 | |
| #endif
 | |
| #if (defined(__unix__) || defined(unix)) && !defined(USG)
 | |
| # include <sys/param.h>
 | |
| #endif
 | |
| 
 | |
| #include <string.h>
 | |
| /* parport functions */
 | |
| 
 | |
| int dev_is_parport(int fd);
 | |
| int ptt_parallel(int fd, int *ntx, int *iptt);
 | |
| int ptt_serial(int fd, int *ntx, int *iptt);
 | |
| 
 | |
| int fd=-1;		/* Used for both serial and parallel */
 | |
| 
 | |
| /*
 | |
|  * ptt_
 | |
|  *
 | |
|  * generic unix PTT routine called from Fortran
 | |
|  *
 | |
|  * Inputs	
 | |
|  * unused	Unused, to satisfy old windows calling convention
 | |
|  * ptt_port	device name serial or parallel
 | |
|  * ntx		pointer to fortran command on or off
 | |
|  * iptt		pointer to fortran command status on or off
 | |
|  * Returns	- non 0 if error
 | |
| */
 | |
| 
 | |
| /* Tiny state machine */
 | |
| #define STATE_PORT_CLOSED		0
 | |
| #define STATE_PORT_OPEN_PARALLEL	1
 | |
| #define STATE_PORT_OPEN_SERIAL		2
 | |
| 
 | |
| //int ptt_(int *unused, char *ptt_port, int *ntx, int *iptt)
 | |
| int ptt_(int *unused, int *ntx, int *iptt)
 | |
| {
 | |
|   static int state=0;
 | |
|   char *p;
 | |
| 
 | |
| // ### Temporary:
 | |
|   char* ptt_port;
 | |
|   if(*unused != -99) {
 | |
|     *iptt=*ntx;
 | |
|     return 0;
 | |
|   }
 | |
| // ###
 | |
| 
 | |
|   /* In the very unlikely event of a NULL pointer, just return.
 | |
|    * Yes, I realise this should not be possible in WSJT.
 | |
|    */
 | |
|   if (ptt_port == NULL) {
 | |
|     *iptt = *ntx;
 | |
|     return (0);
 | |
|   }
 | |
|     
 | |
|   switch (state) {
 | |
|   case STATE_PORT_CLOSED:
 | |
| 
 | |
|      /* Remove trailing ' ' */
 | |
|     if ((p = strchr(ptt_port, ' ')) != NULL)
 | |
|       *p = '\0';
 | |
| 
 | |
|     /* If all that is left is a '\0' then also just return */
 | |
|     if (*ptt_port == '\0') {
 | |
|       *iptt = *ntx;
 | |
|       return(0);
 | |
|     }
 | |
| 
 | |
|     if ((fd = open(ptt_port, O_RDWR|O_NONBLOCK)) < 0) {
 | |
| 	fprintf(stderr, "Can't open %s.\n", ptt_port);
 | |
| 	return (1);
 | |
|     }
 | |
| 
 | |
|     if (dev_is_parport(fd)) {
 | |
|       state = STATE_PORT_OPEN_PARALLEL;
 | |
|       lp_reset(fd);
 | |
|       ptt_parallel(fd, ntx, iptt);
 | |
|     } else {
 | |
|       state = STATE_PORT_OPEN_SERIAL;
 | |
|       ptt_serial(fd, ntx, iptt);
 | |
|     }
 | |
|     break;
 | |
| 
 | |
|   case STATE_PORT_OPEN_PARALLEL:
 | |
|     ptt_parallel(fd, ntx, iptt);
 | |
|     break;
 | |
| 
 | |
|   case STATE_PORT_OPEN_SERIAL:
 | |
|     ptt_serial(fd, ntx, iptt);
 | |
|     break;
 | |
| 
 | |
|   default:
 | |
|     close(fd);
 | |
|     fd = -1;
 | |
|     state = STATE_PORT_CLOSED;
 | |
|     break;
 | |
|   }
 | |
|   return(0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * ptt_serial
 | |
|  *
 | |
|  * generic serial unix PTT routine called indirectly from Fortran
 | |
|  *
 | |
|  * fd		- already opened file descriptor
 | |
|  * ntx		- pointer to fortran command on or off
 | |
|  * iptt		- pointer to fortran command status on or off
 | |
|  */
 | |
| 
 | |
| int
 | |
| ptt_serial(int fd, int *ntx, int *iptt)
 | |
| {
 | |
|   int control = TIOCM_RTS | TIOCM_DTR;
 | |
| 
 | |
|   if(*ntx) {
 | |
|     ioctl(fd, TIOCMBIS, &control);               /* Set DTR and RTS */
 | |
|     *iptt = 1;
 | |
|   } else {
 | |
|     ioctl(fd, TIOCMBIC, &control);
 | |
|     *iptt = 0;
 | |
|   }
 | |
|   return(0);
 | |
| }
 | |
| 
 | |
| 
 | |
| /* parport functions */
 | |
| 
 | |
| /*
 | |
|  * dev_is_parport(fd):
 | |
|  *
 | |
|  * inputs	- Already open fd
 | |
|  * output	- 1 if parallel port, 0 if not
 | |
|  * side effects	- Unfortunately, this is platform specific.
 | |
|  */
 | |
| 
 | |
| #if defined(HAVE_LINUX_PPDEV_H)                /* Linux (ppdev) */
 | |
| 
 | |
| int
 | |
| dev_is_parport(int fd)
 | |
| {
 | |
|        struct stat st;
 | |
|        int m;
 | |
| 
 | |
|        if ((fstat(fd, &st) == -1) ||
 | |
| 	   ((st.st_mode & S_IFMT) != S_IFCHR) ||
 | |
| 	   (ioctl(fd, PPGETMODE, &m) == -1))
 | |
| 	 return(0);
 | |
| 
 | |
|        return(1);
 | |
| }
 | |
| 
 | |
| #elif defined(HAVE_DEV_PPBUS_PPI_H)    /* FreeBSD (ppbus/ppi) */
 | |
| 
 | |
| int
 | |
| dev_is_parport(int fd)
 | |
| {
 | |
|        struct stat st;
 | |
|        unsigned char c;
 | |
| 
 | |
|        if ((fstat(fd, &st) == -1) ||
 | |
| 	   ((st.st_mode & S_IFMT) != S_IFCHR) ||
 | |
| 	   (ioctl(fd, PPISSTATUS, &c) == -1))
 | |
| 	 return(0);
 | |
| 
 | |
|        return(1);
 | |
| }
 | |
| 
 | |
| #else                                  /* Fallback (nothing) */
 | |
| 
 | |
| int
 | |
| dev_is_parport(int fd)
 | |
| {
 | |
|        return(0);
 | |
| }
 | |
| 
 | |
| #endif
 | |
| /* Linux wrapper around PPFCONTROL */
 | |
| #ifdef HAVE_LINUX_PPDEV_H
 | |
| static void
 | |
| parport_control (int fd, unsigned char controlbits, int values)
 | |
| {
 | |
| 	struct ppdev_frob_struct frob;
 | |
| 	frob.mask = controlbits;
 | |
| 	frob.val = values;
 | |
| 
 | |
| 	if (ioctl (fd, PPFCONTROL, &frob) == -1)
 | |
| 	{
 | |
| 		fprintf(stderr, "Parallel port PPFCONTROL");
 | |
| 		exit (1);
 | |
| 	}
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /* FreeBSD wrapper around PPISCTRL */
 | |
| #ifdef HAVE_DEV_PPBUS_PPI_H
 | |
| static void
 | |
| parport_control (int fd, unsigned char controlbits, int values)
 | |
| {
 | |
| 	unsigned char val;
 | |
| 
 | |
| 	if (ioctl (fd, PPIGCTRL, &val) == -1)
 | |
| 	{
 | |
| 		fprintf(stderr, "Parallel port PPIGCTRL");
 | |
| 		exit (1);
 | |
| 	}
 | |
| 
 | |
| 	val &= ~controlbits;
 | |
| 	val |= values;
 | |
| 
 | |
| 	if (ioctl (fd, PPISCTRL, &val) == -1)
 | |
| 	{
 | |
| 		fprintf(stderr, "Parallel port PPISCTRL");
 | |
| 		exit (1);
 | |
| 	}
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /* Initialise a parallel port, given open fd */
 | |
| int
 | |
| lp_init (int fd)
 | |
| {
 | |
| #ifdef HAVE_LINUX_PPDEV_H
 | |
| 	int mode;
 | |
| #endif
 | |
| 
 | |
| #ifdef HAVE_LINUX_PPDEV_H
 | |
| 	mode = PARPORT_MODE_PCSPP;
 | |
| 
 | |
| 	if (ioctl (fd, PPSETMODE, &mode) == -1)
 | |
| 	{
 | |
| 		fprintf(stderr, "Setting parallel port mode");
 | |
| 		close (fd);
 | |
| 		return(-1);
 | |
| 	}
 | |
| 
 | |
| 	if (ioctl (fd, PPEXCL, NULL) == -1)
 | |
| 	{
 | |
| 		fprintf(stderr, "Parallel port is already in use.\n");
 | |
| 		close (fd);
 | |
| 		return(-1);
 | |
| 	}
 | |
| 	if (ioctl (fd, PPCLAIM, NULL) == -1)
 | |
| 	{
 | |
| 		fprintf(stderr, "Claiming parallel port.\n");
 | |
| 		fprintf(stderr, "HINT: did you unload the lp kernel module?");
 | |
| 		close (fd);
 | |
| 		return(-1);
 | |
| 	}
 | |
| 
 | |
| 	/* Enable CW & PTT - /STROBE bit (pin 1) */
 | |
| 	parport_control (fd, PARPORT_CONTROL_STROBE, PARPORT_CONTROL_STROBE);
 | |
| #endif
 | |
| #ifdef HAVE_DEV_PPBUS_PPI_H
 | |
| 	parport_control (fd, STROBE, STROBE);
 | |
| #endif
 | |
| 	lp_reset (fd);
 | |
| 	return(0);
 | |
| }
 | |
| 
 | |
| /* release ppdev and close port */
 | |
| int
 | |
| lp_free (int fd)
 | |
| {
 | |
| #ifdef HAVE_LINUX_PPDEV_H
 | |
| 	lp_reset (fd);
 | |
| 
 | |
| 	/* Disable CW & PTT - /STROBE bit (pin 1) */
 | |
| 	parport_control (fd, PARPORT_CONTROL_STROBE, 0);
 | |
| 
 | |
| 	ioctl (fd, PPRELEASE);
 | |
| #endif
 | |
| #ifdef HAVE_DEV_PPBUS_PPI_H
 | |
| 	/* Disable CW & PTT - /STROBE bit (pin 1) */
 | |
| 	parport_control (fd, STROBE, 0);
 | |
| #endif
 | |
| 	close (fd);
 | |
| 	return(0);
 | |
| }
 | |
| 
 | |
| /* set to a known state */
 | |
| int
 | |
| lp_reset (int fd)
 | |
| {
 | |
| #if defined (HAVE_LINUX_PPDEV_H) || defined (HAVE_DEV_PPBUS_PPI_H)
 | |
| 	lp_ptt (fd, 0);
 | |
| #endif
 | |
| 	return(0);
 | |
| }
 | |
| 
 | |
| /* SSB PTT keying - /INIT bit (pin 16) (inverted) */
 | |
| int
 | |
| lp_ptt (int fd, int onoff)
 | |
| {
 | |
| #ifdef HAVE_LINUX_PPDEV_H
 | |
| 	if (onoff == 1)
 | |
| 		parport_control (fd, PARPORT_CONTROL_INIT,
 | |
| 				PARPORT_CONTROL_INIT);
 | |
| 	else
 | |
| 		parport_control (fd, PARPORT_CONTROL_INIT, 0);
 | |
| #endif
 | |
| #ifdef HAVE_DEV_PPBUS_PPI_H
 | |
| 	if (onoff == 1)
 | |
| 		parport_control (fd, nINIT,
 | |
| 				nINIT);
 | |
| 	else
 | |
| 		parport_control (fd, nINIT, 0);
 | |
| #endif
 | |
| 	return(0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * ptt_parallel
 | |
|  *
 | |
|  * generic parallel unix PTT routine called indirectly from Fortran
 | |
|  *
 | |
|  * fd		- already opened file descriptor
 | |
|  * ntx		- pointer to fortran command on or off
 | |
|  * iptt		- pointer to fortran command status on or off
 | |
|  */
 | |
| 
 | |
| int
 | |
| ptt_parallel(int fd, int *ntx, int *iptt)
 | |
| {
 | |
|   if(*ntx) {
 | |
|     lp_ptt(fd, 1);
 | |
|     *iptt=1;
 | |
|   }  else {
 | |
|     lp_ptt(fd, 0);
 | |
|     *iptt=0;
 | |
|   }
 | |
|   return(0);
 | |
| }
 |