initial check in.

This commit is contained in:
Kelvin Sherlock 2018-11-12 19:36:34 -05:00
commit 56f1b1422f
4 changed files with 698 additions and 0 deletions

17
GNUmakefile Normal file
View File

@ -0,0 +1,17 @@
OBJ = o/main.a o/connection.a
dict : $(OBJ)
iix link o/main o/connection keep=$@
o/main.a : main.c connection.h
o/connection.a : connection.c connection.h
o :
mkdir $@
o/%.a : %.c | o
iix compile $< keep=o/$*

270
connection.c Normal file
View File

@ -0,0 +1,270 @@
#pragma optimize 79
#pragma noroot
#pragma lint -1
#include <IntMath.h>
#include "Memory.h"
#include "connection.h"
#include <string.h>
//#include "s16debug.h"
static char pstring[256];
static Word LoginAndOpen(Connection *buffer)
{
Word ipid;
Word terr;
if (buffer->displayPtr)
{
static char message[] = "\pConnecting to xxx.xxx.xxx.xxx:xxxxx";
Word length;
Word tmp;
length = 15;
// first the ip addresss...
tmp = TCPIPConvertIPToCASCII(buffer->dnr.DNRIPaddress, message + length, 0);
length += tmp;
message[length++] = ':';
// now the port...
Int2Dec(buffer->port, message + length, 5, 0);
length += 5;
message[length] = 0;
message[0] = length;
buffer->displayPtr(message);
}
ipid = TCPIPLogin(
buffer->memID,
buffer->dnr.DNRIPaddress,
buffer->port,
0x0000, 0x0040);
if (_toolErr)
{
buffer->state = kConnectionStateError;
return -1;
}
terr = TCPIPOpenTCP(ipid);
if (_toolErr || terr)
{
TCPIPLogout(ipid);
buffer->state = kConnectionStateError;
buffer->terr = terr;
buffer->ipid = 0;
return -1;
}
buffer->ipid = ipid;
buffer->state = kConnectionStateConnecting;
return 0;
}
Word ConnectionPoll(Connection *buffer)
{
Word state;
if (!buffer) return -1;
state = buffer->state;
if (state == 0) return -1;
if (state == kConnectionStateConnected) return 1;
if (state == kConnectionStateDisconnected) return 1;
if (state == kConnectionStateError) return -1;
TCPIPPoll();
if (state == kConnectionStateDNR)
{
if (buffer->dnr.DNRstatus == DNR_OK)
{
return LoginAndOpen(buffer);
}
else if (buffer->dnr.DNRstatus != DNR_Pending)
{
buffer->state = kConnectionStateError;
if (buffer->displayPtr)
{
static char message[] = "\pDNR lookup failed: $xxxx";
Int2Hex(buffer->dnr.DNRstatus, message + 21, 4);
buffer->displayPtr(message);
}
return -1;
}
}
if (state == kConnectionStateConnecting || state == kConnectionStateDisconnecting)
{
Word terr;
static srBuff sr;
terr = TCPIPStatusTCP(buffer->ipid, &sr);
if (state == kConnectionStateDisconnecting)
{
// these are not errors.
if (terr == tcperrConClosing || terr == tcperrClosing)
terr = tcperrOK;
}
if (terr || _toolErr)
{
//CloseAndLogout(buffer);
//s16_debug_printf("terr = %04x tool error = %04x\n", terr, _toolErr);
//s16_debug_srbuff(&sr);
TCPIPCloseTCP(buffer->ipid);
TCPIPLogout(buffer->ipid);
buffer->ipid = 0;
buffer->state = kConnectionStateError;
buffer->terr = terr;
return -1;
}
if (sr.srState == TCPSESTABLISHED) // && state == kConnectionStateConnecting)
{
buffer->state = kConnectionStateConnected;
return 1;
}
if (sr.srState == TCPSCLOSED || sr.srState == TCPSTIMEWAIT)
{
//s16_debug_srbuff(&sr);
TCPIPLogout(buffer->ipid);
buffer->ipid = 0;
buffer->state = kConnectionStateDisconnected;
return 1;
}
}
return 0;
}
Word ConnectionOpenC(Connection *buffer, const char *host, Word port)
{
Word length;
if (!host) return -1;
length = strlen(host);
if (length > 255) return -1;
pstring[0] = length & 0xff;
memcpy(pstring + 1, host, length);
return ConnectionOpen(buffer, pstring, port);
}
Word ConnectionOpenGS(Connection *buffer, const GSString255 *host, Word port)
{
if (!host) return -1;
if (host->length > 255) return -1;
pstring[0] = host->length & 0xff;
memcpy(pstring + 1, host->text, host->length);
return ConnectionOpen(buffer, pstring, port);
}
Word ConnectionOpen(Connection *buffer, const char *host, Word port)
{
buffer->state = 0;
buffer->ipid = 0;
buffer->terr = 0;
buffer->port = port;
if (!buffer || !*buffer || !host || !*host) return -1;
// 1. check if we need to do DNR.
if (TCPIPValidateIPString(host))
{
cvtRec cvt;
TCPIPConvertIPToHex(&cvt, host);
buffer->dnr.DNRIPaddress = cvt.cvtIPAddress;
buffer->dnr.DNRstatus = DNR_OK; // fake it.
return LoginAndOpen(buffer);
}
// do dnr.
if (buffer->displayPtr)
{
static char message[256] = "\pDNR lookup: ";
BlockMove(host + 1, message + 13, host[0]);
message[0] = 13 + host[0];
buffer->displayPtr(message);
}
TCPIPDNRNameToIP(host, &buffer->dnr);
if (_toolErr)
{
buffer->state = kConnectionStateError;
if (buffer->displayPtr)
{
static char message[] = "\pDNR lookup tool error: $xxxx";
Int2Hex(_toolErr, message + 25, 4);
buffer->displayPtr(message);
}
return -1;
}
buffer->state = kConnectionStateDNR;
return 0;
}
void ConnectionInit(Connection *buffer, Word memID, ConnectionCallback displayPtr)
{
buffer->memID = memID;
buffer->ipid = 0;
buffer->terr = 0;
buffer->state = 0;
buffer->port = 0;
buffer->dnr.DNRstatus = 0;
buffer->dnr.DNRIPaddress = 0;
buffer->displayPtr = displayPtr;
}
Word ConnectionClose(Connection *buffer)
{
Word state = buffer->state;
// todo -- how do you close if not yet connected?
if (state == kConnectionStateConnected)
{
buffer->state = kConnectionStateDisconnecting;
buffer->terr = TCPIPCloseTCP(buffer->ipid);
if (buffer->displayPtr)
{
static char message[] = "\pClosing connection: $0000";
Int2Hex(buffer->terr, message + 22, 4);
buffer->displayPtr(message);
}
return 0;
}
if (state == kConnectionStateDNR)
{
TCPIPCancelDNR(&buffer->dnr);
buffer->state = 0;
if (buffer->displayPtr)
{
static char message[] = "\pDNR lookup canceled";
buffer->displayPtr(message);
}
return 1;
}
return -1;
}

