libx48ng/src/serial.c

516 lines
13 KiB
C
Raw Normal View History

2023-04-28 14:06:51 +02:00
#include "config.h"
2022-03-24 13:41:22 +01:00
#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/time.h>
#include <termios.h>
2022-03-24 13:41:22 +01:00
#include <unistd.h>
#include "device.h"
2022-03-24 13:41:22 +01:00
#include "hp48.h"
#include "hp48_emu.h"
#include "resources.h"
#include "x48_gui.h"
static int wire_fd;
static int ir_fd;
static int ttyp;
extern int rece_instr;
2023-04-27 12:15:59 +02:00
static char* wire_name = ( char* )0;
static char* ir_name = ( char* )0;
void update_connection_display( void ) {
2023-04-27 12:15:59 +02:00
if ( wire_fd == -1 ) {
if ( wire_name )
free( wire_name );
wire_name = ( char* )0;
2022-03-24 13:41:22 +01:00
}
2023-04-27 12:15:59 +02:00
if ( ir_fd == -1 ) {
if ( ir_name )
free( ir_name );
ir_name = ( char* )0;
}
2023-04-27 12:15:59 +02:00
ShowConnections( wire_name, ir_name );
}
int serial_init( void ) {
2023-04-27 12:15:59 +02:00
int c;
int n;
char tty_dev_name[ 128 ];
struct termios ttybuf;
wire_fd = -1;
ttyp = -1;
if ( useTerminal ) {
/* Unix98 PTY (Preferred) */
if ( ( wire_fd = open( "/dev/ptmx", O_RDWR | O_NONBLOCK, 0666 ) ) >=
0 ) {
grantpt( wire_fd );
unlockpt( wire_fd );
if ( ptsname_r( wire_fd, tty_dev_name, 128 ) ) {
perror( "Could not get the name of the wire device." );
exit( -1 );
}
if ( ( ttyp = open( tty_dev_name, O_RDWR | O_NDELAY, 0666 ) ) >=
0 ) {
if ( verbose )
printf( "%s: wire connection on %s\n", progname,
tty_dev_name );
wire_name = strdup( tty_dev_name );
}
}
2023-04-27 12:15:59 +02:00
/* BSD PTY (Legacy) */
else {
c = 'p';
do {
for ( n = 0; n < 16; n++ ) {
sprintf( tty_dev_name, "/dev/pty%c%x", c, n );
if ( ( wire_fd =
open( tty_dev_name, O_RDWR | O_EXCL | O_NDELAY,
0666 ) ) >= 0 ) {
ttyp = wire_fd;
sprintf( tty_dev_name, "/dev/tty%c%x", c, n );
if ( verbose )
printf( "%s: wire connection on %s\n", progname,
tty_dev_name );
wire_name = strdup( tty_dev_name );
break;
}
}
c++;
} while ( ( wire_fd < 0 ) && ( errno != ENOENT ) );
}
}
2023-04-27 12:15:59 +02:00
if ( ttyp >= 0 ) {
#if defined( TCSANOW )
if ( tcgetattr( ttyp, &ttybuf ) < 0 )
#else
2023-04-27 12:15:59 +02:00
if ( ioctl( ttyp, TCGETS, ( char* )&ttybuf ) < 0 )
#endif
2023-04-27 12:15:59 +02:00
{
if ( !quiet )
fprintf( stderr, "%s: ioctl(wire, TCGETS) failed, errno = %d\n",
progname, errno );
wire_fd = -1;
ttyp = -1;
}
}
2023-04-27 12:15:59 +02:00
ttybuf.c_lflag = 0;
ttybuf.c_iflag = 0;
ttybuf.c_oflag = 0;
ttybuf.c_cflag = B9600 | CS8 | CLOCAL | CREAD;
for ( n = 0; n < NCCS; n++ )
ttybuf.c_cc[ n ] = 0;
ttybuf.c_cc[ VTIME ] = 0;
ttybuf.c_cc[ VMIN ] = 1;
if ( ttyp >= 0 ) {
#if defined( TCSANOW )
if ( tcsetattr( ttyp, TCSANOW, &ttybuf ) < 0 )
#else
2023-04-27 12:15:59 +02:00
if ( ioctl( ttyp, TCSETS, ( char* )&ttybuf ) < 0 )
#endif
2023-04-27 12:15:59 +02:00
{
if ( !quiet )
fprintf( stderr, "%s: ioctl(wire, TCSETS) failed, errno = %d\n",
progname, errno );
wire_fd = -1;
ttyp = -1;
}
}
2023-04-27 12:15:59 +02:00
ir_fd = -1;
if ( useSerial ) {
sprintf( tty_dev_name, serialLine );
if ( ( ir_fd = open( tty_dev_name, O_RDWR | O_NDELAY ) ) >= 0 ) {
if ( verbose )
printf( "%s: IR connection on %s\n", progname, tty_dev_name );
ir_name = strdup( tty_dev_name );
}
}
2023-04-27 12:15:59 +02:00
if ( ir_fd >= 0 ) {
#if defined( TCSANOW )
if ( tcgetattr( ir_fd, &ttybuf ) < 0 )
#else
2023-04-27 12:15:59 +02:00
if ( ioctl( ir_fd, TCGETS, ( char* )&ttybuf ) < 0 )
#endif
2023-04-27 12:15:59 +02:00
{
if ( !quiet )
fprintf( stderr, "%s: ioctl(IR, TCGETS) failed, errno = %d\n",
progname, errno );
ir_fd = -1;
}
}
2023-04-27 12:15:59 +02:00
ttybuf.c_lflag = 0;
ttybuf.c_iflag = 0;
ttybuf.c_oflag = 0;
ttybuf.c_cflag = B9600 | CS8 | CLOCAL | CREAD;
for ( n = 0; n < NCCS; n++ )
ttybuf.c_cc[ n ] = 0;
ttybuf.c_cc[ VTIME ] = 0;
ttybuf.c_cc[ VMIN ] = 1;
if ( ir_fd >= 0 ) {
#if defined( TCSANOW )
if ( tcsetattr( ir_fd, TCSANOW, &ttybuf ) < 0 )
#else
2023-04-27 12:15:59 +02:00
if ( ioctl( ir_fd, TCSETS, ( char* )&ttybuf ) < 0 )
#endif
2023-04-27 12:15:59 +02:00
{
if ( !quiet )
fprintf( stderr, "%s: ioctl(IR, TCSETS) failed, errno = %d\n",
progname, errno );
ir_fd = -1;
}
}
2023-04-27 12:15:59 +02:00
update_connection_display();
return 1;
}
2023-04-27 12:15:59 +02:00
void serial_baud( int baud ) {
int error = 0;
struct termios ttybuf;
2023-04-27 12:15:59 +02:00
if ( ir_fd >= 0 ) {
#if defined( TCSANOW )
if ( tcgetattr( ir_fd, &ttybuf ) < 0 )
#else
2023-04-27 12:15:59 +02:00
if ( ioctl( ir_fd, TCGETS, ( char* )&ttybuf ) < 0 )
#endif
2023-04-27 12:15:59 +02:00
{
if ( !quiet )
fprintf( stderr, "%s: ioctl(IR, TCGETS) failed, errno = %d\n",
progname, errno );
ir_fd = -1;
error = 1;
}
}
2023-04-27 12:15:59 +02:00
#if defined( __APPLE__ )
baud &= 0x7;
switch ( baud ) {
case 0: /* 1200 */
ttybuf.c_cflag |= B1200;
break;
case 1: /* 1920 */
2022-03-24 13:41:22 +01:00
#ifdef B1920
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag |= B1920;
2022-03-24 13:41:22 +01:00
#endif
2023-04-27 12:15:59 +02:00
break;
case 2: /* 2400 */
ttybuf.c_cflag |= B2400;
break;
case 3: /* 3840 */
2022-03-24 13:41:22 +01:00
#ifdef B3840
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag |= B3840;
2022-03-24 13:41:22 +01:00
#endif
2023-04-27 12:15:59 +02:00
break;
case 4: /* 4800 */
ttybuf.c_cflag |= B4800;
break;
case 5: /* 7680 */
2022-03-24 13:41:22 +01:00
#ifdef B7680
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag |= B7680;
2022-03-24 13:41:22 +01:00
#endif
2023-04-27 12:15:59 +02:00
break;
case 6: /* 9600 */
ttybuf.c_cflag |= B9600;
break;
case 7: /* 15360 */
2022-03-24 13:41:22 +01:00
#ifdef B15360
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag |= B15360;
2022-03-24 13:41:22 +01:00
#endif
2023-04-27 12:15:59 +02:00
break;
}
if ( ( ir_fd >= 0 ) && ( ( ttybuf.c_ospeed ) == 0 ) ) {
if ( !quiet )
fprintf( stderr, "%s: can\'t set baud rate, using 9600\n",
progname );
ttybuf.c_cflag |= B9600;
}
#else
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag &= ~CBAUD;
baud &= 0x7;
switch ( baud ) {
case 0: /* 1200 */
ttybuf.c_cflag |= B1200;
break;
case 1: /* 1920 */
2022-03-24 13:41:22 +01:00
#ifdef B1920
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag |= B1920;
2022-03-24 13:41:22 +01:00
#endif
2023-04-27 12:15:59 +02:00
break;
case 2: /* 2400 */
ttybuf.c_cflag |= B2400;
break;
case 3: /* 3840 */
2022-03-24 13:41:22 +01:00
#ifdef B3840
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag |= B3840;
2022-03-24 13:41:22 +01:00
#endif
2023-04-27 12:15:59 +02:00
break;
case 4: /* 4800 */
ttybuf.c_cflag |= B4800;
break;
case 5: /* 7680 */
2022-03-24 13:41:22 +01:00
#ifdef B7680
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag |= B7680;
2022-03-24 13:41:22 +01:00
#endif
2023-04-27 12:15:59 +02:00
break;
case 6: /* 9600 */
ttybuf.c_cflag |= B9600;
break;
case 7: /* 15360 */
2022-03-24 13:41:22 +01:00
#ifdef B15360
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag |= B15360;
2022-03-24 13:41:22 +01:00
#endif
2023-04-27 12:15:59 +02:00
break;
}
if ( ( ir_fd >= 0 ) && ( ( ttybuf.c_cflag & CBAUD ) == 0 ) ) {
if ( !quiet )
fprintf( stderr, "%s: can\'t set baud rate, using 9600\n",
progname );
ttybuf.c_cflag |= B9600;
}
#endif
2023-04-27 12:15:59 +02:00
if ( ir_fd >= 0 ) {
#if defined( TCSANOW )
if ( tcsetattr( ir_fd, TCSANOW, &ttybuf ) < 0 )
#else
2023-04-27 12:15:59 +02:00
if ( ioctl( ir_fd, TCSETS, ( char* )&ttybuf ) < 0 )
#endif
2023-04-27 12:15:59 +02:00
{
if ( !quiet )
fprintf( stderr, "%s: ioctl(IR, TCSETS) failed, errno = %d\n",
progname, errno );
ir_fd = -1;
error = 1;
}
}
2023-04-27 12:15:59 +02:00
if ( ttyp >= 0 ) {
#if defined( TCSANOW )
if ( tcgetattr( ttyp, &ttybuf ) < 0 )
#else
2023-04-27 12:15:59 +02:00
if ( ioctl( ttyp, TCGETS, ( char* )&ttybuf ) < 0 )
#endif
2023-04-27 12:15:59 +02:00
{
if ( !quiet )
fprintf( stderr, "%s: ioctl(wire, TCGETS) failed, errno = %d\n",
progname, errno );
wire_fd = -1;
ttyp = -1;
error = 1;
}
}
2023-04-27 12:15:59 +02:00
#if defined( __APPLE__ )
#else
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag &= ~CBAUD;
baud &= 0x7;
switch ( baud ) {
case 0: /* 1200 */
ttybuf.c_cflag |= B1200;
break;
case 1: /* 1920 */
2022-03-24 13:41:22 +01:00
#ifdef B1920
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag |= B1920;
2022-03-24 13:41:22 +01:00
#endif
2023-04-27 12:15:59 +02:00
break;
case 2: /* 2400 */
ttybuf.c_cflag |= B2400;
break;
case 3: /* 3840 */
2022-03-24 13:41:22 +01:00
#ifdef B3840
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag |= B3840;
2022-03-24 13:41:22 +01:00
#endif
2023-04-27 12:15:59 +02:00
break;
case 4: /* 4800 */
ttybuf.c_cflag |= B4800;
break;
case 5: /* 7680 */
2022-03-24 13:41:22 +01:00
#ifdef B7680
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag |= B7680;
2022-03-24 13:41:22 +01:00
#endif
2023-04-27 12:15:59 +02:00
break;
case 6: /* 9600 */
ttybuf.c_cflag |= B9600;
break;
case 7: /* 15360 */
2022-03-24 13:41:22 +01:00
#ifdef B15360
2023-04-27 12:15:59 +02:00
ttybuf.c_cflag |= B15360;
2022-03-24 13:41:22 +01:00
#endif
2023-04-27 12:15:59 +02:00
break;
}
if ( ( ttyp >= 0 ) && ( ( ttybuf.c_cflag & CBAUD ) == 0 ) ) {
if ( !quiet )
fprintf( stderr, "%s: can\'t set baud rate, using 9600\n",
progname );
ttybuf.c_cflag |= B9600;
}
#endif
2023-04-27 12:15:59 +02:00
if ( ttyp >= 0 ) {
#if defined( TCSANOW )
if ( tcsetattr( ttyp, TCSANOW, &ttybuf ) < 0 )
#else
2023-04-27 12:15:59 +02:00
if ( ioctl( ttyp, TCSETS, ( char* )&ttybuf ) < 0 )
#endif
2023-04-27 12:15:59 +02:00
{
if ( !quiet )
fprintf( stderr, "%s: ioctl(wire, TCSETS) failed, errno = %d\n",
progname, errno );
wire_fd = -1;
ttyp = -1;
error = 1;
}
}
2023-04-27 12:15:59 +02:00
if ( error )
update_connection_display();
}
void transmit_char( void ) {
2023-04-27 12:15:59 +02:00
if ( saturn.ir_ctrl & 0x04 ) {
if ( ir_fd == -1 ) {
saturn.tcs &= 0x0e;
if ( saturn.io_ctrl & 0x04 ) {
do_interupt();
}
return;
}
} else {
if ( wire_fd == -1 ) {
saturn.tcs &= 0x0e;
if ( saturn.io_ctrl & 0x04 ) {
do_interupt();
}
return;
}
}
2023-04-27 12:15:59 +02:00
if ( saturn.ir_ctrl & 0x04 ) {
if ( write( ir_fd, &saturn.tbr, 1 ) == 1 ) {
saturn.tcs &= 0x0e;
if ( saturn.io_ctrl & 0x04 ) {
do_interupt();
}
} else {
if ( errno != EAGAIN ) {
fprintf( stderr, "%s: serial write error: %d\n", progname,
errno );
}
saturn.tcs &= 0x0e;
if ( saturn.io_ctrl & 0x04 ) {
do_interupt();
}
}
} else {
2023-04-27 12:15:59 +02:00
if ( write( wire_fd, &saturn.tbr, 1 ) == 1 ) {
saturn.tcs &= 0x0e;
if ( saturn.io_ctrl & 0x04 ) {
do_interupt();
}
} else {
if ( errno != EAGAIN ) {
if ( !quiet )
fprintf( stderr, "%s: serial write error: %d\n", progname,
errno );
}
saturn.tcs &= 0x0e;
if ( saturn.io_ctrl & 0x04 ) {
do_interupt();
}
}
}
}
#define NR_BUFFER 256
void receive_char( void ) {
2023-04-27 12:15:59 +02:00
struct timeval tout;
fd_set rfds;
int nfd;
static unsigned char buf[ NR_BUFFER + 1 ];
static int nrd = 0, bp = 0;
2023-04-27 12:15:59 +02:00
rece_instr = 0;
if ( saturn.ir_ctrl & 0x04 ) {
if ( ir_fd == -1 )
return;
} else {
2023-04-27 12:15:59 +02:00
if ( wire_fd == -1 )
return;
}
2023-04-27 12:15:59 +02:00
if ( saturn.rcs & 0x01 ) {
return;
}
if ( nrd == 0 ) {
tout.tv_sec = 0;
tout.tv_usec = 0;
FD_ZERO( &rfds );
if ( saturn.ir_ctrl & 0x04 ) {
FD_SET( ir_fd, &rfds );
nfd = ir_fd + 1;
} else {
FD_SET( wire_fd, &rfds );
nfd = wire_fd + 1;
}
if ( ( nfd = select( nfd, &rfds, ( fd_set* )0, ( fd_set* )0, &tout ) ) >
0 ) {
if ( saturn.ir_ctrl & 0x04 ) {
if ( FD_ISSET( ir_fd, &rfds ) ) {
nrd = read( ir_fd, buf, NR_BUFFER );
if ( nrd < 0 ) {
nrd = 0;
return;
}
bp = 0;
} else {
return;
}
} else {
if ( FD_ISSET( wire_fd, &rfds ) ) {
nrd = read( wire_fd, buf, NR_BUFFER );
if ( nrd < 0 ) {
nrd = 0;
return;
}
bp = 0;
} else {
return;
}
}
} else {
return;
}
}
2023-04-27 12:15:59 +02:00
if ( nrd == 0 ) {
return;
}
if ( !( saturn.io_ctrl & 0x08 ) ) {
nrd = 0;
return;
}
saturn.rbr = buf[ bp++ ];
nrd--;
saturn.rcs |= 0x01;
if ( saturn.io_ctrl & 0x02 ) {
do_interupt();
}
}