2012-05-05 01:08:27 -04:00
|
|
|
#include "marignotti.h"
|
|
|
|
#include <gno/kerntool.h>
|
|
|
|
#include <errno.h>
|
2012-05-12 19:48:06 -04:00
|
|
|
#include <sys/socket.h>
|
|
|
|
|
2012-05-05 01:08:27 -04:00
|
|
|
#include <misctool.h>
|
2012-05-11 21:21:29 -04:00
|
|
|
|
2012-05-05 02:20:05 -04:00
|
|
|
#include "s16debug.h"
|
2012-05-05 01:08:27 -04:00
|
|
|
|
|
|
|
#pragma noroot
|
|
|
|
#pragma optimize 79
|
|
|
|
|
2012-05-11 21:21:29 -04:00
|
|
|
union split {
|
|
|
|
|
|
|
|
LongWord i32;
|
|
|
|
Word i16[2];
|
|
|
|
Byte i8[4];
|
|
|
|
};
|
|
|
|
|
2012-05-05 01:08:27 -04:00
|
|
|
int mconnect(Entry *e, void *p1, void *p2, void *p3, void *p4, void *p5)
|
|
|
|
{
|
|
|
|
Word port;
|
|
|
|
Word t;
|
|
|
|
Word terr;
|
|
|
|
int xerrno;
|
|
|
|
LongWord timeout;
|
|
|
|
|
|
|
|
// todo -- if non-blocking,
|
|
|
|
// return EINPROGRESS
|
|
|
|
//
|
|
|
|
|
|
|
|
xsockaddr_in *sin = (xsockaddr_in *)p3;
|
|
|
|
int addrlen = *(int *)p4;
|
|
|
|
|
|
|
|
port = sin->sin_port;
|
|
|
|
asm {
|
|
|
|
lda <port
|
|
|
|
xba
|
|
|
|
sta <port
|
|
|
|
}
|
|
|
|
|
2012-05-11 21:21:29 -04:00
|
|
|
if (Debug > 0)
|
|
|
|
{
|
|
|
|
union split s;
|
|
|
|
s.i32 = sin->sin_addr;
|
|
|
|
s16_debug_printf("connect address = %d.%d.%d.%d port = %d",
|
|
|
|
s.i8[0], s.i8[1], s.i8[2], s.i8[3], port);
|
|
|
|
}
|
|
|
|
|
2012-05-12 19:48:06 -04:00
|
|
|
if (e->_TYPE == SOCK_DGRAM)
|
|
|
|
{
|
|
|
|
IncBusy();
|
|
|
|
TCPIPSetNewDestination(e->ipid, sin->sin_addr, port);
|
|
|
|
t = _toolErr;
|
|
|
|
DecBusy();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-05-05 01:08:27 -04:00
|
|
|
// check if already connected.
|
|
|
|
IncBusy();
|
|
|
|
terr = TCPIPStatusTCP(e->ipid, &e->sr);
|
|
|
|
t = _toolErr;
|
|
|
|
if (t) terr = t;
|
|
|
|
DecBusy();
|
|
|
|
|
|
|
|
// todo -- if non-blocking,
|
|
|
|
// return EINPROGRESS first time,
|
|
|
|
// return EALREADY on subsequent calls.
|
|
|
|
|
|
|
|
if (e->sr.srState != TCPSCLOSED)
|
|
|
|
return EISCONN;
|
|
|
|
|
|
|
|
|
|
|
|
IncBusy();
|
|
|
|
TCPIPSetNewDestination(e->ipid, sin->sin_addr, port);
|
|
|
|
t = _toolErr;
|
|
|
|
DecBusy();
|
|
|
|
|
|
|
|
if (t)
|
|
|
|
{
|
|
|
|
return ENETDOWN;
|
|
|
|
}
|
|
|
|
|
|
|
|
IncBusy();
|
|
|
|
terr = TCPIPOpenTCP(e->ipid);
|
|
|
|
t = _toolErr;
|
|
|
|
if (t) terr = t;
|
|
|
|
DecBusy();
|
|
|
|
|
|
|
|
// todo -- better errors.
|
|
|
|
if (terr)
|
|
|
|
{
|
|
|
|
return ENETDOWN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
timeout = GetTick() + 60 * 30;
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
|
|
|
|
int xerrno;
|
|
|
|
int state;
|
|
|
|
|
|
|
|
xerrno = queue_command(e, kCommandConnect, 0, timeout);
|
|
|
|
|
2012-05-05 02:20:05 -04:00
|
|
|
|
2012-05-05 01:08:27 -04:00
|
|
|
// hmmm .. should these abort?
|
|
|
|
if (xerrno == EINTR)
|
|
|
|
{
|
|
|
|
IncBusy();
|
|
|
|
e->command = kCommandNone;
|
|
|
|
TCPIPAbortTCP(e->ipid);
|
|
|
|
DecBusy();
|
|
|
|
|
|
|
|
return EINTR; // ?
|
|
|
|
}
|
|
|
|
if (xerrno) return EIO; // semaphore destroyed?
|
|
|
|
|
|
|
|
if (e->command) continue; // reset to 0 if processed.
|
|
|
|
|
|
|
|
state = e->sr.srState;
|
|
|
|
if (state == TCPSESTABLISHED)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (state == TCPSCLOSED)
|
|
|
|
// todo -- differentiate ECONNREFUSED vs EHOSTUNREACH
|
|
|
|
return ECONNREFUSED;
|
|
|
|
|
2012-05-05 02:20:05 -04:00
|
|
|
if (timeout && timeout <= GetTick())
|
2012-05-05 01:08:27 -04:00
|
|
|
{
|
|
|
|
IncBusy();
|
|
|
|
TCPIPAbortTCP(e->ipid);
|
|
|
|
DecBusy();
|
|
|
|
|
|
|
|
return ETIMEDOUT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return 0; // should never hit.
|
|
|
|
|
|
|
|
}
|