40
connection.h Normal file
View File

@ -0,0 +1,40 @@
#ifndef __CONNECTION_H__
#define __CONNECTION_H__
#ifndef __TCPIP__
#include <tcpip.h>
#endif
enum {
kConnectionStateDNR = 1,
kConnectionStateConnecting,
kConnectionStateConnected,
kConnectionStateDisconnecting,
kConnectionStateDisconnected,
kConnectionStateError
};
typedef void (*ConnectionCallback)(const char *message);
typedef struct Connection {
Word memID;
Word ipid;
Word terr;
Word state;
dnrBuffer dnr;
Word port;
ConnectionCallback displayPtr;
} Connection;
void ConnectionInit(Connection *, Word memID, ConnectionCallback displayPtr);
Word ConnectionOpen(Connection *, const char *host, Word port);
Word ConnectionOpenC(Connection *, const char *host, Word port);
Word ConnectionOpenGS(Connection *, const GSString255 *host, Word port);
Word ConnectionClose(Connection *);
Word ConnectionPoll(Connection *);
#endif

371
main.c Normal file
View File

@ -0,0 +1,371 @@
#pragma optimize 79
#pragma lint -1
#include <Locator.h>
#include <tcpip.h>
#include <MiscTool.h>
#include <Memory.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include "connection.h"
#define IncBusy() asm { jsl 0xE10064 }
#define DecBusy() asm { jsl 0xE10068 }
#define Resched() asm { cop 0x7f }
#define BusyFlag ((byte *)0xE100FFl)
// startup/shutdown flags.
enum {
kLoaded = 1,
kStarted = 2,
kConnected = 4,
kLoadError = -1,
kVersionError = -2
};
int StartUpTCP(displayPtr fx)
{
word status;
word flags = 0;
// TCPIP is an init, not a tool, so it should always
// be loaded.
status = TCPIPStatus();
if (_toolErr)
{
LoadOneTool(54, 0x0300);
if (_toolErr == toolVersionErr) return kVersionError;
if (_toolErr) return kLoadError;
status = 0;
flags |= kLoaded;
}
// require 3.0b3
if (TCPIPLongVersion() < 0x03006003)
{
if (flags & kLoaded)
UnloadOneTool(54);
return kVersionError;
}
if (!status)
{
TCPIPStartUp();
if (_toolErr) return kLoadError;
flags |= kStarted;
}
status = TCPIPGetConnectStatus();
if (!status)
{
TCPIPConnect(fx);
flags |= kConnected;
}
return flags;
}
void ShutDownTCP(int flags, Boolean force, displayPtr fx)
{
if (flags <= 0) return;
if (flags & kConnected)
{
TCPIPDisconnect(force, fx);
if (_toolErr) return;
}
if (flags & kStarted)
{
TCPIPShutDown();
if (_toolErr) return;
}
if (flags & kLoaded)
{
UnloadOneTool(54);
}
}
// #pragma databank [push | pop] would be nice...
#pragma databank 1
pascal void DisplayCallback(const char *message)
{
unsigned length;
// message is a p-string.
length = message ? message[0] : 0;
if (!length) return;
fprintf(stderr, "%.*s\n", length, message + 1);
}
#pragma databank 0
int ConnectLoop(char *host, Word port, Connection *connection)
{
LongWord qtick;
ConnectionInit(connection, MMStartUp(), DisplayCallback);
ConnectionOpenC(connection, host, port);
// 30 second timeout.
qtick = GetTick() + 30 * 60;
while (!ConnectionPoll(connection))
{
if (GetTick() >= qtick)
{
fprintf(stderr, "Connection timed out.\n");
IncBusy();
TCPIPAbortTCP(connection->ipid);
TCPIPLogout(connection->ipid);
DecBusy();
return 0;
}
}
if (connection->state != kConnectionStateConnected)
{
fprintf(stderr, "Unable to open host: %s:%u\n",
host,
port);
return 0;
}
return 1;
}
int CloseLoop(Connection *connection)
{
ConnectionClose(connection);
while (!ConnectionPoll(connection)) ; // wait for it to close.
return 1;
}
static char buffer[512];
int ReadLineSync(word ipid) {
LongWord qtick;
static srBuff sr;
static rlrBuff rlr;
unsigned x;
word terr;
//asm { brk 0xea }
buffer[0] = 0;
qtick = GetTick() + 30 * 60;
for(;;) {
Word terr;
TCPIPPoll();
terr = TCPIPStatusTCP(ipid, &sr);
if (sr.srRcvQueued) break;
if (terr) return -1;
if (GetTick() >= qtick) {
fprintf(stderr, "Read timed out.\n");
return -1;
}
}
for(;;) {
terr = TCPIPReadLineTCP(ipid, "\p\r\n", 0x0000, (Ref)&buffer, sizeof(buffer) - 1, &rlr);
/* if (terr) return -1; */ /* nb - marinetti return bug */
if (!rlr.rlrIsDataFlag) {
if (GetTick() >= qtick) {
fprintf(stderr, "Read timed out.\n");
return -1;
}
TCPIPPoll();
continue;
}
break;
}
x = rlr.rlrBuffCount;
buffer[x] = 0;
return x;
}
int status(void) {
unsigned x;
if (sscanf(buffer, "%u", &x) == 1) {
fprintf(stderr, "status: %d\n", x);
return x;
}
return -1;
}
int client(word ipid) {
word terr;
int ok;
/* also checks for 220 header. */
ok = ReadLineSync(ipid);
if (ok < 0) return ok;
ok = status();
if (ok != 220) return -1;
terr = TCPIPWriteTCP(ipid, "CLIENT dict-iigs\r\n", 18, 1, 0);
if (terr != 0) fprintf(stderr, "terr: %04x\n", terr);
ok = ReadLineSync(ipid);
if (ok < 0) return ok;
/*
3.6.2. Responses
250 ok (optional timing information here)
*/
ok = status();
if (ok != 250) return -1;
return 0;
}
int quit(word ipid) {
word terr;
int ok;
terr = TCPIPWriteTCP(ipid, "QUIT\r\n", 6, 1, 0);
if (terr != 0) fprintf(stderr, "terr: %04x\n", terr);
ok = ReadLineSync(ipid);
if (ok < 0) return ok;
/*
3.9.2. Responses
221 Closing Connection
*/
return 0;
}
int one_def(Word ipid) {
int ok;
for(;;) {
ok = ReadLineSync(ipid);
if (ok < 0) return ok;
ok = status();
if (ok == 250) return 0;
if (ok != 151) return -1;
for(;;) {
ok = ReadLineSync(ipid);
if (ok < 0) return ok;
if (buffer[0] == '.') {
fputc('\n', stdout);
break;
}
fputs(buffer, stdout);
fputc('\n', stdout);
}
}
fputc('\n', stdout);
return 0;
}
int define(Word ipid, const char *term, const char *dict) {
word terr;
int ok;
unsigned x;
static char buffer[512];
if (!dict || !*dict) dict = "!";
if (!term || !*term) return -1;
x = sprintf(buffer, "DEFINE %s \"%s\"\r\n", dict, term);
terr = TCPIPWriteTCP(ipid, buffer, x, 1, 0);
if (terr != 0) fprintf(stderr, "terr: %04x\n", terr);
/*
3.2.2. Responses
550 Invalid database, use "SHOW DB" for list of databases
552 No match
150 n definitions retrieved - definitions follow
151 word database name - text follows
250 ok (optional timing information here)
*/
ok = ReadLineSync(ipid);
if (ok < 0) return ok;
ok = status();
switch(status()) {
default: return -1;
case 550:
fprintf(stdout, "Invalid database.\n");
return -1;
case 552:
fprintf(stdout, "No match.\n");
return 0;
case 150:
return one_def(ipid);
}
}
int main(int argc, char **argv) {
Connection connection;
int mf;
int ok;
word terr;
mf = StartUpTCP(DisplayCallback);
if (argc < 1) exit(1);
if (mf < 0) {
fprintf(stderr, "Marinetti 3.0b3 or greater is required.\n");
exit(1);
}
ok = ConnectLoop("dict.org", 2628, &connection);
if (ok) {
unsigned i;
int ok;
ok = client(connection.ipid);
if (ok == 0) {
for (i = 1; i < argc; ++i) {
ok = define(connection.ipid, argv[i], NULL);
if (ok < 0) break;
}
}
quit(connection.ipid);
CloseLoop(&connection);
}
ShutDownTCP(mf, false, DisplayCallback);
return 0;
}