1999-04-16 03:35:26 +02:00
|
|
|
/* communicate.c -- ARMulator RDP comms code: ARM6 Instruction Emulator.
|
|
|
|
Copyright (C) 1994 Advanced RISC Machines Ltd.
|
|
|
|
|
|
|
|
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 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
|
2005-05-12 09:36:59 +02:00
|
|
|
Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA. */
|
1999-04-16 03:35:26 +02:00
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
/* Functions to read and write characters or groups of characters */
|
|
|
|
/* down sockets or pipes. Those that return a value return -1 on failure */
|
|
|
|
/* and 0 on success. */
|
|
|
|
/**************************************************************************/
|
|
|
|
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
|
|
|
|
#include "armdefs.h"
|
|
|
|
|
|
|
|
/* The socket to the debugger */
|
|
|
|
int debugsock;
|
|
|
|
|
|
|
|
/* The maximum number of file descriptors */
|
|
|
|
extern int nfds;
|
|
|
|
|
|
|
|
/* The socket handle */
|
|
|
|
extern int sockethandle;
|
|
|
|
|
|
|
|
/* Read and Write routines down a pipe or socket */
|
|
|
|
|
|
|
|
/****************************************************************/
|
|
|
|
/* Read an individual character. */
|
|
|
|
/* All other read functions rely on this one. */
|
|
|
|
/* It waits 15 seconds until there is a character available: if */
|
|
|
|
/* no character is available, then it timeouts and returns -1. */
|
|
|
|
/****************************************************************/
|
2000-02-05 08:30:26 +01:00
|
|
|
int
|
|
|
|
MYread_char (int sock, unsigned char *c)
|
|
|
|
{
|
1999-04-16 03:35:26 +02:00
|
|
|
int i;
|
|
|
|
fd_set readfds;
|
2000-02-05 08:30:26 +01:00
|
|
|
struct timeval timeout = { 15, 0 };
|
1999-04-16 03:35:26 +02:00
|
|
|
struct sockaddr_in isa;
|
|
|
|
|
2000-02-05 08:30:26 +01:00
|
|
|
retry:
|
|
|
|
|
|
|
|
FD_ZERO (&readfds);
|
|
|
|
FD_SET (sock, &readfds);
|
|
|
|
|
|
|
|
i = select (nfds, &readfds, (fd_set *) 0, (fd_set *) 0, &timeout);
|
|
|
|
|
|
|
|
if (i < 0)
|
|
|
|
{
|
|
|
|
perror ("select");
|
|
|
|
exit (1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!i)
|
|
|
|
{
|
|
|
|
fprintf (stderr, "read: Timeout\n");
|
1999-04-16 03:35:26 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2000-02-05 08:30:26 +01:00
|
|
|
|
|
|
|
if ((i = read (sock, c, 1)) < 1)
|
|
|
|
{
|
|
|
|
if (!i && sock == debugsock)
|
|
|
|
{
|
|
|
|
fprintf (stderr, "Connection with debugger severed.\n");
|
|
|
|
/* This shouldn't be necessary for a detached armulator, but
|
|
|
|
the armulator cannot be cold started a second time, so
|
|
|
|
this is probably preferable to locking up. */
|
|
|
|
return -1;
|
|
|
|
fprintf (stderr, "Waiting for connection from debugger...");
|
|
|
|
debugsock = accept (sockethandle, &isa, &i);
|
2010-05-27 00:40:24 +02:00
|
|
|
if (debugsock == -1)
|
2000-02-05 08:30:26 +01:00
|
|
|
{ /* Now we are in serious trouble... */
|
|
|
|
perror ("accept");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
fprintf (stderr, " done.\nConnection Established.\n");
|
|
|
|
sock = debugsock;
|
|
|
|
goto retry;
|
|
|
|
}
|
|
|
|
perror ("read");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
1999-04-16 03:35:26 +02:00
|
|
|
#ifdef DEBUG
|
2000-02-05 08:30:26 +01:00
|
|
|
if (sock == debugsock)
|
|
|
|
fprintf (stderr, "<%02x ", *c);
|
1999-04-16 03:35:26 +02:00
|
|
|
#endif
|
2000-02-05 08:30:26 +01:00
|
|
|
|
1999-04-16 03:35:26 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************/
|
|
|
|
/* Read an individual character. */
|
|
|
|
/* It waits until there is a character available. Returns -1 if */
|
|
|
|
/* an error occurs. */
|
|
|
|
/****************************************************************/
|
2000-02-05 08:30:26 +01:00
|
|
|
int
|
|
|
|
MYread_charwait (int sock, unsigned char *c)
|
|
|
|
{
|
1999-04-16 03:35:26 +02:00
|
|
|
int i;
|
|
|
|
fd_set readfds;
|
|
|
|
struct sockaddr_in isa;
|
|
|
|
|
2000-02-05 08:30:26 +01:00
|
|
|
retry:
|
|
|
|
|
|
|
|
FD_ZERO (&readfds);
|
|
|
|
FD_SET (sock, &readfds);
|
|
|
|
|
|
|
|
i = select (nfds, &readfds,
|
|
|
|
(fd_set *) 0, (fd_set *) 0, (struct timeval *) 0);
|
|
|
|
|
|
|
|
if (i < 0)
|
|
|
|
{
|
|
|
|
perror ("select");
|
|
|
|
exit (-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((i = read (sock, c, 1)) < 1)
|
|
|
|
{
|
|
|
|
if (!i && sock == debugsock)
|
|
|
|
{
|
|
|
|
fprintf (stderr, "Connection with debugger severed.\n");
|
|
|
|
return -1;
|
|
|
|
fprintf (stderr, "Waiting for connection from debugger...");
|
|
|
|
debugsock = accept (sockethandle, &isa, &i);
|
2010-05-27 00:40:24 +02:00
|
|
|
if (debugsock == -1)
|
2000-02-05 08:30:26 +01:00
|
|
|
{ /* Now we are in serious trouble... */
|
|
|
|
perror ("accept");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
fprintf (stderr, " done.\nConnection Established.\n");
|
|
|
|
sock = debugsock;
|
|
|
|
goto retry;
|
|
|
|
}
|
|
|
|
perror ("read");
|
1999-04-16 03:35:26 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2000-02-05 08:30:26 +01:00
|
|
|
|
1999-04-16 03:35:26 +02:00
|
|
|
#ifdef DEBUG
|
2000-02-05 08:30:26 +01:00
|
|
|
if (sock == debugsock)
|
|
|
|
fprintf (stderr, "<%02x ", *c);
|
1999-04-16 03:35:26 +02:00
|
|
|
#endif
|
2000-02-05 08:30:26 +01:00
|
|
|
|
1999-04-16 03:35:26 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2000-02-05 08:30:26 +01:00
|
|
|
void
|
|
|
|
MYwrite_char (int sock, unsigned char c)
|
|
|
|
{
|
1999-04-16 03:35:26 +02:00
|
|
|
|
2000-02-05 08:30:26 +01:00
|
|
|
if (write (sock, &c, 1) < 1)
|
|
|
|
perror ("write");
|
1999-04-16 03:35:26 +02:00
|
|
|
#ifdef DEBUG
|
2000-02-05 08:30:26 +01:00
|
|
|
if (sock == debugsock)
|
|
|
|
fprintf (stderr, ">%02x ", c);
|
1999-04-16 03:35:26 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2000-02-05 08:30:26 +01:00
|
|
|
int
|
|
|
|
MYread_word (int sock, ARMword * here)
|
|
|
|
{
|
1999-04-16 03:35:26 +02:00
|
|
|
unsigned char a, b, c, d;
|
2000-02-05 08:30:26 +01:00
|
|
|
|
|
|
|
if (MYread_char (sock, &a) < 0)
|
|
|
|
return -1;
|
|
|
|
if (MYread_char (sock, &b) < 0)
|
|
|
|
return -1;
|
|
|
|
if (MYread_char (sock, &c) < 0)
|
|
|
|
return -1;
|
|
|
|
if (MYread_char (sock, &d) < 0)
|
|
|
|
return -1;
|
1999-04-16 03:35:26 +02:00
|
|
|
*here = a | b << 8 | c << 16 | d << 24;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2000-02-05 08:30:26 +01:00
|
|
|
void
|
|
|
|
MYwrite_word (int sock, ARMword i)
|
|
|
|
{
|
|
|
|
MYwrite_char (sock, i & 0xff);
|
|
|
|
MYwrite_char (sock, (i & 0xff00) >> 8);
|
|
|
|
MYwrite_char (sock, (i & 0xff0000) >> 16);
|
|
|
|
MYwrite_char (sock, (i & 0xff000000) >> 24);
|
1999-04-16 03:35:26 +02:00
|
|
|
}
|
|
|
|
|
2000-02-05 08:30:26 +01:00
|
|
|
void
|
|
|
|
MYwrite_string (int sock, char *s)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; MYwrite_char (sock, s[i]), s[i]; i++);
|
1999-04-16 03:35:26 +02:00
|
|
|
}
|
|
|
|
|
2000-02-05 08:30:26 +01:00
|
|
|
int
|
|
|
|
MYread_FPword (int sock, char *putinhere)
|
|
|
|
{
|
1999-04-16 03:35:26 +02:00
|
|
|
int i;
|
|
|
|
for (i = 0; i < 16; i++)
|
2000-02-05 08:30:26 +01:00
|
|
|
if (MYread_char (sock, &putinhere[i]) < 0)
|
|
|
|
return -1;
|
1999-04-16 03:35:26 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2000-02-05 08:30:26 +01:00
|
|
|
void
|
|
|
|
MYwrite_FPword (int sock, char *fromhere)
|
|
|
|
{
|
1999-04-16 03:35:26 +02:00
|
|
|
int i;
|
|
|
|
for (i = 0; i < 16; i++)
|
2000-02-05 08:30:26 +01:00
|
|
|
MYwrite_char (sock, fromhere[i]);
|
1999-04-16 03:35:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Takes n bytes from source and those n bytes */
|
|
|
|
/* down to dest */
|
2000-02-05 08:30:26 +01:00
|
|
|
int
|
|
|
|
passon (int source, int dest, int n)
|
|
|
|
{
|
1999-04-16 03:35:26 +02:00
|
|
|
char *p;
|
|
|
|
int i;
|
2000-02-05 08:30:26 +01:00
|
|
|
|
|
|
|
p = (char *) malloc (n);
|
|
|
|
if (!p)
|
|
|
|
{
|
|
|
|
perror ("Out of memory\n");
|
|
|
|
exit (1);
|
|
|
|
}
|
|
|
|
if (n)
|
|
|
|
{
|
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
if (MYread_char (source, &p[i]) < 0)
|
|
|
|
return -1;
|
|
|
|
|
1999-04-16 03:35:26 +02:00
|
|
|
#ifdef DEBUG
|
2000-02-05 08:30:26 +01:00
|
|
|
if (dest == debugsock)
|
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
fprintf (stderr, ")%02x ", (unsigned char) p[i]);
|
1999-04-16 03:35:26 +02:00
|
|
|
#endif
|
2000-02-05 08:30:26 +01:00
|
|
|
|
|
|
|
write (dest, p, n);
|
|
|
|
}
|
|
|
|
free (p);
|
1999-04-16 03:35:26 +02:00
|
|
|
return 0;
|
|
|
|
}
|