changeset 4:aa38447a4b21

First entry of Paradise Server 2.9 patch 10 Beta
author darius
date Sat, 06 Dec 1997 04:37:03 +0000 (1997-12-06)
parents cafa94d86546
children 054275999194
files src/fatudp.c src/feature.c src/findslot.c src/gameconf.c src/getentry.c src/getname.c src/getship.c src/getship.h src/getstats.c src/gppackets.h src/grid.c src/grid.h src/imath.c src/input.c src/interface.c src/listen.c src/main.c src/mes src/message.c src/misc.c src/misc.h src/orbit.c src/packets.c src/packets.h src/parsexbm.c src/path.c src/path.h src/phaser.c src/ping.c src/pl_gen0.c src/pl_gen1.c src/pl_gen2.c src/pl_gen3.c src/pl_gen4.c src/pl_gen5.c
diffstat 35 files changed, 14779 insertions(+), 0 deletions(-) [+]
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/fatudp.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,419 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <stdio.h>
+#include <sys/types.h>
+#include <netinet/in.h>
+
+#include "defs.h"
+#include "data.h"
+#include "packets.h"
+#include "shmem.h"
+
+/* #define FATDIAG	/* define to get LOTS of fat UDP debugging messages */
+
+
+/* this stuff is used for Fat UDP */
+typedef void *PTR;		/* adjust this if you lack (void *) */
+typedef struct fat_node_t
+{
+  PTR packet;
+  int pkt_size;
+  struct fat_node_t *prev;
+  struct fat_node_t *next;
+}   FAT_NODE;
+
+FAT_NODE fat_kills[MAXPLAYER];
+FAT_NODE fat_torp_info[MAXPLAYER * MAXTORP];
+FAT_NODE fat_thingy_info[TOTALTHINGIES];
+FAT_NODE fat_phaser[MAXPLAYER];
+FAT_NODE fat_plasma_info[MAXPLAYER * MAXPLASMA];
+FAT_NODE fat_you;
+#if 0
+FAT_NODE fat_status;
+FAT_NODE fat_planet[MAXPLANETS];
+#else
+FAT_NODE fat_status2;
+FAT_NODE fat_planet2[MAXPLANETS];
+#endif
+FAT_NODE fat_flags[MAXPLAYER];
+FAT_NODE fat_hostile[MAXPLAYER];
+
+/* define the lists */
+#define MAX_FAT_LIST	5	/* tweakable; should be > 1 */
+typedef struct
+{
+  FAT_NODE *head;
+  FAT_NODE *tail;
+}   FAT_LIST;
+FAT_LIST fatlist[MAX_FAT_LIST], tmplist[MAX_FAT_LIST];
+/* tweakable parameters; compare with UDPBUFSIZE */
+/* NOTE: FAT_THRESH + MAX_FAT_DATA must be < UDPBUFSIZE                */
+/* MAX_FAT_DATA must be larger than biggest semi-critical packet */
+#define MAX_FAT_DATA	100	/* add at most this many bytes */
+#define MAX_NONFAT	10	/* if we have this much left, stop */
+
+
+void 
+reset_fat_list()
+{
+  int i;
+  for (i = 0; i < MAX_FAT_LIST; i++)
+    fatlist[i].head = fatlist[i].tail = (FAT_NODE *) NULL;
+}
+/*
+ * 
+ * ---------------------------------------------------------------------------
+ * Fat City
+ * ---------------------------------------------------------------------------
+ * */
+
+/*
+ * Remove a FAT_NODE from a queue.  If it's at the head or the tail of a
+ * list, then we need to figure out which list it's in and update the head or
+ * tail pointer.  It's easier to go searching than to maintain a queue number
+ * in every FAT_NODE.
+ * 
+ * This routine looks too complex... there must be a simpler way to do this.
+ */
+void
+dequeue(fatp)
+  FAT_NODE *fatp;
+{
+  int i;
+
+#ifdef V_FATDIAG
+  for (i = 0; i < MAX_FAT_LIST; i++)
+  {
+    printf("fatlist[i].head = 0x%.8lx tail = 0x%.8lx\n",
+	   fatlist[i].head, fatlist[i].tail);
+    if ((fatlist[i].head == NULL && fatlist[i].tail != NULL) ||
+	(fatlist[i].head != NULL && fatlist[i].tail == NULL))
+    {
+      printf("before!\n");
+      kill(getpid(), 15);
+    }
+  }
+#endif
+
+
+  if (fatp->next == NULL)
+  {
+    /* it's at the head or not in a queue */
+    for (i = 0; i < MAX_FAT_LIST; i++)
+    {
+      if (fatlist[i].head == fatp)
+      {
+	fatlist[i].head = fatp->prev;	/* move head back */
+	if (fatlist[i].head != NULL)
+	  (fatlist[i].head)->next = NULL;	/* amputate */
+	break;
+      }
+    }
+  }
+  else
+  {
+    /* it's not at the head */
+    if (fatp->prev != NULL)
+      fatp->prev->next = fatp->next;
+  }
+
+  if (fatp->prev == NULL)
+  {
+    /* it's at the tail or not in a queue */
+    for (i = 0; i < MAX_FAT_LIST; i++)
+    {
+      if (fatlist[i].tail == fatp)
+      {
+	fatlist[i].tail = fatp->next;	/* move head fwd */
+	if (fatlist[i].tail != NULL)
+	  (fatlist[i].tail)->prev = NULL;	/* amputate */
+	break;
+      }
+    }
+  }
+  else
+  {
+    /* it's not at the tail */
+    if (fatp->next != NULL)
+      fatp->next->prev = fatp->prev;
+  }
+
+#ifdef FATDIAG
+  printf("Removed 0x%.8lx...", fatp);	/* FATDIAG */
+  for (i = 0; i < MAX_FAT_LIST; i++)
+  {
+    if ((fatlist[i].head == NULL && fatlist[i].tail != NULL) ||
+	(fatlist[i].head != NULL && fatlist[i].tail == NULL))
+    {
+      printf("after: %d %.8lx %.8lx\n", i, fatlist[i].head, fatlist[i].tail);
+      kill(getpid(), 15);
+    }
+  }
+#endif
+  fatp->prev = NULL;
+  fatp->next = NULL;
+}
+
+/*
+ * Add a FAT_NODE to the tail of a temporary queue.  The merge() routine
+ * merges the temporary queues with the fatlists once the transmission is
+ * sent.
+ */
+void
+enqueue(fatp, list)
+  FAT_NODE *fatp;
+  int list;
+{
+#ifdef FATDIAG
+  printf("added to tmplist %d\n", list);	/* FATDIAG */
+#endif
+
+  if (tmplist[list].tail == NULL)
+  {
+    /* list was empty */
+    tmplist[list].tail = tmplist[list].head = fatp;
+  }
+  else
+  {
+    /* list wasn't empty */
+    fatp->next = tmplist[list].tail;
+    fatp->next->prev = fatp;
+    tmplist[list].tail = fatp;
+  }
+}
+
+/*
+ * This updates the "fat" tables; it's called from sendClientData().
+ */
+void
+updateFat(packet)
+/* Pick a random type for the packet */
+  struct player_spacket *packet;
+{
+  FAT_NODE *fatp;
+  struct kills_spacket *kp;
+  struct torp_info_spacket *tip;
+  struct thingy_info_spacket *thip;
+  struct phaser_spacket *php;
+  struct plasma_info_spacket *pip;
+  /* struct you_spacket *yp; */
+  /* struct status_spacket2 *sp2; */
+  /* struct planet_spacket *plp; */
+  struct planet_spacket2 *plp2;
+  struct flags_spacket *fp;
+  struct hostile_spacket *hp;
+  int idx;
+
+  /* step 1 : find the FAT_NODE for this packet */
+  switch (packet->type)
+  {
+   case SP_KILLS:
+    kp = (struct kills_spacket *) packet;
+    idx = (int) kp->pnum;
+    fatp = &fat_kills[idx];
+    break;
+   case SP_TORP_INFO:
+    tip = (struct torp_info_spacket *) packet;
+    idx = (int) ntohs(tip->tnum);
+    fatp = &fat_torp_info[idx];
+    break;
+   case SP_THINGY_INFO:
+    thip = (struct thingy_info_spacket *) packet;
+    idx = (int) ntohs(thip->tnum);
+    fatp = &fat_thingy_info[idx];
+    break;
+   case SP_PHASER:
+    php = (struct phaser_spacket *) packet;
+    idx = (int) php->pnum;
+    fatp = &fat_phaser[idx];
+    break;
+   case SP_PLASMA_INFO:
+    pip = (struct plasma_info_spacket *) packet;
+    idx = (int) ntohs(pip->pnum);
+    fatp = &fat_plasma_info[idx];
+    break;
+   case SP_YOU:
+    /* yp = (struct you_spacket *) packet; */
+    fatp = &fat_you;
+    break;
+#if 0
+   case SP_STATUS:
+    /* sp = (struct status_spacket *) packet; */
+    fatp = &fat_status;
+    break;
+   case SP_PLANET:
+    plp = (struct planet_spacket *) packet;
+    idx = plp->pnum;
+    fatp = &fat_planet[idx];
+    break;
+#else
+   case SP_STATUS2:
+    /* sp = (struct status_spacket *) packet; */
+    fatp = &fat_status2;
+    break;
+   case SP_PLANET2:
+    plp2 = (struct planet_spacket2 *) packet;
+    idx = plp2->pnum;
+    fatp = &fat_planet2[idx];
+    break;
+#endif
+   case SP_FLAGS:
+    fp = (struct flags_spacket *) packet;
+    idx = (int) fp->pnum;
+    fatp = &fat_flags[idx];
+    break;
+   case SP_HOSTILE:
+    hp = (struct hostile_spacket *) packet;
+    idx = (int) hp->pnum;
+    fatp = &fat_hostile[idx];
+    break;
+   default:
+    fprintf(stderr, "Fat error: bad semi-critical type (%d) in updateFat\n", (CARD8) packet->type);
+    return;
+  }
+
+  if (fatp->packet != (PTR) packet)
+  {
+    fprintf(stderr, "Fat error: fatp->packet=0x%.8lx, packet=0x%.8lx\n",
+	    (unsigned long) fatp->packet, (unsigned long) packet);
+    return;
+  }
+  /* step 2 : move this dude to temporary list 0 */
+  dequeue(fatp);
+  enqueue(fatp, 0);
+}
+
+/*
+ * This fattens up the transmission, adding up to MAX_FAT_DATA bytes.  The
+ * packets which get added will be moved to a higher queue, giving them less
+ * priority for next time.  Note that they are added to a parallel temporary
+ * list (otherwise they'd could be sent several times in the same
+ * transmission as the algorithm steps through the queues), and merged later
+ * on.
+ * 
+ * Packets are assigned from head to tail, on a first-fit basis.  If a
+ * semi-critical packet is larger than MAX_FAT_DATA, this routine will never
+ * send it, but it will skip around it.
+ * 
+ * This routine is called from flushSockBuf, before the transmission is sent.
+ * 
+ * A possible improvement is to have certain packets "expire", never to be seen
+ * again.  This way we don't keep resending torp packets for players who are
+ * long dead.  This raises the possibility that the dead player's torps will
+ * never go away though.
+ */
+int
+fatten()
+{
+  int bytesleft;
+  FAT_NODE *fatp, *nextfatp;
+  int list;
+
+#ifdef FATDIAG
+  printf("--- fattening\n");
+#endif
+  bytesleft = MAX_FAT_DATA;
+  for (list = 0; list < MAX_FAT_LIST; list++)
+  {
+    fatp = fatlist[list].head;
+    while (fatp != NULL)
+    {
+      nextfatp = fatp->prev;	/* move toward tail */
+#ifdef FATDIAG
+      if (nextfatp == fatp)
+      {
+	printf("Hey!  nextfatp == fatp!\n");
+	kill(getpid(), 15);
+      }
+#endif
+      if (fatp->pkt_size < bytesleft)
+      {
+	/* got one! */
+	sendUDPbuffered(0, fatp->packet, fatp->pkt_size);
+	bytesleft -= fatp->pkt_size;
+
+	packets_sent++;		/* counts as a udp packet sent */
+
+	/* move the packet to a higher queue (if there is one) */
+	dequeue(fatp);
+	if (list + 1 == MAX_FAT_LIST)
+	{
+	  /* enqueue(fatp, list); *//* keep packets on high queue? */
+	}
+	else
+	{
+	  enqueue(fatp, list + 1);
+	}
+
+	/* done yet? */
+	if (bytesleft < MAX_NONFAT)
+	  goto done;		/* don't waste time searching anymore */
+      }
+      fatp = nextfatp;
+    }
+  }
+
+done:
+  /* at this point, we either filled with fat or ran out of queued packets */
+#ifdef FATDIAG
+  printf("--- done\n");
+#endif
+  V_UDPDIAG(("- Added %d grams of fat\n", MAX_FAT_DATA - bytesleft));
+  return (0);			/* some compilers need something after a goto */
+}
+
+
+/*
+ * This gets called from flushSockBuf after the transmission is sent.  It
+ * appends all the packets sitting in temporary queues to the corresponding
+ * fat queues, where they will be eligible for fattening next transmission.
+ */
+void
+fatMerge()
+{
+  int i;
+
+  for (i = 0; i < MAX_FAT_LIST; i++)
+  {
+    if (tmplist[i].head == NULL)
+      continue;			/* temp list is empty, nothing to do */
+
+    if (fatlist[i].head == NULL)
+    {
+      /* fatlist is empty; just copy pointers */
+      fatlist[i].head = tmplist[i].head;
+      fatlist[i].tail = tmplist[i].tail;
+
+    }
+    else
+    {
+      /* stuff in both */
+      (tmplist[i].head)->next = fatlist[i].tail;	/* fwd pointer */
+      (fatlist[i].tail)->prev = tmplist[i].head;	/* back pointer */
+      fatlist[i].tail = tmplist[i].tail;	/* move tail back */
+      tmplist[i].head = tmplist[i].tail = NULL;	/* clear the tmp list */
+    }
+    tmplist[i].head = tmplist[i].tail = NULL;
+#ifdef FATDIAG
+    printf("merged list %d: %.8lx %.8lx\n", i, fatlist[i].head,
+	   fatlist[i].tail);
+#endif
+  }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/feature.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,228 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#ifdef FEATURE
+#ifdef HPUX
+#include <sys/types.h>
+#include <netinet/in.h>
+#endif
+#include "defs.h"
+#include "data.h"
+#include "struct.h"
+#include "packets.h"
+
+void sendClientPacket();
+
+#ifdef UNIXWARE			/* can you BELIEVE it doesn't have strcmpi?? */
+#if 0				/* just  use strcmp for now */
+int 
+strcmpi(char *a, char *b)
+{
+
+  char ta, tb;
+
+  while ((*a) && (*b))
+  {
+    ta = ((*a) >= 'a') && ((*b) <= 'z')) ? (*a) & 223 : (*a);
+    tb = ((*b) >= 'a') && ((*b) <= 'z')) ? (*b) & 223 : (*b);
+    if (ta < tb)
+      return (-1);
+    if (ta > tb)
+      return (1);
+    a++;
+    b++;
+  }
+  if (!(*a) && (*b))
+    return (1);
+  if (!(*b) && (*a))
+    return (-1);
+  return (0);
+}
+#endif
+#endif
+
+/*
+ * Feature.c
+ * 
+ * March, 1994.    Joe Rumsey, Tedd Hadley
+ * 
+ * most of the functions needed to handle SP_FEATURE/CP_FEATURE packets.  fill
+ * in the features list below for your client, and add a call to
+ * reportFeatures just before the RSA response is sent. handleFeature should
+ * just call checkFeature, which will search the list and set the appropriate
+ * variable.  features unknown to the server are set to the desired value for
+ * client features, and off for server/client features.
+ * 
+ * feature packets look like: struct feature_cpacket { char
+ * type; char                 feature_type; char                 arg1, arg2;
+ * int                  value; char                 name[80]; };
+ * 
+ * type is CP_FEATURE, which is 60.  feature_spacket is identical.
+ * 
+ * Code lifted July 1995 by Bob Glamm - enabled into server code.
+ * 
+ */
+
+void handleFeature(struct feature_cpacket *);
+void 
+sendFeature(char *name, int feature_type,
+	    int value, int arg1, int arg2);
+
+/* not the actual packets: this holds a list of features to report for */
+/* this client. */
+struct feature
+{
+  char *name;
+  int *var;			/* holds allowed/enabled status */
+  char feature_type;		/* 'S' or 'C' for server/client */
+  int value;			/* desired status */
+  char *arg1, *arg2;		/* where to copy args, if non-null */
+}   features[] =
+{
+  /*
+   * also sent seperately, put here for checking later. should be ok that
+   * it's sent twice.
+   */
+  {
+    "FEATURE_PACKETS", &F_feature_packets, 'S', 1, 0, 0
+  },
+
+  {
+    "VIEW_BOX", &F_allowViewBox, 'C', 1, 0, 0
+  },
+  {
+    "SHOW_ALL_TRACTORS", &F_allowShowAllTractorPressor, 'S', 1, 0, 0
+  },
+#ifdef CONTINUOUS_MOUSE
+  {
+    "CONTINUOUS_MOUSE", &F_allowContinuousMouse, 'C', 1, 0, 0
+  },
+#endif
+  {
+    "NEWMACRO", &F_UseNewMacro, 'C', 1, 0, 0
+  },
+  /* {"SMARTMACRO",&F_UseSmartMacro, 'C', 1, 0, 0}, */
+  {
+    "MULTIMACROS", &F_multiline_enabled, 'S', 1, 0, 0
+  },
+  {
+    "WHY_DEAD", &F_why_dead, 'S', 1, 0, 0
+  },
+  {
+    "CLOAK_MAXWARP", &F_cloakerMaxWarp, 'S', 1, 0, 0
+  },
+  /* {"DEAD_WARP", &F_dead_warp, 'S', 1, 0, 0}, */
+#ifdef RC_DISTRESS
+  {
+    "RC_DISTRESS", &F_gen_distress, 'S', 1, 0, 0
+  },
+#ifdef BEEPLITE
+  {
+    "BEEPLITE", &F_allow_beeplite, 'C', 1, (char *) &F_beeplite_flags, 0
+  },
+#endif
+#endif
+  /* terrain features */
+  {
+    "TERRAIN", &F_terrain, 'S', 1, (char *) &F_terrain_major, (char *) &F_terrain_minor
+  },
+  /* Gzipped MOTD */
+  {
+    "GZ_MOTD", &F_gz_motd, 'S', 1, (char *) &F_gz_motd_major, (char *) &F_gz_motd_minor
+  },
+  {
+    0, 0, 0, 0, 0, 0
+  }
+};
+
+void 
+handleFeature(packet)
+  struct feature_cpacket *packet;
+{
+  int feature = 0;
+  int value = ntohl(packet->value);
+
+#ifdef FEATURE_DIAG
+  pmessage("Whoohoo!  Getting a feature packet.", 0, MALL, MSERVA);
+#endif
+  while (features[feature].name)
+  {
+    if (!strcmp(features[feature].name, packet->name))
+    {
+#ifdef FEATURE_DIAG
+      {
+	char buf[80];
+	sprintf(buf, "BEANS!  Matched feature %s", packet->name);
+	pmessage(buf, 0, MALL, MSERVA);
+      }
+#endif
+      /* A match was found. */
+      if (features[feature].var)
+	*features[feature].var = packet->value;
+      if (features[feature].arg1)
+	*features[feature].arg1 = packet->arg1;
+      if (features[feature].arg2)
+	*features[feature].arg2 = packet->arg2;
+
+      /*
+       * tell the client that the server handles all of the above server
+       * features.
+       */
+      if (features[feature].feature_type == 'S')
+      {
+#ifdef FEATURE_DIAG
+	pmessage("Sending FEATURE_PACKETS right back at ya!", 0, MALL, MSERVA);
+#endif
+	sendFeature(features[feature].name,
+		    features[feature].feature_type,
+		    features[feature].value,
+		    (features[feature].arg1 ? *features[feature].arg1 : 0),
+		    (features[feature].arg2 ? *features[feature].arg2 : 0));
+      }
+    }
+    feature++;
+  }
+}
+
+void
+sendFeature(name, feature_type, value, arg1, arg2)
+  char *name;
+  int feature_type;
+  int value;
+  int arg1, arg2;
+{
+  struct feature_cpacket packet;
+#ifdef FEATURE_DIAG
+  char buf[80];
+
+  sprintf(buf, "Sending packet %d, name %s", SP_FEATURE, name);
+  pmessage(buf, 0, MALL, MSERVA);
+#endif
+
+  strncpy(packet.name, name, sizeof(packet.name));
+  packet.type = SP_FEATURE;
+  packet.name[sizeof(packet.name) - 1] = 0;
+  packet.feature_type = feature_type;
+  packet.value = htonl(value);
+  packet.arg1 = arg1;
+  packet.arg2 = arg2;
+  sendClientPacket((struct player_spacket *) & packet);
+}
+
+#endif				/* FEATURE */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/findslot.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,491 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+
+#include "config.h"
+#include <stdio.h>
+#include <sys/types.h>
+#include <sys/ipc.h>
+#include <sys/shm.h>
+#include <errno.h>
+#include <pwd.h>
+#include <string.h>
+#include <ctype.h>
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "packets.h"
+#include "shmem.h"
+
+int grabslot( /* int */ );
+void mapWaitCount( /* int */ );
+
+
+extern int isClientDead();
+extern unsigned int sleep();
+extern int sendQueuePacket();
+extern int flushSockBuf();
+
+int
+findslot(overload, homeaway)
+  int overload;
+  enum HomeAway homeaway;
+{
+  int i;
+
+  i = grabslot(overload, homeaway);
+  players[i].p_pos = -1;
+  memset(&players[i].p_stats, 0, sizeof(struct stats));
+  players[i].p_stats.st_tticks = 1;
+#if 0
+  for (j = 0; j < 95; j++)
+  {
+    players[i].p_stats.st_keymap[j] = j + 32;
+  }
+  players[i].p_stats.st_keymap[95] = 0;
+#endif
+  players[i].p_stats.st_flags = ST_INITIAL;
+  return (i);
+}
+
+static int 
+allocate_slot(homeaway, overload, allocate)
+  enum HomeAway homeaway;
+  int overload;
+  int allocate;
+{
+  int i;
+  if (overload)
+  {
+    for (i = MAXPLAYER - 1; i >= 0; i--)
+    {
+      if (players[i].p_status == PFREE)
+      {
+	if (allocate)
+	{
+	  players[i].p_status = POUTFIT;
+	  players[i].p_team = NOBODY;
+	}
+	return i;
+      }
+    }
+  }
+  else
+  {
+#ifdef LEAGUE_SUPPORT
+    if (status2->league)
+    {
+      /*
+       * for league play, make sure one team doesn't crowd out the other.
+       */
+      int count = 0;
+      for (i = 0; i < MAXPLAYER; i++)
+      {
+	if (players[i].p_status == PFREE)
+	  continue;
+	if (players[i].p_homeaway == homeaway)
+	  count++;
+      }
+      if (count * 2 >= MAXPLAYER - configvals->ntesters)
+	return -1;		/* our team is full */
+    }
+#endif
+    for (i = 0; i < MAXPLAYER - configvals->ntesters; i++)
+    {
+      if (players[i].p_status == PFREE)
+      {
+	if (allocate)
+	{
+	  players[i].p_status = POUTFIT;
+	  players[i].p_team = NOBODY;
+	}
+	return i;
+      }
+    }
+  }
+  return -1;
+}
+#if 0
+static int 
+ImAllowed(slotnum, homeaway)
+  int slotnum;
+  enum HomeAway homeaway;
+{
+#ifdef LEAGUE_SUPPORT
+  int half;
+  if (!status2->league)
+    return 1;
+
+  half = (MAXPLAYER - configvals->ntesters) / 2;
+  return (homeaway == AWAY) ? (slotnum < half) :
+    (slotnum >= half);
+#else
+  return 1;
+#endif
+}
+#endif
+/*
+ * The following code for grabslot() is really bizarre, and needs an
+ * explaination.
+ * 
+ * Basically, the queue works like this:  Each process that needs to wait takes
+ * a number, and when that number comes up, the process enters the game.
+ * status->count is the next number to take for a new process, and
+ * status->wait is the current process being served.
+ * 
+ * However, this is not enough to determine exactly how many people are waiting,
+ * because people may drop out.  So, 3 more variables are added,
+ * status->request posts a request of some sort to all of the other
+ * processes, and status->number is the process posting the request. Also,
+ * status->answer is available for any kind of response needed from the other
+ * processes.  (Needless to say, only one process can make a request at any
+ * one time).
+ * 
+ * Every process will wait for a second (sleep(1)), and then check the queue
+ * status, do what is appropriate, and repeat.
+ * 
+ * Above and beyond this, processes may die, and not report it (for whatever
+ * reason, and every process waiting on the queue watches the behavior of
+ * every other supposed process to make sure it is still alive).
+ * 
+ * When a space opens up, and the person who holds the number currently being
+ * served does not respond, then the processes argue over who deserves to get
+ * in next.  The first process to decide that no one is going to take the
+ * free slot says that his number is the next on to be served. He waits for
+ * anyone to disagree with him.  Any other processes which notice that their
+ * number is lower than the number being served claim that their number is
+ * the next one to be served.  They also wait for anyone to disagree with
+ * them.  Eventually, everyone is done waiting, and the process with the
+ * lowest count will be served.
+ * 
+ * Variables: status->wait:  	Number being served. status->count:	Next
+ * number to take.
+ * 
+ * status->request:	Process request. (status->number) (status->answer)
+ * REQFREE:	No requests pending. REQWHO:		How many people are
+ * before me?  In this case, every process in from of this process (whose
+ * position is recorded in status->number) increments status->answer.
+ * REQDEAD:	I am leaving the queue (Either to quit, or enter game).  Any
+ * process whose position is higher than this process will note that they are
+ * closer to the top of the queue.
+ * 
+ * (Local) waitWin:	The window. qwin:	The quit half of waitWin.
+ * countWin:	The count half of waitWin. count:	My number (Position
+ * in queue). pseudocount: Number of people in front of me (-1 means I don't
+ * know). myRequest:	This keeps track of requests I've made.  If it is non
+ * zero, then this is the number of times I will leave it there before I get
+ * my answer (if there is one), and reset status->request. idie:	This
+ * is set to one if I need to leave the queue.  When I get a chance, I will
+ * submit my request, and leave. wearein:	This is the slot we grabbed
+ * to enter the game.  If it is -1, then we haven't got a slot yet.  If we
+ * can grab a slot, then wearein is set to the slot #, and idie is set to 1.
+ * 
+ * Because we need to monitor other processes, the following variables also
+ * exist:
+ * 
+ * oldcount:	The number that was being served last time I looked. waits:
+ * Number of times I've seen oldcount as the number being served. If a
+ * position opens in the game, and no one takes it for a while, we assume
+ * that someone died. lastRequest:	The last request I have seen.
+ * lastNumber:	The process making this request. reqCount:	Number of
+ * times I've seen this request.  If I see this request 9 times, I assume it
+ * is obsolete, and I reset it.
+ */
+
+int
+grabslot(overload, homeaway)
+  int overload;			/* Indicates a request for a tester's slot. */
+  enum HomeAway homeaway;
+{
+  int count;			/* My number */
+  int oldcount;			/* Number that was being served last check */
+  int i;
+  int waits;			/* # times I have waited for someone else to
+				 * act */
+  int oldwait;			/* Number being served last check */
+  int pseudocount = -1;		/* Count on queue for sake of person waiting */
+  int myRequest = 0;		/* To keep track of any request I'm making */
+  int lastRequest = 0;		/* Last request I've seen */
+  int lastNumber = 0;		/* Last person making request */
+  int reqCount = 0;		/* Number of times I've seen this */
+  int idie = 0;			/* Do I want to die? */
+  int wearein = -1;		/* Slot we got in the game */
+  int rep = 0;
+
+  /* If other players waiting, we get in line behind them */
+  if (!overload && status->wait != status->count)
+  {
+    count = status->count++;
+  }
+  else
+  {
+    /* Get in game if posible */
+#if 1
+    i = allocate_slot(homeaway, overload, 1);
+    if (i >= 0)
+      return i;
+#else
+    if (overload)
+      for (i = MAXPLAYER - 1; i >= 0; i--)
+      {
+	if (players[i].p_status == PFREE)
+	{			/* We have a free slot */
+	  players[i].p_status = POUTFIT;	/* possible race code */
+	  players[i].p_team = NOBODY;
+	  return (i);
+	}
+      }
+    else
+      for (i = 0; i < MAXPLAYER - configvals->ntesters; i++)
+      {
+	if (ImAllowed(i, homeaway)
+	    && players[i].p_status == PFREE)
+	{			/* We have a free slot */
+	  players[i].p_status = POUTFIT;	/* possible race code */
+	  players[i].p_team = NOBODY;
+	  return (i);
+	}
+      }
+#endif
+    /* Game full.  We will wait. */
+    count = status->count++;
+  }
+  waits = 0;
+  oldwait = -1;
+  oldcount = status->wait;
+
+  /* For count = 0,1,2  I know that it is right */
+  if (count - status->wait < 1)
+  {
+    pseudocount = count - status->wait;
+  }
+  for (;;)
+  {
+    /* Send packets occasionally to see if he is accepting... */
+    if (rep++ % 10 == 0)
+    {
+      mapWaitCount(pseudocount);
+    }
+    if (isClientDead())
+    {
+      if (count == status->count - 1)
+      {
+	status->count--;
+	exit(0);
+      }
+      /* If we are at top, decrease it */
+      if (count == status->wait)
+      {
+	status->wait++;
+      }
+      idie = 1;
+      /* break;		warning, function has return e and return */
+      exit(0);
+    }
+    if (status->wait != oldcount)
+    {
+      mapWaitCount(pseudocount);
+      oldcount = status->wait;
+    }
+    /* To mimize process overhead and aid synchronization */
+    sleep(1);
+    /* Message from daemon that it died */
+    if (status->count == 0)
+      exit(0);
+    /* I have a completed request? */
+    if (myRequest != 0 && --myRequest == 0)
+    {
+      if (idie && status->request == REQDEAD)
+      {
+	status->request = REQFREE;
+	/* Out of queue, into game */
+	if (wearein != -1)
+	{
+	  return (wearein);
+	}
+	exit(0);
+      }
+      pseudocount = status->answer;
+      status->request = REQFREE;
+      if (pseudocount > 18)
+	idie = 1;
+      mapWaitCount(pseudocount);
+    }
+    /* Tell the world I am going bye bye */
+    if (idie && status->request == REQFREE)
+    {
+      status->request = REQDEAD;
+      status->number = count;
+      myRequest = 4;
+    }
+    /* Should I request a count for # of people waiting? */
+    if (pseudocount == -1 && status->request == REQFREE)
+    {
+      status->request = REQWHO;
+      status->number = count;
+      status->answer = 0;
+      myRequest = 4;
+      /* I give people 4 seconds to respond */
+    }
+    /* Is someone else making a request? */
+    if (status->request != REQFREE && myRequest == 0)
+    {
+      if (status->request == lastRequest &&
+	  status->number == lastNumber)
+      {
+	reqCount++;
+	/* 9 occurances of the same request implies that the process */
+	/* died.  I will reset request. */
+	if (reqCount > 8)
+	{
+	  status->request = REQFREE;
+	}
+      }
+      else
+      {
+	lastRequest = status->request;
+	lastNumber = status->number;
+	reqCount = 1;
+	if (lastRequest == REQWHO)
+	{
+	  if (count < lastNumber)
+	  {
+	    status->answer++;
+	  }
+	}
+	else if (lastRequest == REQDEAD)
+	{
+	  if (count > lastNumber && pseudocount != -1)
+	  {
+	    pseudocount--;
+	    mapWaitCount(pseudocount);
+	  }
+	}
+      }
+    }
+    /* If someone raised wait too high, I claim that I * am next in line */
+    if (status->wait > count && !idie)
+    {
+      status->wait = count;
+      /* Give people a chance to correct me */
+      sleep(2);
+    }
+#if 1
+    if (idie)
+      continue;
+    if (count == status->wait)
+    {
+      i = allocate_slot(homeaway, overload, 1);
+      if (i < 0)
+	continue;
+      status->wait++;
+      wearein = i;
+      idie = 1;
+    }
+    else
+    {
+      i = allocate_slot(homeaway, overload, 0);
+      if (i < 0)
+	continue;
+      if (oldwait == status->wait)
+      {
+	waits++;
+      }
+      else
+      {
+	oldwait = status->wait;
+	waits = 1;
+      }
+      /* If this is our fifth wait (5 sec), then something is */
+      /* wrong.  We assume someone died, and fix this problem */
+      if (waits == 5 && !idie)
+      {
+	/* I want to be next in line, so I say so. */
+	status->wait = count;
+	/* And I allow someone to correct me if I'm wrong */
+	sleep(2);
+	waits = 0;
+      }
+
+    }
+#else
+    for (i = 0; i < MAXPLAYER - configvals->ntesters; i++)
+    {
+      /* If we want to die anyway, we have no right looking for */
+      /* a free slot */
+      if (idie)
+	break;
+      if (ImAllowed(i, homeaway) && players[i].p_status == PFREE)
+      {
+	/* If I am next in line... */
+	if (count == status->wait)
+	{
+	  players[i].p_status = POUTFIT;
+	  players[i].p_team = NOBODY;
+	  /* Increase count for next player */
+	  status->wait++;
+	  /* I should check idie, but maybe he wants in, eh? */
+	  wearein = i;
+	  idie = 1;
+	  break;
+	}
+	else
+	{
+	  if (oldwait == status->wait)
+	  {
+	    waits++;
+	  }
+	  else
+	  {
+	    oldwait = status->wait;
+	    waits = 1;
+	  }
+	  /* If this is our fifth wait (5 sec), then something is */
+	  /* wrong.  We assume someone died, and fix this problem */
+	  if (waits == 5 && !idie)
+	  {
+	    /* I want to be next in line, so I say so. */
+	    status->wait = count;
+	    /* And I allow someone to correct me if I'm wrong */
+	    sleep(2);
+	    waits = 0;
+	  }
+	  break;
+	}
+      }
+    }
+#endif
+    /* this location is skipped if we didn't find a slot */
+  }
+}
+
+void
+mapWaitCount(count)
+  unsigned int count;
+{
+  if (count == -1)
+    return;
+
+  sendQueuePacket((short) count);
+  blk_flag = 1;
+  updateMOTD();
+  blk_flag = 0;
+
+  undeferDeferred();		/* send the MOTD through the TCP buffers */
+  flushSockBuf();
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/gameconf.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,343 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <sys/types.h>
+#include <stdio.h>
+#include <netinet/in.h>
+#include "defs.h"
+#include "gppackets.h"
+#include "shmem.h"
+#include "data.h"
+
+/* some game params packet stuff */
+/* status experimental */
+
+void
+updateGPrank()
+{
+  int i;
+  struct gp_rank_spacket pkt;
+
+  pkt.type = SP_GPARAM;
+  pkt.subtype = 5;
+
+  for (i = 0; i < NUMRANKS; i++)
+  {
+    pkt.rankn = i;
+    pkt.genocides = htonl(ranks[i].genocides);
+#ifndef BAD_SVR4_HACKS
+    pkt.milliDI = htonl(1000 * ranks[i].di);
+    pkt.millibattle = htonl(1000 * ranks[i].battle);
+    pkt.millistrat = htonl(1000 * ranks[i].strategy);
+    pkt.millispec = htonl(1000 * ranks[i].specship);
+#else				/* Unixware/IRIX requires explicit casts */
+    /* this next define will make the next more readable */
+#define BAD_SVR4_HACKS_CAST(a,b) (unsigned long)( (long double)a * (long double)b )
+    pkt.milliDI = htonl(BAD_SVR4_HACKS_CAST(1000, ranks[i].di));
+    pkt.millibattle = htonl(BAD_SVR4_HACKS_CAST(1000, ranks[i].battle));
+    pkt.millistrat = htonl(BAD_SVR4_HACKS_CAST(1000, ranks[i].strategy));
+    pkt.millispec = htonl(BAD_SVR4_HACKS_CAST(1000, ranks[i].specship));
+#endif				/* BAD_SVR4_HACKS */
+    strcpy(pkt.name, ranks[i].name);
+
+    sendClientPacket((struct player_spacket *) & pkt);
+  }
+}
+
+void
+updateGProyal()
+{
+  int i;
+  struct gp_royal_spacket pkt;
+
+  pkt.type = SP_GPARAM;
+  pkt.subtype = 6;
+
+  for (i = 0; i < NUMROYALRANKS; i++)
+  {
+    pkt.rankn = i;
+    strcpy(pkt.name, royal[i].name);
+    sendClientPacket((struct player_spacket *) & pkt);
+  }
+}
+
+void
+updateGPteams()
+{
+  struct gp_team_spacket pkt;
+  int i;
+
+  for (i = 0; i < NUMTEAM; i++)
+  {
+    pkt.type = SP_GPARAM;
+    pkt.subtype = 1;
+
+    pkt.index = i;
+    pkt.letter = teams[idx_to_mask(i)].letter;
+    strncpy(pkt.shortname, teams[idx_to_mask(i)].shortname,
+	    sizeof(pkt.shortname));
+    strncpy(pkt.teamname, teams[idx_to_mask(i)].name, sizeof(pkt.teamname));
+
+    sendClientPacket((struct player_spacket *) & pkt);
+  }
+}
+
+static void
+send_one_teamlogo(teamidx, data, width, height)
+  int teamidx;
+  unsigned char *data;
+  int width, height;
+{
+  struct gp_teamlogo_spacket pkt;
+  int pwidth;
+
+  if (width > 99 || height > 99)
+  {
+    printf("logo too big: %dx%d\n", width, height);
+    return;
+  }
+
+  pkt.type = SP_GPARAM;
+  pkt.subtype = 2;
+
+  pkt.logowidth = width;
+  pkt.logoheight = height;
+  pkt.teamindex = teamidx;
+
+  pwidth = (width - 1) / 8 + 1;
+  pkt.thisheight = sizeof(pkt.data) / pwidth;
+
+  for (pkt.y = 0; pkt.y < height; pkt.y += pkt.thisheight)
+  {
+
+    if (pkt.y + pkt.thisheight > height)
+      pkt.thisheight = height - pkt.y;
+
+    memcpy(pkt.data, data + pkt.y * pwidth, pkt.thisheight * pwidth);
+    sendClientPacket((struct player_spacket *) & pkt);
+  }
+}
+
+void
+updateGPteamlogos()
+{
+  char buf[40];
+  char *data;
+  int width, height;
+  int i;
+
+  for (i = 0; i < NUMTEAM; i++)
+  {
+    sprintf(buf, "artwork/%d/logo", i);
+#ifdef LEAGUE_SUPPORT
+    if (status2->league == 1)
+    {
+      if (i == 0)
+	sprintf(buf, "artwork/%s/logo", status2->home.name);
+      else if (i == 1)
+	sprintf(buf, "artwork/%s/logo", status2->away.name);
+    }
+    else if (status2->league)
+    {
+      if (i == status2->home.index)
+	sprintf(buf, "artwork/%s/logo", status2->home.name);
+      else if (i == status2->away.index)
+	sprintf(buf, "artwork/%s/logo", status2->away.name);
+    }
+#endif
+    {
+      FILE *fp;
+      fp = fopen(build_path(buf), "r");
+      if (!fp)
+	continue;		/* no image to transmit */
+      ParseXbmFile(fp, &width, &height, &data);
+      fclose(fp);
+    }
+    if (!data)
+    {
+      continue;
+    }
+    send_one_teamlogo(i, data, width, height);
+    free(data);
+  }
+}
+
+/* #include "borgcube.bm" */
+
+void
+updateGPshipshapes()
+{
+#if 0
+  {
+    struct gp_shipshape_spacket pkt;
+
+    pkt.type = SP_GPARAM;
+    pkt.subtype = 3;
+
+    pkt.shipno = ATT;
+    pkt.race = -1;
+    pkt.nviews = 1;
+    pkt.width = borgcube_width;
+    pkt.height = borgcube_height;
+
+    sendClientPacket((struct player_spacket *) & pkt);
+  }
+
+  {
+    struct gp_shipbitmap_spacket pkt;
+
+    pkt.type = SP_GPARAM;
+    pkt.subtype = 4;
+
+    pkt.shipno = ATT;
+    pkt.race = -1;
+    pkt.thisview = 0;
+
+    memcpy(pkt.bitmapdata, borgcube_bits, sizeof(borgcube_bits));
+
+    sendClientPacket((struct player_spacket *) & pkt);
+  }
+#endif
+}
+
+void
+updateGPplanetbitmaps()
+{
+  struct gp_teamplanet_spacket pkt;
+  char buf[40];
+  char *data;
+  int width, height;
+  int i;
+
+  for (i = 0; i < NUMTEAM; i++)
+  {
+
+    pkt.type = SP_GPARAM;
+    pkt.subtype = 7;
+
+    pkt.teamn = i;
+
+    sprintf(buf, "artwork/%d/tplanet", i);
+#ifdef LEAGUE_SUPPORT
+    if (status2->league == 1)
+    {
+      if (i == 0)
+	sprintf(buf, "artwork/%s/tplanet", status2->home.name);
+      else if (i == 1)
+	sprintf(buf, "artwork/%s/tplanet", status2->away.name);
+    }
+    else if (status2->league)
+    {
+      if (i == status2->home.index)
+	sprintf(buf, "artwork/%s/tplanet", status2->home.name);
+      else if (i == status2->away.index)
+	sprintf(buf, "artwork/%s/tplanet", status2->away.name);
+    }
+#endif
+    {
+      FILE *fp;
+      fp = fopen(build_path(buf), "r");
+      if (!fp)
+	continue;		/* no image to transmit */
+      ParseXbmFile(fp, &width, &height, &data);
+      fclose(fp);
+    }
+    if (!data)
+    {
+      continue;
+    }
+    memcpy(pkt.tactical, data, 120);
+    memcpy(pkt.tacticalM, data, 120);
+    free(data);
+
+    sprintf(buf, "artwork/%d/gplanet", i);
+#ifdef LEAGUE_SUPPORT
+    if (status2->league == 1)
+    {
+      if (i == 0)
+	sprintf(buf, "artwork/%s/gplanet", status2->home.name);
+      else if (i == 1)
+	sprintf(buf, "artwork/%s/gplanet", status2->away.name);
+    }
+    else if (status2->league)
+    {
+      if (i == status2->home.index)
+	sprintf(buf, "artwork/%s/gplanet", status2->home.name);
+      else if (i == status2->away.index)
+	sprintf(buf, "artwork/%s/gplanet", status2->away.name);
+    }
+#endif
+    {
+      FILE *fp;
+      fp = fopen(build_path(buf), "r");
+      if (!fp)
+	continue;		/* no image to transmit */
+      ParseXbmFile(fp, &width, &height, &data);
+      fclose(fp);
+    }
+    if (!data)
+    {
+      continue;
+    }
+    memcpy(pkt.galactic, data, 32);
+    memcpy(pkt.galacticM, data, 32);
+    free(data);
+
+    sendClientPacket((struct player_spacket *) & pkt);
+  }
+
+}
+
+void
+updateGameparams()
+{
+  struct gp_sizes_spacket pkt;
+
+  memset((char *) &pkt, 0, sizeof(pkt));
+
+  pkt.type = SP_GPARAM;
+  pkt.subtype = 0;
+
+  pkt.nplayers = MAXPLAYER;
+  pkt.nteams = 4;
+  pkt.nshiptypes = NUM_TYPES;
+  pkt.nranks = NUMRANKS;
+  pkt.nroyal = NUMROYALRANKS;
+  pkt.nphasers = 1;
+  pkt.ntorps = MAXTORP;
+  pkt.nplasmas = MAXPLASMA;
+  pkt.nthingies = NPTHINGIES;
+  pkt.gthingies = NGTHINGIES;
+  pkt.gwidth = GWIDTH;
+  pkt.flags = 0;
+  pkt.nplanets = configvals->numplanets;
+  sendClientPacket((struct player_spacket *) & pkt);
+
+  updateGPteams();
+  if (me == 0 || !(me->p_stats.st_flags & ST_NOBITMAPS))
+  {
+    updateGPteamlogos();
+    updateGPshipshapes();
+    updateGPplanetbitmaps();
+  }
+  updateGPrank();
+  updateGProyal();
+}
+
+/* end game params stuff */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/getentry.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,421 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <stdio.h>
+#include <sys/types.h>
+#include <sys/time.h>
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "packets.h"
+#include "shmem.h"
+
+extern int overload;		/* 7/31/91 TC let overloaded join any team */
+
+extern int updateShips();
+extern int sendMaskPacket();
+int tournamentMask();
+extern int briefUpdateClient();
+extern int flushSockBuf();
+extern int socketPause();
+extern int readFromClient();
+extern int isClientDead();
+extern int exitGame();
+#ifndef	IRIX
+extern int printf();
+#endif
+extern int closeUdpConn();
+extern unsigned int sleep();
+extern int connectToClient();
+extern int updateSelf();
+extern int sendPickokPacket();
+extern int allowed_ship();
+extern int time_access();
+int deadTeam();
+int realNumShips();
+
+#if 1
+
+void
+detourneyqueue()
+{
+  me->p_status = POUTFIT;
+  me->p_whydead = KPROVIDENCE;
+}
+
+#endif
+
+
+void
+getEntry(team, stype)
+  int *team;
+  int *stype;
+{
+
+
+  int switching = -1;		/* confirm switches 7/27/91 TC */
+  inputMask = CP_OUTFIT;
+  for (;;)
+  {
+    /* updateShips so he knows how many players on each team */
+    if (blk_flag)
+      updateShips();
+    sendMaskPacket(tournamentMask(me->p_team));
+    if (blk_flag)
+      briefUpdateClient();
+    flushSockBuf();
+    /* Have we been busted? */
+    if (me->p_status == PFREE)
+    {
+      me->p_status = PDEAD;
+      me->p_explode = 600;
+    }
+    socketPause();
+    readFromClient();
+    if (isClientDead())
+    {
+      int i;
+
+      if (noressurect)
+	exitGame();
+      printf("Ack!  The client went away!\n");
+      printf("I will attempt to resurrect him!\n");
+
+      /* UDP fail-safe */
+      commMode = COMM_TCP;
+      if (udpSock >= 0)
+	closeUdpConn();
+
+      /* For next two minutes, we try to restore connection */
+      shutdown(sock, 2);
+      sock = -1;
+      for (i = 0;; i++)
+      {
+	switch (me->p_status)
+	{
+	 case PFREE:
+	  me->p_status = PDEAD;
+	  me->p_explode = 600;
+	  break;
+	 case PALIVE:
+	 case POBSERVE:
+	  me->p_ghostbuster = 0;
+	  break;
+	 case PDEAD:
+	  me->p_explode = 600;
+	  break;
+	 default:
+	  me->p_explode = 600;
+	  me->p_ghostbuster = 0;
+	  break;
+	}
+	sleep(5);
+	if (connectToClient(host, nextSocket))
+	  break;
+	if (i == 23)
+	{
+	  printf("Oh well, maybe I'm getting rusty!\n");
+	  switch (me->p_status)
+	  {
+	   case PFREE:
+	    break;
+	   case PALIVE:
+	   case POBSERVE:
+	    me->p_ghostbuster = 100000;
+	    break;
+	   case PDEAD:
+	    me->p_explode = 0;
+	    break;
+	   default:
+	    me->p_explode = 0;
+	    me->p_ghostbuster = 100000;
+	    break;
+	  }
+	  exitGame();
+	}
+      }
+      printf("A miracle!  He's alive!\n");
+      teamPick = -1;
+      updateSelf();
+      updateShips();
+      flushSockBuf();
+    }
+    if (teamPick != -1)
+    {
+      if (teamPick < 0 || teamPick > 3)
+      {
+	warning("Get real!");
+	sendPickokPacket(0);
+	teamPick = -1;
+	continue;
+      }
+#if 1
+      if (!(tournamentMask(me->p_team) & (1 << teamPick)))
+      {
+	warning("I cannot allow that.  Pick another team");
+	sendPickokPacket(0);
+	teamPick = -1;
+	continue;
+      }
+#endif
+
+      if (((1 << teamPick) != me->p_team) &&
+	  (me->p_team != ALLTEAM))	/* switching teams 7/27/91 TC */
+	if ((switching != teamPick)
+	    && (me->p_whydead != KGENOCIDE)
+#ifdef LEAGUE_SUPPORT
+	    && !status2->league
+#endif
+	  )
+	{
+	  switching = teamPick;
+	  warning("Please confirm change of teams.  Select the new team again.");
+	  sendPickokPacket(0);
+	  teamPick = -1;
+	  continue;
+	}
+      /* His team choice is ok. */
+      if (shipPick < 0 || shipPick >= NUM_TYPES)
+      {
+	warning("That is an illegal ship type.  Try again.");
+	sendPickokPacket(0);
+	teamPick = -1;
+	continue;
+      }
+      if (!allowed_ship(1 << teamPick, mystats->st_rank, mystats->st_royal, shipPick))
+      {
+	sendPickokPacket(0);
+	teamPick = -1;
+	continue;
+      }
+      if (shipvals[shipPick].s_nflags & SFNMASSPRODUCED)
+      {
+	warning("o)utpost, u)tility, or s)tandard?");
+	sendPickokPacket(0);
+	tmpPick = shipPick;
+	continue;
+      }
+      break;
+    }
+#if 1
+    if (me->p_status == PTQUEUE)
+    {
+      if (status->tourn)
+	detourneyqueue();
+      /* You don't time out on the tourney queue */
+      me->p_ghostbuster = 0;
+    }
+#endif
+  }
+  *team = teamPick;
+  if ((shipvals[tmpPick].s_nflags & SFNMASSPRODUCED) &&
+      ((shipvals[shipPick].s_numports) || (shipPick == SCOUT)))
+  {
+
+    *stype = tmpPick;
+    tmpPick = CRUISER;
+  }
+  else
+  {
+    *stype = shipPick;
+    tmpPick = CRUISER;
+  }
+  sendPickokPacket(1);
+  flushSockBuf();
+}
+
+
+
+
+/*------------------------------TOURNAMENTMASK----------------------------*/
+/*
+ * This function takes the players current team and returns a mask of the
+ * teams that the player can join.
+ */
+
+int
+leaguemask(ishome, idx)
+  int ishome;			/* team index */
+  int idx;
+{
+#ifdef LEAGUE_SUPPORT
+  if (status2->league == 1)
+    return idx_to_mask(ishome);
+  else
+#endif
+    return idx_to_mask(idx);
+}
+
+int
+tournamentMask(team)
+  int team;			/* players current team */
+{
+  int i;			/* looping var */
+  int team1, team2;		/* for getting two possible teams */
+  int num1, num2;		/* to hold num players on teams */
+
+  if (!blk_flag)		/* not a paradise client */
+    return (0);
+  if (status->gameup == 0)	/* if game over, can join no team */
+    return (0);
+  if (overload)			/* if tester slot then can join */
+    return (ALLTEAM);		/* any team */
+  if (mustexit)			/* should we force player out */
+    return (0);
+  if (!time_access())		/* if server closed then can join */
+    return (0);			/* no team */
+
+#ifdef LEAGUE_SUPPORT
+  /* league stuff */
+  if (status2->league)
+  {
+    if (me->p_homeaway == AWAY)
+      return leaguemask(1, status2->away.index);
+    else if (me->p_homeaway == HOME)
+      return leaguemask(0, status2->home.index);
+    else
+    {
+      warning("You have connected to a league server but aren't on a side.");
+      me->p_homeaway = (lrand48() & 1) + 1;	/* dangerous int->enum cast */
+      return 0;
+    }
+    /* NOTREACHED */
+  }
+#endif
+  if (me->p_homeaway != NEITHER)
+  {
+    warning("You have connected to a non-league server with a league ntserv.");
+    me->p_homeaway = NEITHER;
+  }
+
+  if (!status->tourn)
+  {
+    return status2->nontteamlock;
+  }
+
+  if (team != ALLTEAM && deadTeam(team))	/* if former team dead */
+    team = ALLTEAM;		/* then check all others */
+  for (i = 0; i < NUMTEAM; i++)
+  {				/* go through all teams and eliminate */
+    if ((team & (1 << i)) && (deadTeam(1 << i)))	/* from team var all
+							 * teams */
+      team &= ~(1 << i);	/* that are dead */
+  }
+  team1 = 0;			/* no team in team 1 */
+  num1 = 0;			/* 0 players on team 1 */
+  team2 = 0;			/* no team in team 2 */
+  num2 = 0;			/* 0 players on team 2 */
+  for (i = 0; i < NUMTEAM; i++)
+  {				/* go through all teams */
+    if (deadTeam(1 << i))	/* if team is dead then */
+      continue;			/* disregard it */
+    if (realNumShips(1 << i) >= configvals->tournplayers)
+    {				/* enough players */
+      if (!team1)		/* if no team in team1 yet */
+	team1 = (1 << i);	/* then this will be team 1 */
+      else
+      {				/* otherwise its team2 */
+	team2 = (1 << i);
+	num1 = realNumShips(team1);	/* get num players on two teams */
+	num2 = realNumShips(team2);
+	if (num1 == num2)
+	{			/* if teams same size then */
+	  if (team & (team1 | team2))	/* if player on one team */
+	    return (team & (team1 | team2));	/* let him join same team */
+	  return (team1 | team2);	/* otherwise, he can join either */
+	}
+	else if ((num1 > num2) && (team != team1))
+	  return (team2);	/* if slight imabalance */
+	else if ((num1 < num2) && (team != team2))
+	  return (team1);	/* if slight imbalance */
+	else
+	{
+	  if (ABS(num1 - num2) < 2 || (((num1 > num2) && (team == team2)) ||
+				       (num2 > num1 && team == team1)))
+	    return (team);
+	  else
+	    return (team1 | team2);
+	}
+      }
+    }
+  }				/* end of for loop */
+  return (team);		/* just in case */
+}
+
+
+
+
+/*--------------------------------REALNUMSHIPS-----------------------------*/
+/*
+ * Count the real number of ships on a team.  This function returns a count
+ * of all the ships on a team that are not PFREE.
+ */
+
+int
+realNumShips(owner)
+  int owner;			/* the team to check for */
+{
+  int i;			/* looping var */
+  int num;			/* to hold ship count */
+  struct player *p;		/* to point to players */
+
+  num = 0;			/* initialize count */
+  for (i = 0, p = players; i < MAXPLAYER; i++, p++)
+  {
+    if ((p->p_status != PFREE) &&
+	!(p->p_flags & (PFSNAKE | PFBIRD)) &&
+	(p->p_team == owner))
+    {
+      num++;			/* if ship not free and on our team */
+    }
+  }
+  return (num);			/* retun number of ships */
+}
+
+
+
+
+/*----------------------------------DEADTEAM-------------------------------*/
+/*
+ * This function counts the number of planets that a team owns.  It returns a
+ * 1 if the team has no planets, and a 0 if they have at least one planet.
+ */
+
+int
+deadTeam(owner)
+  int owner;			/* team to check for */
+{
+  int i;			/* looping var */
+  struct planet *p;		/* to point to a planets */
+
+  for (i = 0, p = planets; i < NUMPLANETS; i++, p++)
+  {
+    if (p->pl_owner & owner)	/* if planet owned by team then */
+      return (0);		/* found one planet owned by team */
+  }
+  return (1);			/* no planets found, team is dead */
+}
+
+/*-------------------------------------------------------------------------*/
+
+
+
+
+
+/*---------END OF FILE--------*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/getname.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,287 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <stdio.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <sys/file.h>
+#include <sys/ipc.h>
+#include <sys/shm.h>
+#include <sys/time.h>
+#ifndef ULTRIX
+#include <sys/fcntl.h>
+#endif
+#include <errno.h>
+#include <pwd.h>
+#include <string.h>
+#include <ctype.h>
+#include <time.h>
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "packets.h"
+#include "path.h"
+#include "shmem.h"
+
+#define crypt(a, b) (a)
+
+void handleLogin();
+extern int exitGame();
+extern int isClientDead();
+extern int socketPause();
+extern int readFromClient();
+int lockout();
+extern int sendClientLogin();
+extern int flushSockBuf();
+#ifdef BAD_SVR4_HACKS
+int updateMOTD();
+#else
+extern int updateMOTD();
+#endif
+#ifndef	IRIX
+extern int printf();
+#endif
+extern int read();
+extern int close();
+extern off_t lseek();
+extern int write();
+extern time_t time();
+
+/* replace non-printable characters in string with spaces */
+static void
+remove_ctl(str)
+  char *str;
+{
+  while (*str)
+  {
+    if (!isgraph(*str))		/* not a printable character */
+      *str = ' ';		/* replace it with space */
+    str++;
+  }
+}
+
+void
+getname()
+/* Let person identify themselves from w */
+{
+  *(me->p_name) = 0;
+  while (*(me->p_name) == 0)
+  {
+    handleLogin();
+  }
+}
+
+void
+handleLogin()
+{
+  static struct statentry player;
+  static int position = -1;
+  int plfd;
+  int entries;
+  struct stat buf;
+  char *paths;
+
+  paths = build_path(PLAYERFILE);
+  *namePick = '\0';
+  inputMask = CP_LOGIN;
+  while (*namePick == '\0')
+  {
+    /* Been ghostbusted? */
+    if (me->p_status == PFREE)
+      exitGame();
+    if (isClientDead())
+      exitGame();
+    socketPause();
+    readFromClient();
+  }
+#ifdef REMOVE_CTL		/* untested */
+  /* Change those annoying control characters to spaces */
+  remove_ctl(namePick);
+#endif
+  if ((strcmp(namePick, "Guest") == 0 || strcmp(namePick, "guest") == 0) &&
+      !lockout())
+  {
+    hourratio = 5;
+    memset(&player.stats, 0, sizeof(struct stats));
+    player.stats.st_tticks = 1;
+    player.stats.st_flags = ST_INITIAL;
+    /*
+     * If this is a query on Guest, the client is screwed, but I'll send him
+     * some crud anyway.
+     */
+    if (passPick[15] != 0)
+    {
+      sendClientLogin(&player.stats);
+      flushSockBuf();
+      return;
+    }
+    sendClientLogin(&player.stats);
+
+    updateMOTD();		/* added here 1/19/93 KAO */
+
+    flushSockBuf();
+    strcpy(me->p_name, namePick);
+    me->p_pos = -1;
+    memcpy(&(me->p_stats), &player.stats, sizeof(struct stats));
+    return;
+  }
+  hourratio = 1;
+  /* We look for the guy in the stat file */
+  if (strcmp(player.name, namePick) != 0)
+  {
+    for (;;)
+    {				/* so I can use break; */
+      plfd = open(paths, O_RDONLY, 0644);
+      if (plfd < 0)
+      {
+	printf("I cannot open the player file!\n");
+	strcpy(player.name, namePick);
+	position = -1;
+	printf("Error number: %d\n", errno);
+	break;
+      }
+      position = 0;
+      while (read(plfd, (char *) &player, sizeof(struct statentry)) ==
+	     sizeof(struct statentry))
+      {
+#ifdef REMOVE_CTL		/* untested */
+	/*
+	 * this is a temporary thing to remove control chars from existing
+	 * entries in the player db
+	 */
+	remove_ctl(player.name);
+#endif
+	if (strcmp(namePick, player.name) == 0)
+	{
+	  close(plfd);
+	  break;
+	}
+	position++;
+      }
+      if (strcmp(namePick, player.name) == 0)
+	break;
+      close(plfd);
+      position = -1;
+      strcpy(player.name, namePick);
+      break;
+    }
+  }
+  /* Was this just a query? */
+  if (passPick[15] != 0)
+  {
+    if (position == -1)
+    {
+      sendClientLogin(NULL);
+    }
+    else
+    {
+      sendClientLogin(&player.stats);
+    }
+    flushSockBuf();
+    return;
+  }
+  /* A new guy? */
+  if ((position == -1) && !lockout())
+  {
+    strcpy(player.name, namePick);
+    strcpy(player.password, crypt(passPick, namePick));
+    memset(&player.stats, 0, sizeof(struct stats));
+    player.stats.st_tticks = 1;
+    player.stats.st_flags = ST_INITIAL;
+    player.stats.st_royal = 0;
+    plfd = open(paths, O_RDWR | O_CREAT, 0644);
+    if (plfd < 0)
+    {
+      sendClientLogin(NULL);
+    }
+    else
+    {
+      fstat(plfd, &buf);
+      entries = buf.st_size / sizeof(struct statentry);
+      lseek(plfd, entries * sizeof(struct statentry), 0);
+      write(plfd, (char *) &player, sizeof(struct statentry));
+      close(plfd);
+      me->p_pos = entries;
+      memcpy(&(me->p_stats), &player.stats, sizeof(struct stats));
+      strcpy(me->p_name, namePick);
+      sendClientLogin(&player.stats);
+      updateMOTD();		/* added here 1/19/93 KAO */
+    }
+    flushSockBuf();
+    return;
+  }
+  /* An actual login attempt */
+  if ((strcmp(player.password, crypt(passPick, player.password)) != 0) ||
+      lockout())
+  {
+    sendClientLogin(NULL);
+  }
+  else
+  {
+    strcpy(me->p_name, namePick);
+    me->p_pos = position;
+    memcpy(&(me->p_stats), &player.stats, sizeof(struct stats));
+
+    sendClientLogin(&player.stats);
+
+    updateMOTD();		/* st_flags&ST_NOBITMAPS */
+  }
+  flushSockBuf();
+
+#if 1
+  /* Try to make the first person in the player database an Emperor */
+  if (position == 0)
+    me->p_stats.st_royal = NUMROYALRANKS - 1;
+#endif
+
+  return;
+}
+
+void
+savestats()
+{
+  int fd;
+  char *paths;
+
+  if (me->p_pos < 0)
+    return;
+
+  paths = build_path(PLAYERFILE);
+
+  fd = open(paths, O_WRONLY, 0644);
+  if (fd >= 0)
+  {
+    me->p_stats.st_lastlogin = time(NULL);
+    lseek(fd, 32 + me->p_pos * sizeof(struct statentry), 0);
+    write(fd, (char *) &me->p_stats, sizeof(struct stats));
+    close(fd);
+  }
+}
+
+/* return true if we want a lockout */
+int
+lockout()
+{
+  return (
+  /*
+   * (strncmp (login, "bozo", 4) == 0) ||
+   */
+	  0
+  );
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/getship.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,114 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+
+#include "config.h"
+#include <stdio.h>
+#include <sys/types.h>
+#include <sys/ipc.h>
+#include <sys/shm.h>
+#include <string.h>
+
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "shmem.h"
+
+
+
+
+/*-------------------------------INTERNAL FUNCTONS------------------------*/
+
+/*----------------------------------GETSHIP--------------------------------*/
+/*
+ * This function gets the particular ship type the player wants.  It takes
+ * the ship values from the shipvals array.
+ */
+
+void
+getship(shipp, s_type)
+  struct ship *shipp;		/* the ship structure to load */
+  int s_type;			/* the type of ship to get */
+{
+  memcpy((char *) shipp, (char *) &(shipvals[s_type]), sizeof(struct ship));
+}
+
+
+
+/*------------------------------------------------------------------------*/
+
+
+
+
+
+void
+get_ship_for_player(me, s_type)
+  struct player *me;
+  int s_type;
+{
+  getship(&me->p_ship, s_type);
+
+  me->p_shield = me->p_ship.s_maxshield;	/* shields are at max */
+  me->p_damage = 0;		/* no damage to ship */
+  me->p_subdamage = 0;		/* no fractional damage either */
+  me->p_subshield = 0;		/* no fractional damage to shield */
+
+  me->p_fuel = me->p_ship.s_maxfuel;	/* fuel is at max */
+  me->p_etemp = 0;		/* engines are ice cold */
+  me->p_etime = 0;		/* not counting down for E-temp */
+  me->p_warptime = 0;		/* we are not preparing for warp */
+  me->p_wtemp = 0;		/* weapons cool too */
+  me->p_wtime = 0;		/* not counting down for W-temp */
+
+  if (allows_docking(me->p_ship))
+  {				/* if ship can be docked to */
+    int i;
+    me->p_docked = 0;		/* then set all ports as */
+    for (i = 0; i < MAXPORTS; i++)	/* vacant */
+      me->p_port[i] = VACANT;
+  }
+  if (weaponsallowed[WP_MISSILE] &&
+      (me->p_ship.s_missilestored > 0) &&
+      (me->p_kills >= configvals->mskills))
+  {
+    me->p_ship.s_nflags |= SFNHASMISSILE;	/* arm ship with missile
+						 * launcher */
+  }
+  if (!(me->p_ship.s_nflags & SFNHASMISSILE))
+  {
+    me->p_ship.s_missilestored = 0;	/* no missiles if no launcher */
+  }
+  if (weaponsallowed[WP_PLASMA] &&
+      (me->p_ship.s_plasma.cost > 0) &&
+      (me->p_kills >= configvals->plkills))
+  {
+    me->p_ship.s_nflags |= SFNPLASMAARMED;	/* arm ship with plasma
+						 * launcher */
+  }
+  me->p_specweap = 0;
+
+  /*
+   * fix this now since we can't do it right in a conf.sysdef/default/pl_gen
+   * ordering... sigh.
+   */
+  if (!configvals->warpdrive)
+    me->p_ship.s_nflags &= ~SFNCANWARP;
+}
+
+
+/*----------END OF FILE-----*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/getship.h	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,30 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+
+
+/*---------------------------FUNCTION PROTOTYPES--------------------------*/
+void initshipvals();
+int getship();
+/*------------------------------------------------------------------------*/
+
+
+
+
+
+/*-------END OF FILE-------*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/getstats.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,104 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <stdio.h>
+#include <stdlib.h>
+
+struct stats
+{
+  int st_genocides;		/* number of genocides participated in */
+  float st_tmaxkills;		/* max kills ever */
+  float st_di;			/* total destruction inflicted for all time */
+  int st_tkills;		/* Kills in tournament play */
+  int st_tlosses;		/* Losses in tournament play */
+  int st_tarmsbomb;		/* Tournament armies bombed */
+  int st_tresbomb;		/* resources bombed off */
+  int st_tdooshes;		/* armies killed while being carried */
+  int st_tplanets;		/* Tournament planets conquered */
+  int st_tticks;		/* Tournament ticks */
+  /* SB/WB/JS stats are entirely separate */
+  int st_sbkills;		/* Kills as starbase */
+  int st_sblosses;		/* Losses as starbase */
+  int st_sbticks;		/* Time as starbase */
+  float st_sbmaxkills;		/* Max kills as starbase */
+  int st_wbkills;		/* Kills as warbase */
+  int st_wblosses;		/* Losses as warbase */
+  int st_wbticks;		/* Time as warbase */
+  float st_wbmaxkills;		/* Max kills as warbase */
+  int st_jsplanets;		/* planets assisted with in JS */
+  int st_jsticks;		/* ticks played as a JS */
+  long st_lastlogin;		/* Last time this player was played */
+  int st_flags;			/* Misc option flags */
+  char st_keymap[96];		/* keymap for this player */
+  int st_rank;			/* Ranking of the player */
+  int st_royal;			/* royaly, specialty, rank */
+};
+
+struct statentry
+{
+  char name[16];		/* player's name */
+  char password[16];		/* player's password */
+  struct stats stats;		/* player's stats */
+};
+
+
+main(argn, argv)
+  int argn;
+  char **argv;
+{
+  FILE *f;
+  struct statentry s;
+
+  f = fopen(argv[1], "r");
+  if (f == NULL)
+  {
+    printf("Cannot open players file\n");
+    exit(1);
+  }
+  while (fread(&s, sizeof(struct statentry), 1, f) == 1)
+  {
+    printf("\nPlayer: %s\n", s.name);
+    printf("Genocides: %d\n", s.stats.st_genocides);
+    printf("Maxkills: %f\n", s.stats.st_tmaxkills);
+    printf("DI: %f\n", s.stats.st_di);
+    printf("Kills: %d\n", s.stats.st_tkills);
+    printf("Losses: %d\n", s.stats.st_tlosses);
+    printf("Armies bombed: %d\n", s.stats.st_tarmsbomb);
+    printf("Resources bombed: %d\n", s.stats.st_tresbomb);
+    printf("Dooshes: %d\n", s.stats.st_tdooshes);
+    printf("Planets: %d\n", s.stats.st_tplanets);
+    printf("Time: %f\n", (float) s.stats.st_tticks / 36000.0);
+    printf("Rank: %d\n", s.stats.st_rank);
+    printf("Royalty: %d\n", s.stats.st_royal);
+
+    printf("SB kills: %d\n", s.stats.st_sbkills);
+    printf("SB losses: %d\n", s.stats.st_sblosses);
+    printf("SB time: %f\n", (float) s.stats.st_sbticks / 36000.0);
+    printf("SB maxkills: %f\n", s.stats.st_sbmaxkills);
+
+    printf("WB kills: %d\n", s.stats.st_wbkills);
+    printf("WB losses: %d\n", s.stats.st_wblosses);
+    printf("WB time: %f\n", (float) s.stats.st_wbticks / 36000.0);
+    printf("WB maxkills: %f\n", s.stats.st_wbmaxkills);
+
+    printf("JS planets: %f\n", s.stats.st_jsplanets);
+    printf("JS time: %f\n", (float) s.stats.st_jsticks / 36000.0);
+  }
+  fclose(f);
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/gppackets.h	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,191 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#ifndef gppackets_h_
+#define gppackets_h_
+
+/* the definitions of {INT,CARD}{8,16,32} are in packets.h */
+#include "packets.h"
+
+struct gameparam_spacket
+{
+  INT8 type;
+  INT8 subtype;			/* this packet is not real */
+  /* generic game parameter packet */
+  INT16 pad;
+};
+
+struct gp_sizes_spacket
+{
+  INT8 type;
+  INT8 subtype;			/* =0 */
+
+  CARD8 nplayers;
+  CARD8 nteams;			/* max of 8 */
+
+  CARD8 nshiptypes;
+  CARD8 nranks;			/* number of ranks */
+  CARD8 nroyal;			/* number of royalties */
+  CARD8 nphasers;
+
+  CARD8 ntorps;
+  CARD8 nplasmas;
+  CARD8 nthingies;		/* per-player */
+  CARD8 gthingies;		/* auxiliary thingies */
+
+  CARD32 gwidth;		/* galaxy width */
+  /* 16 bytes */
+  CARD32 flags;			/* some game parameter flags */
+#define	GP_WRAPVERT	(1<<0)
+#define	GP_WRAPHORIZ	(1<<1)
+#define GP_WRAPALL	(GP_WRAPVERT|GP_WRAPHORIZ)
+
+  /*
+   * The following bytes are room for growth. The size of this packet is
+   * unimportant because it only gets sent once.  hopefully we've got plenty
+   * of room.
+   */
+  CARD16 nplanets;
+  INT16 ext1;
+
+  INT32 ext2;
+  INT32 ext3;
+  /* 32 bytes */
+
+  INT32 ext4;
+  INT32 ext5;
+  INT32 ext6;
+  INT32 ext7;
+
+  INT32 ext8;
+  INT32 ext9;
+  INT32 ext10;
+  INT32 ext11;			/* 16 ints, 64 bytes */
+};
+
+struct gp_team_spacket
+{
+  INT8 type;
+  INT8 subtype;			/* =1 */
+
+  CARD8 index;			/* team index */
+  CARD8 letter;			/* team letter 'F' */
+
+  CARD8 shortname[3];		/* non-null-terminated 3-letter abbrev 'FED' */
+  CARD8 pad;
+  /* 8 bytes */
+  CARD8 teamname[56];		/* padding to 64 byte packet */
+};
+
+struct gp_teamlogo_spacket
+{
+  /*
+   * This packet contains several adjacent rows of a team's logo bitmap Data
+   * is in raw XBM format (scanline-padded to 8 bits). Maximum bitmap size is
+   * 99x99, which takes 1287 (99x13) bytes.
+   */
+  INT8 type;
+  INT8 subtype;			/* =2 */
+
+  INT8 logowidth;		/* <= 99 */
+  INT8 logoheight;		/* <= 99 */
+
+  INT8 y;			/* y coord of the start of this packets info */
+  INT8 thisheight;		/* the height of this packet's info */
+  INT8 teamindex;		/* which team's logo this is */
+
+  CARD8 data[768 - 7];		/* pad packet to 768 bytes. */
+};
+
+struct gp_shipshape_spacket
+{
+  INT8 type;
+  INT8 subtype;			/* =3 */
+
+  CARD8 shipno;
+  INT8 race;			/* -1 is independent */
+  CARD8 nviews;			/* typically 16 */
+
+  CARD8 width, height;
+  CARD8 pad1;
+};
+
+struct gp_shipbitmap_spacket
+{
+  INT8 type;
+  INT8 subtype;			/* =4 */
+
+  CARD8 shipno;
+  INT8 race;			/* -1 is independent */
+  CARD8 thisview;		/* 0..nviews-1 */
+
+  CARD8 bitmapdata[999];
+};
+
+struct gp_rank_spacket
+{
+  INT8 type;
+  INT8 subtype;			/* =5 */
+
+  INT8 rankn;			/* rank number */
+
+  CARD8 name[-3 + 64 - 20];	/* name of the rank */
+
+  INT32 genocides;
+  INT32 milliDI;		/* DI*1000 */
+  INT32 millibattle;		/* battle*1000 */
+  INT32 millistrat;		/* strategy*1000 */
+  INT32 millispec;		/* special ships*1000 */
+};
+
+struct gp_royal_spacket
+{
+  INT8 type;
+  INT8 subtype;			/* =6 */
+
+  CARD8 rankn;			/* rank number */
+
+  CARD8 name[-3 + 64];		/* name of the rank */
+};
+
+struct gp_teamplanet_spacket
+{
+  INT8 type;
+  INT8 subtype;			/* =7 */
+
+  INT8 teamn;			/* 0..7 */
+  CARD8 pad1;
+
+  INT32 ext1;			/* extensions? */
+
+  /*
+   * Bitmaps of the team logo and masks.  The bitmap of the planet will be
+   * constructed with (mask ? logo : planet), applied bitwise. This
+   * calculation is equivalent to (logo&mask)|(planet&~mask)
+   */
+
+  /* bitmap 30x30, X bitmap format (scanline padded to 8 bits) */
+  CARD8 tactical[120];
+  CARD8 tacticalM[120];
+
+  /* bitmap 16x16, X bitmap format (scanline padded to 8 bits) */
+  CARD8 galactic[32];
+  CARD8 galacticM[32];
+};
+
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/grid.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,98 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include "struct.h"
+#include "shmem.h"
+
+/*
+ * This function should be called any time a planet is relocated. It is in
+ * charge of updating the space grid.
+ */
+void
+move_planet(pno, x, y, isold)
+  int pno;			/* planet number */
+  int x, y;
+  int isold;			/* 0 if the planet has not yet been entered
+				 * into the grid */
+{
+  struct planet *pl = &planets[pno];
+
+  if (isold)
+  {
+    /* remove from previous space grid position */
+  }
+  pl->pl_x = x;
+  pl->pl_y = y;
+
+  /* enter into space grid */
+}
+
+void
+move_player(pno, x, y, isold)
+  int pno;			/* player number */
+  int x, y;
+  int isold;			/* 0 if the player has not yet been entered
+				 * into the grid */
+{
+  struct player *pl = &players[pno];
+
+  if (isold)
+  {
+  }
+  pl->p_x = x;
+  pl->p_y = y;
+
+  /* enter into space grid */
+}
+
+void
+move_torp(tno, x, y, isold)
+  int tno;			/* torp number */
+  int x, y;
+  int isold;			/* 0 if the torp has not yet been entered
+				 * into the grid */
+{
+  struct torp *t = &torps[tno];
+
+  if (isold)
+  {
+  }
+  t->t_x = x;
+  t->t_y = y;
+
+  /* enter into space grid */
+}
+
+void
+move_missile(dno, x, y, isold)
+  int dno;			/* missile number */
+  int x, y;
+  int isold;			/* 0 if the missile has not yet been entered
+				 * into the grid */
+{
+  struct missile *d = &missiles[dno];
+
+  if (isold)
+  {
+  }
+  d->ms_x = x;
+  d->ms_y = y;
+
+  /* enter into space grid */
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/grid.h	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,32 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#ifndef grid_h_
+#define grid_h_
+
+/* specifying an x,y of -1,-1 removes the item from the space grid */
+
+/* the current coordinates of the object in question are not relevant */
+
+void move_planet( /* pno, x, y, isold */ );
+void move_player( /* pno, x, y, isold */ );
+
+void move_torp( /* tno, x, y, isold */ );
+void move_missile( /* dno, x, y, isold */ );
+
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/imath.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,226 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+/* collection of integer math routines. [BDyess] */
+
+/*
+ * Based on (n+1)^2 = n^2 + 2n + 1 given that	1^2 = 1, then 2^2 = 1 + (2 +
+ * 1) = 1 + 3 = 4 3^2 = 4 + (4 + 1) = 4 + 5 = 1 + 3 + 5 = 9 4^2 = 9 + (6 + 1)
+ * = 9 + 7 = 1 + 3 + 5 + 7 = 16 ... In other words, a square number can be
+ * express as the sum of the series n^2 = 1 + 3 + ... + (2n-1)
+ * 
+ * Note that NO multiplication or floating point math is needed. [BDyess]
+ */
+int
+isqrt(n)
+  int n;
+{
+  int result = 0, sum = 1, prev = 1;
+
+  while (sum <= n)
+  {
+    prev += 2;
+    sum += prev;
+    ++result;
+  }
+  return result;
+}
+
+/*
+ * Calculates the distance directly using a lookup table. Very fast, esp.
+ * compared to hypot(), but less accurate.
+ * 
+ * Produces results exactly as (int) hypot( (double)x, (double)y) up to
+ * hypot(HYPOTMIN,HYPOTMIN), and then loses accuracy in the trailing digits.
+ * With HYPOTMIN = 1000, error is .01% at 200000,200000.
+ * 
+ * Bill Dyess
+ */
+
+#define HYPOTMIN 1000
+int hypotlookup[] = {
+  1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000,
+  1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000,
+  1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000,
+  1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000,
+  1000, 1000, 1000, 1000, 1000, 1001, 1001, 1001, 1001, 1001,
+  1001, 1001, 1001, 1001, 1001, 1001, 1001, 1001, 1001, 1001,
+  1001, 1001, 1001, 1001, 1002, 1002, 1002, 1002, 1002, 1002,
+  1002, 1002, 1002, 1002, 1002, 1002, 1002, 1002, 1003, 1003,
+  1003, 1003, 1003, 1003, 1003, 1003, 1003, 1003, 1003, 1003,
+  1004, 1004, 1004, 1004, 1004, 1004, 1004, 1004, 1004, 1004,
+  1004, 1005, 1005, 1005, 1005, 1005, 1005, 1005, 1005, 1005,
+  1006, 1006, 1006, 1006, 1006, 1006, 1006, 1006, 1006, 1007,
+  1007, 1007, 1007, 1007, 1007, 1007, 1007, 1008, 1008, 1008,
+  1008, 1008, 1008, 1008, 1008, 1009, 1009, 1009, 1009, 1009,
+  1009, 1009, 1010, 1010, 1010, 1010, 1010, 1010, 1010, 1011,
+  1011, 1011, 1011, 1011, 1011, 1011, 1012, 1012, 1012, 1012,
+  1012, 1012, 1013, 1013, 1013, 1013, 1013, 1013, 1014, 1014,
+  1014, 1014, 1014, 1014, 1015, 1015, 1015, 1015, 1015, 1015,
+  1016, 1016, 1016, 1016, 1016, 1016, 1017, 1017, 1017, 1017,
+  1017, 1018, 1018, 1018, 1018, 1018, 1019, 1019, 1019, 1019,
+  1019, 1020, 1020, 1020, 1020, 1020, 1020, 1021, 1021, 1021,
+  1021, 1022, 1022, 1022, 1022, 1022, 1023, 1023, 1023, 1023,
+  1023, 1024, 1024, 1024, 1024, 1025, 1025, 1025, 1025, 1025,
+  1026, 1026, 1026, 1026, 1027, 1027, 1027, 1027, 1027, 1028,
+  1028, 1028, 1028, 1029, 1029, 1029, 1029, 1030, 1030, 1030,
+  1030, 1031, 1031, 1031, 1031, 1032, 1032, 1032, 1032, 1032,
+  1033, 1033, 1033, 1034, 1034, 1034, 1034, 1035, 1035, 1035,
+  1035, 1036, 1036, 1036, 1036, 1037, 1037, 1037, 1037, 1038,
+  1038, 1038, 1039, 1039, 1039, 1039, 1040, 1040, 1040, 1040,
+  1041, 1041, 1041, 1042, 1042, 1042, 1042, 1043, 1043, 1043,
+  1044, 1044, 1044, 1044, 1045, 1045, 1045, 1046, 1046, 1046,
+  1046, 1047, 1047, 1047, 1048, 1048, 1048, 1049, 1049, 1049,
+  1049, 1050, 1050, 1050, 1051, 1051, 1051, 1052, 1052, 1052,
+  1053, 1053, 1053, 1053, 1054, 1054, 1054, 1055, 1055, 1055,
+  1056, 1056, 1056, 1057, 1057, 1057, 1058, 1058, 1058, 1059,
+  1059, 1059, 1060, 1060, 1060, 1061, 1061, 1061, 1062, 1062,
+  1062, 1063, 1063, 1063, 1064, 1064, 1064, 1065, 1065, 1065,
+  1066, 1066, 1066, 1067, 1067, 1068, 1068, 1068, 1069, 1069,
+  1069, 1070, 1070, 1070, 1071, 1071, 1071, 1072, 1072, 1072,
+  1073, 1073, 1074, 1074, 1074, 1075, 1075, 1075, 1076, 1076,
+  1077, 1077, 1077, 1078, 1078, 1078, 1079, 1079, 1080, 1080,
+  1080, 1081, 1081, 1081, 1082, 1082, 1083, 1083, 1083, 1084,
+  1084, 1085, 1085, 1085, 1086, 1086, 1086, 1087, 1087, 1088,
+  1088, 1088, 1089, 1089, 1090, 1090, 1090, 1091, 1091, 1092,
+  1092, 1092, 1093, 1093, 1094, 1094, 1094, 1095, 1095, 1096,
+  1096, 1096, 1097, 1097, 1098, 1098, 1099, 1099, 1099, 1100,
+  1100, 1101, 1101, 1101, 1102, 1102, 1103, 1103, 1104, 1104,
+  1104, 1105, 1105, 1106, 1106, 1107, 1107, 1107, 1108, 1108,
+  1109, 1109, 1110, 1110, 1110, 1111, 1111, 1112, 1112, 1113,
+  1113, 1114, 1114, 1114, 1115, 1115, 1116, 1116, 1117, 1117,
+  1118, 1118, 1118, 1119, 1119, 1120, 1120, 1121, 1121, 1122,
+  1122, 1122, 1123, 1123, 1124, 1124, 1125, 1125, 1126, 1126,
+  1127, 1127, 1128, 1128, 1128, 1129, 1129, 1130, 1130, 1131,
+  1131, 1132, 1132, 1133, 1133, 1134, 1134, 1135, 1135, 1136,
+  1136, 1136, 1137, 1137, 1138, 1138, 1139, 1139, 1140, 1140,
+  1141, 1141, 1142, 1142, 1143, 1143, 1144, 1144, 1145, 1145,
+  1146, 1146, 1147, 1147, 1148, 1148, 1149, 1149, 1150, 1150,
+  1151, 1151, 1152, 1152, 1153, 1153, 1154, 1154, 1155, 1155,
+  1156, 1156, 1157, 1157, 1158, 1158, 1159, 1159, 1160, 1160,
+  1161, 1161, 1162, 1162, 1163, 1163, 1164, 1164, 1165, 1165,
+  1166, 1166, 1167, 1167, 1168, 1168, 1169, 1169, 1170, 1170,
+  1171, 1171, 1172, 1172, 1173, 1173, 1174, 1175, 1175, 1176,
+  1176, 1177, 1177, 1178, 1178, 1179, 1179, 1180, 1180, 1181,
+  1181, 1182, 1182, 1183, 1184, 1184, 1185, 1185, 1186, 1186,
+  1187, 1187, 1188, 1188, 1189, 1189, 1190, 1191, 1191, 1192,
+  1192, 1193, 1193, 1194, 1194, 1195, 1195, 1196, 1197, 1197,
+  1198, 1198, 1199, 1199, 1200, 1200, 1201, 1202, 1202, 1203,
+  1203, 1204, 1204, 1205, 1205, 1206, 1207, 1207, 1208, 1208,
+  1209, 1209, 1210, 1210, 1211, 1212, 1212, 1213, 1213, 1214,
+  1214, 1215, 1216, 1216, 1217, 1217, 1218, 1218, 1219, 1220,
+  1220, 1221, 1221, 1222, 1222, 1223, 1224, 1224, 1225, 1225,
+  1226, 1226, 1227, 1228, 1228, 1229, 1229, 1230, 1231, 1231,
+  1232, 1232, 1233, 1233, 1234, 1235, 1235, 1236, 1236, 1237,
+  1238, 1238, 1239, 1239, 1240, 1241, 1241, 1242, 1242, 1243,
+  1244, 1244, 1245, 1245, 1246, 1247, 1247, 1248, 1248, 1249,
+  1250, 1250, 1251, 1251, 1252, 1253, 1253, 1254, 1254, 1255,
+  1256, 1256, 1257, 1257, 1258, 1259, 1259, 1260, 1260, 1261,
+  1262, 1262, 1263, 1263, 1264, 1265, 1265, 1266, 1266, 1267,
+  1268, 1268, 1269, 1270, 1270, 1271, 1271, 1272, 1273, 1273,
+  1274, 1275, 1275, 1276, 1276, 1277, 1278, 1278, 1279, 1280,
+  1280, 1281, 1281, 1282, 1283, 1283, 1284, 1285, 1285, 1286,
+  1286, 1287, 1288, 1288, 1289, 1290, 1290, 1291, 1291, 1292,
+  1293, 1293, 1294, 1295, 1295, 1296, 1297, 1297, 1298, 1298,
+  1299, 1300, 1300, 1301, 1302, 1302, 1303, 1304, 1304, 1305,
+  1305, 1306, 1307, 1307, 1308, 1309, 1309, 1310, 1311, 1311,
+  1312, 1313, 1313, 1314, 1315, 1315, 1316, 1316, 1317, 1318,
+  1318, 1319, 1320, 1320, 1321, 1322, 1322, 1323, 1324, 1324,
+  1325, 1326, 1326, 1327, 1328, 1328, 1329, 1330, 1330, 1331,
+  1332, 1332, 1333, 1334, 1334, 1335, 1336, 1336, 1337, 1338,
+  1338, 1339, 1340, 1340, 1341, 1342, 1342, 1343, 1344, 1344,
+  1345, 1346, 1346, 1347, 1348, 1348, 1349, 1350, 1350, 1351,
+  1352, 1352, 1353, 1354, 1354, 1355, 1356, 1356, 1357, 1358,
+  1358, 1359, 1360, 1360, 1361, 1362, 1362, 1363, 1364, 1364,
+  1365, 1366, 1366, 1367, 1368, 1369, 1369, 1370, 1371, 1371,
+  1372, 1373, 1373, 1374, 1375, 1375, 1376, 1377, 1377, 1378,
+  1379, 1380, 1380, 1381, 1382, 1382, 1383, 1384, 1384, 1385,
+  1386, 1386, 1387, 1388, 1388, 1389, 1390, 1391, 1391, 1392,
+  1393, 1393, 1394, 1395, 1395, 1396, 1397, 1398, 1398, 1399,
+  1400, 1400, 1401, 1402, 1402, 1403, 1404, 1405, 1405, 1406,
+  1407, 1407, 1408, 1409, 1409, 1410, 1411, 1412, 1412, 1413,
+  1414
+};
+
+int 
+ihypot(x, y)
+  int x, y;
+{
+  int max, min;
+
+  x = (x < 0) ? -x : x;
+  y = (y < 0) ? -y : y;
+  if (x > y)
+  {
+    max = x;
+    min = y;
+  }
+  else
+  {
+    max = y;
+    min = x;
+  }
+  if (max == 0)
+    return 0;
+  /* if(max < 32768) return isqrt(max*max+min*min); */
+  return hypotlookup[HYPOTMIN * min / max] * max / HYPOTMIN;
+}
+
+#if 0
+/* code to calculate the lookup table. */
+#include<math.h>
+
+int 
+main(int argc, char **argv)
+{
+  int i, j, max;
+
+  if (argc != 2)
+  {
+    printf("Usage: %s <size of ihypot() lookup table>\n", argv[0]);
+    return 0;
+  }
+  max = atoi(argv[1]);
+  printf("#define HYPOTMIN %d\nint hypotlookup[] = {\n  %d", max, max);
+  for (i = 1, j = 1; i <= max; i++, j++)
+  {
+    if (j % 10 == 0)
+      printf(",\n  ");
+    else
+      printf(", ");
+    printf("%d", (int) hypot((double) i, (double) max));
+  }
+  printf("\n};\n");
+  return 0;
+}
+#endif				/* 0 */
+
+#if 0
+/* code to test various routines */
+int 
+main(int argc, char **argv)
+{
+  if (argc != 3)
+  {
+    printf("Usage: %s <x> <y>\n", argv[0]);
+    return 0;
+  }
+  printf("hypot = %d\n", ihypot(atoi(argv[1]), atoi(argv[2])));
+  return 0;
+}
+#endif				/* 0 */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/input.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,163 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation for any NON-COMMERCIAL purpose (following the terms of
+the GNU General Public License (read the file 'COPYING')) and without
+fee is hereby granted, provided that this copyright notice appear in all
+copies.  No representations are made about the suitability of this
+software for any purpose.  This software is provided "as is" without
+express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+
+    Comprehensive credits are available in the file "CREDITS"
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <stdio.h>
+#include <string.h>
+#include <math.h>
+#include <sys/types.h>
+#include <sys/time.h>
+#include <signal.h>
+#include <errno.h>
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "shmem.h"
+
+static int sendflag = 0;
+
+extern int intrupt();
+extern void start_interruptor();
+extern int isClientDead();
+int reconnect();
+extern int readFromClient();
+extern void stop_interruptor();
+extern int exitGame();
+extern int closeUdpConn();
+extern unsigned int sleep();
+extern int connectToClient();
+extern void (*r_signal()) ();
+
+void
+setflag()
+{
+  sendflag = 1;
+}
+
+
+void
+input()
+{
+  fd_set readfds;
+
+  start_interruptor();
+
+  /* Idea:  read from client often, send to client not so often */
+  while (me->p_status == PALIVE ||
+	 me->p_status == PEXPLODE ||
+	 me->p_status == PDEAD ||
+	 me->p_status == POBSERVE)
+  {
+    if (isClientDead())
+    {
+      if (!reconnect())
+	exit(0);
+    }
+    FD_ZERO(&readfds);
+    FD_SET(sock, &readfds);
+    if (udpSock >= 0)
+      FD_SET(udpSock, &readfds);
+
+    /*
+     * blocks indefinitely unless client input or timerDelay SIGALRM causes
+     * EINTR
+     */
+
+    if (socketWait() > 0)
+    {
+      readFromClient();
+    }
+    if (sendflag)
+    {
+      intrupt();
+      sendflag = 0;
+    }
+
+    /* statements below are executed once per timerDelay */
+    if (me->p_updates > delay)
+    {
+      me->p_flags &= ~(PFWAR);
+    }
+    if (me->p_updates > rdelay)
+    {
+      me->p_flags &= ~(PFREFITTING);
+    }
+
+    /*
+     * this will ghostbust a player if no ping has been received in
+     * ping_ghostbust_interval seconds
+     */
+
+    if (configvals->ping_allow_ghostbust
+	&& (me->p_status == PALIVE || me->p_status == POBSERVE)
+	&& ping_ghostbust > configvals->ping_ghostbust_interval)
+      me->p_ghostbuster = 100000;	/* ghostbusted */
+    else
+      /* Keep self from being nuked... */
+      me->p_ghostbuster = 0;
+  }
+  stop_interruptor();
+}
+
+int
+reconnect()
+{
+  int i;
+
+  if (noressurect)
+    exitGame();
+  stop_interruptor();
+
+  r_signal(SIGIO, SIG_IGN);
+#ifdef DEBUG
+  printf("Ack!  The client went away!\n");
+  printf("I will attempt to resurrect him!\n");
+  fflush(stdout);
+#endif
+  commMode = COMM_TCP;
+  if (udpSock >= 0)
+    closeUdpConn();
+
+  /* For next two minutes, we try to restore connection */
+  shutdown(sock, 2);
+  sock = -1;
+  for (i = 0;; i++)
+  {
+    me->p_ghostbuster = 0;
+    sleep(5);
+    if (connectToClient(host, nextSocket))
+      break;
+    if (i == 23)
+    {
+#ifdef DEBUG
+      printf("Oh well, maybe I'm getting rusty!\n");
+      fflush(stdout);
+#endif
+      return 0;
+    }
+  }
+  start_interruptor();
+#ifdef DEBUG
+  printf("A miracle!  He's alive!\n");
+  fflush(stdout);
+#endif
+  return 1;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/interface.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,1236 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+
+#include "config.h"
+
+#include <stdio.h>
+#include <math.h>
+#include <signal.h>
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "shmem.h"
+
+void cloak_off();
+void cloak_on();
+int numShips();
+void sendwarn();
+extern int pmessage2();
+int allowed_ship();
+int numPlanets();
+void suspendPrep();
+void unsuspendPrep();
+
+/*-----------------------------VISIBLE FUNCTIONS---------------------------*/
+
+/*--------------------------------SETSPEED---------------------------------*/
+/*
+ * This function takes the desired speed from the client and translates it
+ * into impulse, afterburners, or warp.  99 = warp.  98 = afterburners. 97 =
+ * suspend warp prep, 96 = resume warp prep, 0-x = impulse.  This function is
+ * used for both requests to set a ships speed from the client and from the
+ * server.  The client requests are requests that are received directly from
+ * the client.  A speed of 99 is treated as a request for warp speed.  A
+ * request of 98 is a request for afterburners.
+ */
+
+
+
+void
+set_speed(speed, type)
+  int speed;			/* the desired speed */
+  int type;			/* type of request. 0=server 1=client */
+{
+  if (me->p_flags & PFDOCK)
+  {				/* if docked then */
+    if (players[me->p_docked].p_speed > 4)
+    {				/* you cannot get off >= 4 */
+      warning("You are going too fast to disengage safely");
+      return;
+    }
+    else
+      undock_player(me);
+  }
+  speed = (speed < 0) ? 0 : speed;	/* no negative speeds */
+  /* suspend warp  [BDyess] */
+  if (speed == 97 && configvals->warpprep_suspendable)
+  {				/* suspend warp */
+    suspendPrep();
+  }
+  else if (speed == 96 && configvals->warpprep_suspendable)
+  {
+    /* unsuspend warp [BDyess] */
+    unsuspendPrep();
+  }
+  else if ((speed == 98) && (configvals->afterburners))
+  {				/* afterburners */
+    /* turn off warp, warpprep, and warp suspended flags */
+    me->p_flags &= ~(PFWARP | PFWARPPREP | PFWPSUSPENDED);
+    me->p_warptime = 0;
+    me->p_flags |= PFAFTER;	/* turn on after burners */
+    me->p_desspeed = me->p_ship.s_after.maxspeed;	/* set speed of
+							 * afterburn */
+    /*
+     * go to warp if speed 99 is sent, OR if the speed sent is greater than
+     * max impulse and the VARIABLE_WARP sysdef is set, AND the ship can warp
+     * [BDyess]
+     */
+  }
+  else if ((speed == 99
+	    || (configvals->variable_warp
+		&& speed > me->p_ship.s_imp.maxspeed
+		&& me->p_flags & (PFWARPPREP | PFWARP | PFWPSUSPENDED))
+	    )
+	   && (me->p_ship.s_nflags & SFNCANWARP))
+  {
+    int maxspeed = me->p_ship.s_warp.maxspeed;
+    int preptime = myship->s_warpinittime;	/* timer before warping */
+
+    if (configvals->warpzone)
+    {
+      if (me->p_zone < 0)
+      {
+	warning("The enemy is neutralizing our warp field, sir!");
+	return;
+      }
+      else if (me->p_zone > 0)
+      {
+	maxspeed = maxspeed * 3 / 2;	/* can go faster with boost [BDyess] */
+	if (maxspeed > 99)
+	  maxspeed = 99;
+	preptime /= 2;		/* takes less time to build up when you have
+				 * help */
+	if (preptime == 0)
+	  preptime = 1;
+	warning("warp boost!");
+      }
+    }
+
+    speed = (speed > maxspeed)
+      ? maxspeed
+      : speed;			/* cap the request at max warp */
+
+    me->p_warpdesspeed = speed;	/* remember how fast to warp for later
+				 * [BDyess] */
+    if (me->p_warptime > 0)
+    {
+      warning("Hold your horses, Speed Racer.  These things take time.");
+      return;
+    }
+    if (me->p_flags & PFWARP)
+    {
+      me->p_desspeed = speed;
+      return;
+    }
+    if (me->p_damage > me->p_ship.s_maxdamage / 2)
+    {
+      warning("We are too damaged to risk warp speeds, Captain.");
+      return;
+    }
+    if (me->p_fuel < me->p_ship.s_warpinitcost)
+    {
+      warning("Not enough fuel for warp, Captain!");
+      return;
+    }
+    me->p_desspeed = myship->s_warpprepspeed;	/* slow ship down */
+    me->p_flags |= PFWARPPREP;	/* set warp prep flag [BDyess] */
+    if (!configvals->cloakduringwarpprep)
+      cloak_off();
+
+    if (me->p_speed < me->p_desspeed)
+      me->p_speed = me->p_desspeed;
+
+    /* turn off after burners and warpprep suspended flag [BDyess] */
+    me->p_flags &= ~(PFAFTER | PFWPSUSPENDED);
+    me->p_warptime = preptime;	/* timer before warping */
+    me->p_fuel -= me->p_ship.s_warpinitcost;	/* subtract off fuel */
+    warning("Warp drive engaged!");	/* let him know its working */
+  }
+  else if (type == 0)
+  {				/* else if server request */
+    me->p_desspeed = speed;	/* grant speed request */
+  }
+  else
+  {				/* else client request for impulse */
+
+    if (!configvals->warpdecel)
+      me->p_flags &= ~(PFWARP | PFAFTER);
+
+    me->p_warptime = 0;
+    me->p_flags &= ~(PFWARPPREP | PFWPSUSPENDED);	/* turn off warpprep
+							 * flag [BDyess] */
+    speed = (speed > me->p_ship.s_imp.maxspeed) ? me->p_ship.s_imp.maxspeed :
+      speed;
+    me->p_desspeed = speed;	/* set the speed */
+  }
+#if 0
+  if (me->p_flags & PFORBIT)	/* if we were orbiting */
+    planets[me->p_planet].pl_torbit &= ~me->p_team;
+#endif
+  me->p_flags &= ~(PFREPAIR | PFBOMB | PFORBIT | PFDOCK | PFBEAMUP | PFBEAMDOWN);
+  me->p_lastman = 0;
+}
+
+
+
+
+/*--------------------------------SET_COURSE-------------------------------*/
+/*
+ * Set the players course.  This function takes a direction and sets the
+ * players direction to it.  Certain flags are cleared.
+ */
+
+void
+set_course(dir)
+  unsigned char dir;
+{
+  me->p_desdir = dir;		/* set the desired direction */
+  if (me->p_flags & PFDOCK)
+  {				/* if player is docked then he cannot */
+    if (players[me->p_docked].p_speed > 4)
+    {				/* undock if dockee going fast */
+      warning("It's unsafe to disengage from bases while moving more that warp 4.");
+      return;
+    }
+    else
+      undock_player(me);
+
+  }
+#if 0
+  if (me->p_flags & PFORBIT)	/* if in orbit then take */
+    planets[me->p_planet].pl_torbit &= ~me->p_team;	/* team out of orbit */
+#endif
+  me->p_flags &= ~(PFBOMB | PFORBIT | PFDOCK | PFBEAMUP | PFBEAMDOWN);
+  me->p_lastman = 0;		/* not beamin all armies up */
+}
+
+
+
+
+/*----------------------------------SHIELD_UP-----------------------------*/
+/* This function raises the players shields.  */
+
+void
+shield_up()
+{
+  me->p_flags |= PFSHIELD;	/* some flags go up and some down */
+  me->p_flags &= ~(PFBOMB | PFREPAIR | PFBEAMUP | PFBEAMDOWN);
+  me->p_lastman = 0;		/* not beaming up all armies */
+}
+
+
+
+
+/*---------------------------------SHIELD_DOWN----------------------------*/
+/* This function lowers the players shields.  */
+
+void
+shield_down()
+{
+  me->p_flags &= ~PFSHIELD;	/* shield flag clear */
+}
+
+
+
+
+/*--------------------------------SHIELD_TOGGLE---------------------------*/
+/* This function toggles the players shields on and off.  */
+
+void
+shield_tog()
+{
+  me->p_flags ^= PFSHIELD;	/* toggle shield flag */
+  me->p_flags &= ~(PFBOMB | PFREPAIR | PFBEAMUP | PFBEAMDOWN);
+  me->p_lastman = 0;
+}
+
+
+
+
+/*--------------------------------BOMB_PLANET------------------------------*/
+/*
+ * This function sets the bomb flag if certain conditions are met, such as
+ * you are bombing enemy planets.
+ */
+
+void
+bomb_planet()
+{
+  static int bombsOutOfTmode = 0;	/* confirm bomb out of t-mode */
+  int owner;			/* owner of planet */
+
+  owner = planets[me->p_planet].pl_owner;	/* get planet's owner */
+  if (!(me->p_flags & PFORBIT))
+  {				/* can't bomb anything in */
+    warning("Must be orbiting to bomb");	/* open space, now can you? */
+    return;
+  }
+  if (me->p_team == planets[me->p_planet].pl_owner)
+  {
+    warning("Can't bomb your own armies.  Have you been reading Catch-22 again?");	/* can't bomb own armies */
+    return;
+  }
+  if (!((me->p_swar | me->p_hostile) & owner) && (owner != 0))
+  {
+    warning("Must declare war first (no Pearl Harbor syndrome allowed here).");
+    return;			/* can't bomb friendlies */
+  }
+  if ((!status->tourn) && (bombsOutOfTmode == 0))
+  {
+    warning("Bomb out of T-mode?  Please verify your order to bomb.");
+    bombsOutOfTmode++;		/* warning about bombing out of T */
+    return;
+  }
+  if ((!status->tourn) && (bombsOutOfTmode == 1))
+  {
+    warning("Hoser!");		/* if he really wants to bomb, then */
+    bombsOutOfTmode++;		/* let him bomb out of T.  Hoser */
+  }				/* warning */
+  if (!numShips(planets[me->p_planet].pl_owner)
+      && (planets[me->p_planet].pl_owner != NOBODY))
+  {
+    warning("You can't break the peace treaty, Captain");
+    return;
+  }
+  if (status->tourn)		/* reset the bombs out of t-mode */
+    bombsOutOfTmode = 0;	/* variable */
+  me->p_flags |= PFBOMB;	/* set the bomb flag */
+#if 1
+  planets[me->p_planet].pl_hostile |= me->p_team;
+#else
+  planets[me->p_planet].pl_torbit |= (me->p_team) << 4;
+#endif
+  me->p_flags &= ~(PFSHIELD | PFREPAIR | PFBEAMUP | PFBEAMDOWN);
+}
+
+
+
+
+/*---------------------------------BEAM_UP---------------------------------*/
+/* This function set the beam up flag if it is allowable to beam men up.  */
+
+void
+beam_up()
+{
+  if (!status2->starttourn && configvals->nopregamebeamup)
+  {
+    warning("You can't beam up armies before a game starts");
+    return;
+  }
+  if (!(me->p_flags & (PFORBIT | PFDOCK)))
+  {
+    warning("Must be orbiting or docked to beam up.");
+    return;			/* have to be orbiting or docked */
+  }
+  if (me->p_flags & PFORBIT)
+  {				/* if orbiting and not owner */
+    if (me->p_team != planets[me->p_planet].pl_owner)
+    {
+      warning("Those aren't our men.");
+      return;
+    }
+    if (me->p_lastman == 0)	/* if we have not signalled to pick */
+      me->p_lastman = 1;	/* up all armies */
+    else if (planets[me->p_planet].pl_armies <= 4)
+    {				/* else if less than 5 */
+      if (numPlanets((int) me->p_team && configvals->beamlastarmies) == 1)
+      {
+	warning("You can't take those armies, they're on your only planet!");
+	return;
+      }
+      me->p_lastman = 2;	/* set status to beaming up all */
+    }
+  }
+  else if (me->p_flags & PFDOCK)
+  {				/* if docked must be to own team ship */
+    if (me->p_team != players[me->p_docked].p_team)
+    {
+      warning("Comm Officer: Uhhh, they aren't our men!");
+      return;
+    }
+  }
+  me->p_flags |= PFBEAMUP;	/* set the beam up flag */
+  me->p_flags &= ~(PFSHIELD | PFREPAIR | PFBOMB | PFBEAMDOWN);
+}
+
+
+
+
+/*--------------------------------BEAM_DOWN--------------------------------*/
+/* This function sets the beam up flag if certain conditions are met */
+
+void
+beam_down()
+{
+  int i;			/* looping var */
+  struct planet *pl;		/* to access planets */
+  int total;			/* for total number of planets */
+  char buf[80];
+
+  if (!(me->p_flags & (PFORBIT | PFDOCK)))
+  {
+    warning("Must be orbiting or docked to beam down.");
+    return;			/* have to be docked or orbiting */
+  }
+  if ((!((me->p_swar | me->p_hostile) & planets[me->p_planet].pl_owner))
+      && (me->p_team != planets[me->p_planet].pl_owner)
+      && (planets[me->p_planet].pl_owner != NOBODY))
+  {
+    warning("You can't beam down, you have not declared war");
+    return;			/* no beaming down if not hostile */
+  }
+  if (!numShips(planets[me->p_planet].pl_owner)
+      && (planets[me->p_planet].pl_owner != NOBODY))
+  {
+    warning("You can't break the peace treaty, Captain");
+    return;
+  }
+  if (me->p_flags & PFDOCK)
+  {
+    if (me->p_team != players[me->p_docked].p_team)
+    {
+      warning("Comm Officer: Starbase refuses permission to beam our troops over.");	/* no beaming to foreign
+											 * bases */
+      return;
+    }
+  }
+
+  else if (configvals->planetlimittype)
+  {				/* new planet limit */
+    total = numPlanets(me->p_team);
+
+    if (total >= configvals->planetsinplay
+	&& planets[me->p_planet].pl_owner == NOBODY
+	&& planets[me->p_planet].pl_armies)
+    {
+      sprintf(buf, "You may only take enemy planets when your team has %d or more planets.",
+	      configvals->planetsinplay);
+      warning(buf);
+      me->p_flags &= ~(PFBOMB);
+      return;
+    }
+  }
+  else
+  {
+    int ns[MAXTEAM];
+    for (i = 0; i < NUMTEAM; i++)
+      ns[1 << i] = numShips(1 << i);
+
+    total = 0;
+    for (i = 0, pl = &planets[i]; i < NUMPLANETS; i++, pl++)
+    {
+      if (pl->pl_owner && ns[pl->pl_owner] > configvals->tournplayers - 1)
+	total++;
+    }
+    if ((total >= configvals->planetsinplay)
+	&& (!(me->p_flags & PFDOCK))
+	&& (planets[me->p_planet].pl_owner == NOBODY))
+    {
+      sprintf(buf, "Sorry, there are already %d planets in play.",
+	      configvals->planetsinplay);
+      warning(buf);
+      me->p_flags &= ~(PFBOMB);
+      return;
+    }
+  }
+  me->p_flags |= PFBEAMDOWN;	/* set beam down flag */
+  me->p_flags &= ~(PFSHIELD | PFREPAIR | PFBOMB | PFBEAMUP);
+  me->p_lastman = 0;
+}
+
+
+
+
+/*---------------------------------REPAIR---------------------------------*/
+/*
+ * This function sets the reapair flag and turns off other flags.  You cannot
+ * repair while in warp.
+ */
+
+void
+repair()
+{
+#if !defined(AEDILE) || !defined(REPAIR_IN_WARP)
+  if (me->p_flags & PFWARP)
+  {				/* no repairing in warp */
+    warning("You cannot repair while in warp!");
+    return;
+  }
+#endif
+  me->p_desspeed = 0;		/* speed goes to zero */
+  me->p_warptime = 0;		/* turn off warp prep */
+  me->p_flags |= PFREPAIR;	/* reair flag goes up */
+  me->p_flags &= ~(PFSHIELD | PFBOMB | PFBEAMUP | PFBEAMDOWN | PFPLOCK | PFPLLOCK | PFWARP | PFAFTER | PFWARPPREP | PFWPSUSPENDED);
+  me->p_lastman = 0;		/* not beaming all armies */
+}
+
+
+
+
+/*--------------------------------REPAIR_OFF------------------------------*/
+/* This function takes the repair flag off.  */
+
+void
+repair_off()
+{
+  me->p_flags &= ~PFREPAIR;
+}
+
+
+
+
+/*---------------------------------REPEAT---------------------------------*/
+/* This function repeats the last message.  */
+
+void
+repeat_message()
+{
+  if (++lastm == MAXMESSAGE)	/* go to next message */
+    lastm = 0;			/* account for rollover */
+}
+
+
+
+
+/*---------------------------------CLOAK----------------------------------*/
+/*
+ * This function toggles the cloak flag.  Tractors and pressor are turned
+ * off.
+ */
+
+void
+cloak()
+{
+  if (me->p_flags & PFCLOAK)
+    cloak_off();
+  else
+    cloak_on();
+}
+
+
+
+
+/*---------------------------------CLOAK_ON-------------------------------*/
+/* This function turns cloak on.  Tractors and pressors are turned off.  */
+
+void
+cloak_on()
+{
+  if (me->p_warptime && !configvals->cloakduringwarpprep)
+  {
+    warning("Can't cloak during warp preparation.");
+    return;
+  }
+  if (me->p_flags & PFWARP && !configvals->cloakwhilewarping)
+  {
+    warning("Can't cloak while warp engines are engaged.");
+    return;
+  }
+  me->p_flags |= PFCLOAK;
+  me->p_flags &= ~(PFTRACT | PFPRESS);
+}
+
+
+
+
+/*---------------------------------CLOAK_OFF------------------------------*/
+/* This function turns the cloak off.  */
+
+void
+cloak_off()
+{
+  me->p_flags &= ~PFCLOAK;
+}
+
+
+
+
+/*-------------------------------LOCK_PLANET-----------------------------*/
+/* This function locks the player onto a planet.  */
+
+void
+lock_planet(planet)
+  int planet;			/* planet locking onto */
+{
+  char buf[80];			/* to sprintf into */
+
+  if ((planet < 0) || (planet >= NUMPLANETS))	/* bad planet number? */
+    return;
+
+  if (me->p_status == POBSERVE)
+  {
+    if (planets[planet].pl_owner != me->p_team)
+    {
+      warning("Unable to observe enemy planets.");
+      me->p_flags &= ~(PFPLLOCK | PFPLOCK);
+      return;
+    }
+    else if (!(planets[planet].pl_owner & me->p_teamspy))
+    {
+      warning("That team has barred you from observing.");
+      me->p_flags &= ~(PFPLLOCK | PFPLOCK);
+      return;
+    }
+  }
+
+  me->p_flags |= PFPLLOCK;	/* turn on the lock */
+#if 0
+  if (me->p_flags & PFORBIT)	/* if orbiting then leave torbit var */
+    planets[me->p_planet].pl_torbit &= ~me->p_team;
+#endif
+  me->p_flags &= ~(PFPLOCK | PFORBIT | PFBEAMUP | PFBEAMDOWN | PFBOMB);
+  me->p_lastman = 0;
+  me->p_planet = planet;	/* set planet locked onto */
+  sprintf(buf, "Locking onto %s", planets[planet].pl_name);
+  warning(buf);			/* send message to player */
+}
+
+
+
+
+/*-------------------------------LOCK_PLAYER-------------------------------*/
+/* This function locks the player onto another player.  */
+
+void
+lock_player(player)
+  int player;			/* player locking onto */
+{
+  char buf[80];
+
+  if ((player < 0) || (player >= MAXPLAYER))	/* bad planet num? */
+    return;
+  if (players[player].p_status != PALIVE)	/* player not alive? */
+    return;
+  if (players[player].p_flags & PFCLOAK)	/* player is cloaked */
+    return;
+
+  if (me->p_status == POBSERVE && players[player].p_team != me->p_team)
+  {
+    sprintf(buf, "Unable to observe enemy players.");
+    warning(buf);
+    me->p_flags &= ~(PFPLLOCK | PFPLOCK);
+    return;
+  }
+
+  me->p_flags |= PFPLOCK;	/* set player lock flag */
+#if 0
+  if (me->p_flags & PFORBIT)	/* if in orbit take team out */
+    planets[me->p_planet].pl_torbit &= ~me->p_team;	/* of torbit */
+#endif
+  me->p_flags &= ~(PFPLLOCK | PFORBIT | PFBEAMUP | PFBEAMDOWN | PFBOMB);
+  me->p_lastman = 0;		/* turn off taking all armies */
+  me->p_playerl = player;	/* set player loked oto */
+  /* player locking onto own base */
+  if ((players[player].p_team == me->p_team) &&
+      allows_docking(players[player].p_ship))
+    sprintf(buf, "Locking onto %s (%s) (docking is %s)",
+	    players[player].p_name, twoletters(&players[player]),
+	    (players[player].p_flags & PFDOCKOK) ? "enabled" : "disabled");
+  else				/* message for locking onto player */
+    sprintf(buf, "Locking onto %s (%s)", players[player].p_name,
+	    twoletters(&players[player]));
+  warning(buf);
+}
+
+
+
+
+/*-------------------------------TRACTOR_PLAYER----------------------------*/
+/*
+ * This function does the tractoring for the player.  There are a few times
+ * when the tractor cannot be used.
+ */
+
+void
+tractor_player(player)
+  int player;
+{
+  struct player *victim;
+
+  if (weaponsallowed[WP_TRACTOR] == 0)	/* tractors allowed?  */
+    return;
+  if ((player < 0) || (player > MAXPLAYER))
+  {				/* out of bounds = cancel */
+    me->p_flags &= ~(PFTRACT | PFPRESS);
+    return;
+  }
+  if (me->p_flags & PFCLOAK)	/* no tractoring while */
+    return;			/* cloaked */
+  if (player == me->p_no)	/* can't tractor yourself */
+    return;
+  victim = &players[player];	/* get player number */
+  if (victim->p_flags & PFCLOAK)
+    return;
+  if (ihypot(me->p_x - victim->p_x, me->p_y - victim->p_y) <
+      (TRACTDIST) * me->p_ship.s_tractrng)
+  {
+    if (me->p_flags & PFDOCK && players[me->p_docked].p_speed > 4)
+    {
+      warning("It's unsafe to tractor while docked and base is moving more then warp 4.");
+      return;
+    }
+#if 0
+    undock_player(victim);	/* harmless if not docked, handles js */
+    undock_player(me);
+
+#if 0
+    if (me->p_flags & PFORBIT)	/* are we orbiting */
+      planets[me->p_planet].pl_torbit &= ~me->p_team;
+    if (victim->p_flags & PFORBIT)	/* is victim orbiting */
+      planets[victim->p_planet].pl_torbit &= ~victim->p_team;
+    victim->p_flags &= ~(PFORBIT | PFDOCK);	/* clear DOCK/ORBIT flags */
+#endif
+    me->p_flags &= ~(PFORBIT | PFDOCK);
+#endif
+    me->p_tractor = player;	/* set victim number */
+    me->p_flags |= PFTRACT;	/* set tractor flag */
+  }
+}
+
+
+
+
+/*---------------------------------PRESSOR--------------------------------*/
+/*
+ * This function activates the tractors for the player.  There are a number
+ * of conditions where tractors will not work.
+ */
+
+void
+pressor_player(player)
+  int player;			/* the player to presoor */
+{
+  int target;
+  struct player *victim;
+
+  if (weaponsallowed[WP_TRACTOR] == 0)
+  {				/* tractors allowed? */
+    warning("Pressor beams haven't been invented yet.");
+    return;
+  }
+  target = player;		/* save target */
+  if ((target < 0) || (target > MAXPLAYER))
+  {				/* out of bounds = cancel */
+    me->p_flags &= ~(PFTRACT | PFPRESS);
+    return;
+  }
+  if (me->p_flags & PFPRESS)	/* already pressoring? */
+    return;
+  if (me->p_flags & PFCLOAK)
+  {				/* cloaked? */
+    warning("Weapons's Officer:  Cannot pressor while cloaked, sir!");
+    return;
+  }
+  victim = &players[target];	/* get victim's struct */
+  if (victim->p_flags & PFCLOAK)/* victim cloaked */
+    return;
+  if (ihypot(me->p_x - victim->p_x, me->p_y - victim->p_y) <
+      (TRACTDIST) * me->p_ship.s_tractrng)
+  {
+    if (me->p_flags & PFDOCK && players[me->p_docked].p_speed > 4)
+    {
+      warning("It's unsafe to pressor while docked and base is moving more then warp 4.");
+      return;
+    }
+#if 0
+    undock_player(victim);
+    undock_player(me);
+
+#if 0
+    if (me->p_flags & PFORBIT)	/* are we orbiting */
+      planets[me->p_planet].pl_torbit &= ~me->p_team;
+    if (victim->p_flags & PFORBIT)	/* is victim orbiting */
+      planets[victim->p_planet].pl_torbit &= ~victim->p_team;
+#endif
+    victim->p_flags &= ~(PFORBIT | PFDOCK);	/* clear orbit and dock flags */
+    me->p_flags &= ~(PFORBIT | PFDOCK);
+#endif
+    me->p_tractor = target;	/* set the target */
+    me->p_flags |= (PFTRACT | PFPRESS);	/* set the tract and press */
+  }
+  /* flags */
+  else
+    warning("Weapon's Officer:  Vessel is out of range of our pressor beam.");
+}
+
+
+
+
+/*-------------------------------DECLARE_WAR-----------------------------*/
+/* This function changes the war mask of a player.  */
+
+void
+declare_war(mask)
+  int mask;			/* who are we declaring war against */
+{
+  int changes;			/* to hold changes in war mask */
+  int i;			/* looping var */
+
+  mask &= ALLTEAM;		/* mask off extraneous bits */
+  mask &= ~me->p_team;		/* mask out our team bit */
+  mask |= me->p_swar;
+  changes = mask ^ me->p_hostile;	/* determine changes */
+  if (changes == 0)		/* if no changes then we are done */
+    return;
+  if (changes & FED)		/* if Fed status changed then warn them */
+    sendwarn("Federation", mask & FED, FED);
+  if (changes & ROM)		/* if Rom status changed then warn them */
+    sendwarn("Romulans", mask & ROM, ROM);
+  if (changes & KLI)		/* if Kli status changed then warn them */
+    sendwarn("Klingons", mask & KLI, KLI);
+  if (changes & ORI)		/* if Orion status changed then warn them */
+    sendwarn("Orions", mask & ORI, ORI);
+  if (me->p_flags & PFDOCK)
+  {				/* if docked and now at war with */
+    if (players[me->p_docked].p_team & mask)
+    {				/* dockee then undock */
+      undock_player(me);
+    }
+  }
+  else if (allows_docking(me->p_ship))
+  {				/* if ship is dockable then */
+    if (me->p_docked > 0)
+    {				/* go through ports and */
+      for (i = 0; i < me->p_ship.s_numports; i++)
+      {				/* kick off warring ships */
+	if (me->p_port[i] >= 0 &&
+	    (mask & players[me->p_port[i]].p_team))
+	{
+	  base_undock(me, i);
+	}
+      }
+    }
+  }
+  if (mask & ~me->p_hostile)
+  {				/* no sudden changing of */
+    me->p_flags |= PFWAR;	/* war status */
+    delay = me->p_updates + 100;
+    warning("Pausing ten seconds to re-program battle computers.");
+  }
+  me->p_hostile = mask;		/* new hostile mask */
+}
+
+
+
+
+/*-----------------------------------SENDWARN------------------------------*/
+/*
+ * This function sends the warning message to the other players when a player
+ * switches his war status with that team.
+ */
+
+void
+sendwarn(string, atwar, team)
+  char *string;			/* the name of the team changing status
+				 * towards */
+  int atwar;			/* 1 if declarig war 0 if decalring peace */
+  int team;			/* name of team we are switching in regards
+				 * to */
+{
+  char buf[BUFSIZ];		/* to hold sprintfed message */
+  char addrbuf[10];		/* to hold ship number and team letter */
+
+  if (atwar)
+  {				/* if decalring war */
+    (void) sprintf(buf, "%s (%s) declaring war on the %s",
+		   me->p_name, twoletters(me), string);
+  }
+  else
+  {				/* else decalring peace */
+    (void) sprintf(buf, "%s (%s) declaring peace with the %s",
+		   me->p_name, twoletters(me), string);
+  }
+  (void) sprintf(addrbuf, " %s->%-3s", twoletters(me), teams[team].shortname);
+  pmessage2(buf, team, MTEAM, addrbuf, me->p_no);
+}
+
+
+/* switches the special weapon of the player */
+
+void
+switch_special_weapon()
+{
+  int mask = 0;
+  int currflag;
+  int i, j;
+
+  mask = me->p_ship.s_nflags & (SFNHASMISSILE | SFNPLASMAARMED);
+
+  if (!mask)
+  {
+    warning("This ship is not armed with any special weapons");
+    me->p_specweap = 0;
+    return;
+  }
+  for (i = 0; i < sizeof(int) * 8; i++)
+  {
+    currflag = 1 << i;
+    if (!mask & currflag)
+      continue;
+    if (me->p_specweap & currflag)
+    {
+      i++;
+      break;
+    }
+  }
+
+  for (j = 0; j < sizeof(int) * 8; i++, j++)
+  {
+    if (i > sizeof(int) * 8)
+      i = 0;
+    currflag = 1 << i;
+    if (mask & currflag)
+    {
+      me->p_specweap = currflag;
+      break;
+    }
+  }
+
+  if (me->p_specweap & SFNHASMISSILE)
+  {
+    if (me->p_ship.s_nflags & SFNHASFIGHTERS)
+      warning("Fighters ready");
+    else
+      warning("Missiles armed");
+  }
+  else if (me->p_specweap & SFNPLASMAARMED)
+  {
+    warning("Plasmas armed");
+  }
+}
+
+/*-------------------------------DO_REFIT---------------------------------*/
+/*
+ * This function will attempt to refit a player to another ship.  There are
+ * all sorts of nasty reasons why it might fail.
+ */
+
+void
+do_refit(type)
+  int type;			/* type of ship to refit to */
+{
+  int i;			/* looping var */
+
+  if (type < 0 || type >= NUM_TYPES)	/* ship type select out of range */
+    return;
+  if (me->p_flags & PFORBIT)
+  {				/* if orbiting must be a shipyard */
+    if (!(planets[me->p_planet].pl_flags & PLSHIPYARD))
+    {
+      warning("You can change ships at a planet with a shipyard on it.");
+      return;
+    }
+#if 0
+    if (planets[me->p_planet].pl_owner != me->p_team)
+    {
+      warning("You can only refit on planets that your team owns.");
+      return;
+    }
+#endif
+  }
+  else if (me->p_flags & PFDOCK)
+  {				/* if docked then */
+    if (allows_docking(shipvals[type]))
+    {				/* cannot refit to a starbase */
+      warning("Can only refit to a base in a shipyard");
+      return;
+    }
+    if (players[me->p_docked].p_team != me->p_team)
+    {
+      warning("You must dock with YOUR base to apply for command reassignment!");	/* no refitting on enemy
+											 * SB's */
+      return;
+    }
+    if (!(players[me->p_docked].p_ship.s_nflags & SFNCANREFIT))
+    {
+      warning("This base can not provide you with a new ship.");
+      return;
+    }
+  }
+  else
+  {				/* planet does not have shipyard */
+    warning("Must orbit a shipyard or dock with your base to apply for command reassignment!");
+    return;
+  }
+  if ((me->p_damage > ((float) me->p_ship.s_maxdamage) * .75) ||
+      (me->p_shield < ((float) me->p_ship.s_maxshield) * .75) ||
+      (me->p_fuel < ((float) me->p_ship.s_maxfuel) * .75))
+  {
+    warning("Central Command refuses to accept a ship in this condition!");
+    return;			/* ship must be in good condition */
+  }
+  if ((me->p_armies > 0))
+  {				/* no refitting with armies on board */
+    warning("You must beam your armies down before moving to your new ship");
+    return;
+  }
+
+  {
+    int oldt, oldd, oldp, allowed;
+    oldt = me->p_ship.s_type;
+    oldd = me->p_ship.s_numdefn;
+    oldp = me->p_ship.s_numplan;
+    me->p_ship.s_type = -1;
+    me->p_ship.s_numdefn = 0;
+    me->p_ship.s_numplan = 0;
+    allowed = allowed_ship(me->p_team, me->p_stats.st_rank,
+			   me->p_stats.st_royal, type);
+    me->p_ship.s_type = oldt;
+    me->p_ship.s_numdefn = oldd;
+    me->p_ship.s_numplan = oldp;
+    if (!allowed)
+      return;
+  }
+
+  if (me->p_ship.s_type == STARBASE ||
+      me->p_ship.s_type == WARBASE ||
+      me->p_ship.s_type == CARRIER)
+    me->p_kills = 0;		/* reset kills to zero if coming */
+  /* from a base */
+
+  if (allows_docking(me->p_ship))
+  {				/* if ship dockable then */
+    for (i = 0; i < me->p_ship.s_numports; i++)
+    {				/* dump all ship out of dock */
+      base_undock(me, i);
+    }
+  }
+  get_ship_for_player(me, type);/* place ship in player strct */
+
+  switch_special_weapon();
+
+  if (me->p_flags & PFORBIT)
+  {
+    char buf[120];
+    me->p_lastrefit = me->p_planet;
+    sprintf(buf, "%s is now your home port", planets[me->p_lastrefit].pl_name);
+    warning(buf);
+  }
+
+  me->p_flags &= ~(PFREFIT);
+  me->p_flags |= PFREFITTING;
+  rdelay = me->p_updates + 50;
+  warning("You are being transported to your new vessel .... ");
+}
+
+
+/*
+ * Is a person with this rank on this team able to requisition a ship of this
+ * type?
+ */
+int
+allowed_ship(team, rank, royal, type)
+  int team;
+  int rank;
+  int royal;
+  int type;
+{
+  struct ship *tempship;
+  int shipcount;
+  int used_teamstr;		/* people required to support flying bases */
+  int used_planstr;		/* planets required to support flying bases */
+  int i, maxrank = 0;
+
+  if (!shipsallowed[type])
+  {
+    warning("That ship hasn't been designed yet.");
+    return 0;
+  }
+  tempship = &shipvals[type];
+
+#ifdef LEAGUE_SUPPORT
+  if (status2->league == 1)
+    return 1;
+
+  if (!status2->league)
+  {
+#endif
+    if (configvals->baserankstyle)
+    {
+      for (i = 0; i < MAXPLAYER; i++)
+	if ((players[i].p_status == PALIVE ||
+	     players[i].p_status == POUTFIT)
+	    && players[i].p_team == team)
+	  maxrank = MAX(rank, players[i].p_stats.st_rank);
+    }
+    else
+      maxrank = NUMRANKS;
+
+    if ((rank < MIN(tempship->s_rank, maxrank) && !(royal > GODLIKE)) &&
+	!(shipvals[tmpPick].s_nflags & SFNMASSPRODUCED))
+    {
+      char buf[100];
+      sprintf(buf, "Sorry %s, a rank of %s is required to command that ship.",
+	      ranks[rank].name, ranks[tempship->s_rank].name);
+      warning(buf);
+      return 0;
+    }
+#ifdef LEAGUE_SUPPORT
+  }
+#endif
+  used_teamstr = used_planstr = 0;
+  shipcount = 0;		/* How many ships of that type */
+  /* are on our team already? */
+  for (i = 0; i < MAXPLAYER; i++)
+  {
+    if (players[i].p_status == PALIVE && players[i].p_team == team)
+    {
+      if (players[i].p_ship.s_type == type)
+	shipcount++;
+      used_teamstr += players[i].p_ship.s_numdefn;
+      used_planstr += players[i].p_ship.s_numplan;
+    }
+  }
+  /*
+   * if you are in a base, and near the limits, you will sometimes have to
+   * switch into a normal ship before changing into another base
+   */
+
+  if ((shipcount >= tempship->s_maxnum) && !(shipvals[tmpPick].s_nflags & SFNMASSPRODUCED))
+  {
+    char buf[100];
+    sprintf(buf, "We already have %d %ss.", shipcount, tempship->s_name);
+    warning(buf);
+    return 0;
+  }
+  if ((teams[team].s_turns[type] >
+       (tempship->s_maxnum - shipcount - 1) * tempship->s_timer) &&
+      !(shipvals[tmpPick].s_nflags & SFNMASSPRODUCED))
+  {
+    char buf[100];
+    int turnsleft = teams[team].s_turns[type] -
+    (tempship->s_maxnum - shipcount - 1) * tempship->s_timer;
+    if (teams[team].s_turns[type] > 1)
+    {
+      sprintf(buf, "We are still building that ship (%d minutes until finished.)", turnsleft);
+      warning(buf);
+    }
+    else
+      warning("We are still building that ship (1 minute until finished.)");
+    return 0;
+  }
+  if ((tempship->s_numdefn &&
+       numShips(team) < tempship->s_numdefn + used_teamstr) &&
+      !(shipvals[tmpPick].s_nflags & SFNMASSPRODUCED))
+  {
+    char buf[100];
+    sprintf(buf, "Your team is not strong enough to defend a %s (%d+%d players needed).",
+	    tempship->s_name, tempship->s_numdefn, used_teamstr);
+    warning(buf);
+    return 0;
+  }
+  if ((tempship->s_numplan &&
+       numPlanets(team) < tempship->s_numplan + used_planstr) &&
+      !(shipvals[tmpPick].s_nflags & SFNMASSPRODUCED))
+  {
+    char buf[100];
+    sprintf(buf, "Your team's struggling economy cannot support a %s (%d+%d planets needed).", tempship->s_name, tempship->s_numplan, used_planstr);
+    warning(buf);
+    return 0;			/* have enough planets? */
+  }
+  return 1;
+}
+
+
+/*----------------------------------NUMSHIPS------------------------------*/
+/* This function counts the number of players on a team.  */
+
+int
+numShips(owner)
+  int owner;			/* the team to count for */
+{
+  int i;			/* looping var */
+  int num;			/* to hold player count */
+  struct player *p;		/* to point to players */
+
+  num = 0;			/* zero the count */
+  for (i = 0, p = players; i < MAXPLAYER; i++, p++)
+  {				/* go throgh players */
+    if (p->p_status != PFREE && p->p_team == owner)	/* if alive then */
+      num++;			/* inc player count */
+  }
+  return (num);			/* return number of ships */
+}
+
+
+
+
+/*------------------------------NUMPLANETS--------------------------------*/
+/* This function counts the number of planets a team has.  */
+
+int
+numPlanets(owner)
+  int owner;			/* the team to check for */
+{
+  int i;			/* looping var */
+  int num;			/* to hold count */
+  struct planet *p;		/* to point to a planet */
+
+  num = 0;			/* no planets found yet */
+  for (i = 0, p = planets; i < MAXPLANETS; i++, p++)
+  {
+    if (p->pl_owner == owner)	/* if planet owned by team then inc */
+      num++;
+  }
+  return (num);			/* return number of planets */
+}
+
+/*-------------------------------------------------------------------------*/
+
+/*
+ * Suspends warp prep [BDyess]
+ */
+void
+suspendPrep()
+{
+  /* check to see if the server allows warp prep to be suspended [BDyess] */
+  if (!configvals->warpprep_suspendable)
+    return;
+  if (me->p_flags & PFWARPPREP)
+  {
+    me->p_flags |= PFWPSUSPENDED;	/* turn on suspended flag */
+    warning("Warp prep suspended.");
+  }
+  else
+  {
+    warning("Not in warp prep!");
+  }
+}
+
+/*
+ * Unsuspends warp prep [BDyess]
+ */
+void
+unsuspendPrep()
+{
+  /* check to see if the server allows warp prep to be suspended [BDyess] */
+  if (!configvals->warpprep_suspendable)
+    return;
+  if (me->p_flags & PFWARPPREP)
+  {
+    me->p_flags &= ~PFWPSUSPENDED;	/* turn off suspended flag */
+    warning("Warp prep continued.");
+  }
+  else
+  {
+    warning("Not in warp prep!");
+  }
+}
+
+/*-------END OF LINE----*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/listen.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,634 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+/*------------------------------------------------------------------------
+Startup program for netrek.  Listens for connections, then forks off
+servers.  Based on code written by Brett McCoy, but heavily modified.
+
+Note that descriptor 2 is duped to descriptor 1, so that stdout and
+stderr go to the same file.
+--------------------------------------------------------------------------*/
+
+#ifndef apollo
+#include <unistd.h>
+#endif
+#include <sys/types.h>
+#include <sys/time.h>
+#include <sys/socket.h>
+#include <sys/ioctl.h>
+#include <sys/stat.h>
+#include <sys/wait.h>
+#ifdef SVR4
+#include <sys/termios.h>
+#endif				/* SVR4 */
+#include <netdb.h>
+#include <errno.h>
+#include <time.h>
+#include <netinet/in.h>
+#include <fcntl.h>
+#include <varargs.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <signal.h>
+#include <string.h>
+#ifdef SYSV
+#include <fcntl.h>
+#endif
+#ifdef hpux
+#include <sys/ptyio.h>
+#endif
+#include "defs.h"
+#include "path.h"
+#include "data.h"
+
+/* #define DEF_PORT	2592	/* port to listen on */
+/*
+ * #define NTSERV		"bin/ntserv"
+ * 
+ * #define METASERVER	"metaserver.ecst.csuchico.edu"
+ */
+/*
+ * Since the metaserver address is a nameserver alias, we can't just compare
+ * the peer hostname with the setting of METASERVER.  The peer hostname
+ * returned by gethostbyaddr() will be the machine's real name, which will be
+ * different (and could change).
+ * 
+ * So, we'll do a gethostbyname() on METASERVER to get an IP address, then we
+ * can compare that with the connecting peer.
+ * 
+ * It'd be overkill to get the metaserver's IP address with every connection;
+ * and it may be inadequate to get it just once at startup, since listen can
+ * exist for long periods of time, and the metaserver's IP address could be
+ * changed in that time.
+ * 
+ * So, we'll grab the IP address at least every META_UPDATE_TIME seconds
+ */
+#define META_UPDATE_TIME (5*60*60)	/* five hours */
+
+void printlistenUsage();
+void set_env();
+int listenSock;
+short port = PORT;
+char *program;
+int meta_addr;
+
+char *dateTime();
+void detach();
+void getConnections();
+void getListenSock();
+void multClose();
+void reaper();
+void terminate();
+
+/*
+ * Error reporting functions ripped from my library.
+ */
+void syserr();
+void warnerr();
+void fatlerr();
+void err();
+char *lasterr();
+void print_pid();
+
+struct hostent *gethostbyaddr();
+char *inet_ntoa();
+int metaserverflag;
+char *leagueflag = 0;
+char *observerflag = 0;
+
+#define NEA	10
+char *extraargs[NEA];
+int extraargc = 0;
+
+char *ntserv_binary = NTSERV;
+
+/*
+ * the System-V signal() function provides the older, unreliable signal
+ * semantics.  So, this is an implementation of signal using sigaction.
+ */
+
+void (*
+      r_signal(sig, func)) ()
+  int sig;
+  void (*func) ();
+{
+  struct sigaction act, oact;
+
+  act.sa_handler = func;
+
+  sigemptyset(&act.sa_mask);
+  act.sa_flags = 0;
+#ifdef SA_RESTART
+  act.sa_flags |= SA_RESTART;
+#endif
+
+  if (sigaction(sig, &act, &oact) < 0)
+    return (SIG_ERR);
+
+  return (oact.sa_handler);
+}
+
+
+int
+main(argc, argv)
+  int argc;
+  char *argv[];
+{
+  int i, key;
+  int nogo = 0;
+  struct hostent *he;
+  struct timeval tv;
+  time_t stamp, now;
+
+  metaserverflag = 0;
+#ifndef apollo
+  nice(-20);
+  nice(-20);
+  nice(-20);
+#endif
+
+  for (i = 1; i < argc; i++)
+  {
+    if (*argv[i] == '-')
+    {
+      switch (argv[i][1])
+      {
+       case 'p':
+	port = atoi(argv[i + 1]);
+	break;
+       case 'k':
+	if (++i < argc && sscanf(argv[i], "%d", &key) > 0 && key > 0)
+	  set_env("TREKSHMKEY", argv[i]);
+	else
+	  nogo++;
+	break;
+       case 'b':
+	ntserv_binary = argv[++i];
+	break;
+       case 'h':
+       case 'u':		/* for old times sake, the others don't do
+				 * this, but this one does so it doesn't pass
+				 * to ntserv */
+       case '-':		/* same with this */
+	nogo++;
+	break;
+       default:
+	/* all unknown arguments are passed through to ntserv. */
+	extraargs[extraargc++] = argv[i];
+      }
+    }
+    /* else just ignore non flags */
+  }
+
+  if ((program = strrchr(argv[0], '/')))
+    ++program;
+  else
+    program = argv[0];		/* let err functions know our name */
+
+  if (nogo)
+    printlistenUsage(program);
+
+  detach();			/* detach from terminal, close files, etc. */
+  print_pid();			/* log the new PID */
+  getListenSock();
+  r_signal(SIGCHLD, reaper);
+  r_signal(SIGTERM, terminate);
+  r_signal(SIGHUP, SIG_IGN);
+
+  meta_addr = 0;
+  stamp = 0;
+  tv.tv_sec = 1;
+  tv.tv_usec = 0;
+
+  while (1)
+  {
+    now = time(0);
+    if (now - stamp > META_UPDATE_TIME)
+    {
+      he = gethostbyname(METASERVER);
+      if (he)
+	meta_addr = *((int *) he->h_addr);
+      stamp = now;
+    }
+    getConnections();
+    select(0, 0, 0, 0, &tv);	/* wait one sec between each connection */
+  }
+}
+
+/***********************************************************************
+ * Detach process in various ways.
+ */
+
+void
+detach()
+{
+  int fd, rc, mode;
+  char *fname;
+
+  mode = S_IRUSR | S_IRGRP | S_IROTH | S_IWUSR;
+  /* fname = build_path("startup.log"); */
+  fname = build_path("logs/startup.log");
+  if ((fd = open(fname, O_WRONLY | O_CREAT | O_APPEND, mode)) == -1)
+    syserr(1, "detach", "couldn't open log file. [%s]", dateTime());
+  dup2(fd, 2);
+  dup2(fd, 1);
+  multClose(1, 2, -1);		/* close all other file descriptors */
+  warnerr(0, "started at %s on port %d.", dateTime(), port);
+
+  /* fork once to escape the shells job control */
+  if ((rc = fork()) > 0)
+    exit(0);
+  else if (rc < 0)
+    syserr(1, "detach", "couldn't fork. [%s]", dateTime());
+
+  /* now detach from the controlling terminal */
+
+#ifdef _SEQUENT_
+  if ((fd = open("/dev/tty", O_RDWR | O_NOCTTY, 0)) >= 0)
+  {
+    (void) close(fd);
+  }
+#else
+  if ((fd = open("/dev/tty", O_RDWR, 0)) == -1)
+  {
+    warnerr("detach", "couldn't open tty, assuming still okay. [%s]",
+	    dateTime());
+    return;
+  }
+#if defined(SYSV) && defined(TIOCTTY)
+  {
+    int zero = 0;
+    ioctl(fd, TIOCTTY, &zero);
+  }
+#else
+  ioctl(fd, TIOCNOTTY, 0);
+#endif
+  close(fd);
+#endif				/* _SEQUENT_ */
+
+  setsid();			/* make us a new process group/session */
+}
+
+/***********************************************************************
+ */
+
+void
+getListenSock()
+{
+  struct sockaddr_in addr;
+
+  if ((listenSock = socket(AF_INET, SOCK_STREAM, 0)) < 0)
+    syserr(1, "getListenSock", "can't create listen socket. [%s]",
+	   dateTime());
+
+  addr.sin_family = AF_INET;
+  addr.sin_addr.s_addr = INADDR_ANY;
+  addr.sin_port = htons(port);
+  {
+    /* added this so we could handle nuking listen. KAO 3/26/93 */
+    int foo = 1;
+    if (setsockopt(listenSock, SOL_SOCKET, SO_REUSEADDR, (char *) &foo,
+		   sizeof(foo)) == -1)
+    {
+      fprintf(stderr, "Error setting socket options in listen.\n");
+      exit(1);
+    }
+  }
+  if (bind(listenSock, (struct sockaddr *) & addr, sizeof(addr)) < 0)
+    syserr(1, "getListenSock", "can't bind listen socket. [%s]",
+	   dateTime());
+
+  if (listen(listenSock, 5) != 0)
+    syserr(1, "getListenSock", "can't listen to socket. [%s]", dateTime());
+}
+
+/***********************************************************************
+ */
+
+void
+getConnections()
+{
+  int len, sock, pid, pa;
+  struct sockaddr_in addr;
+  struct hostent *he;
+  char host[100];
+
+  len = sizeof(addr);
+  while ((sock = accept(listenSock, (struct sockaddr *) & addr, &len)) < 0)
+  {
+    /* if we got interrupted by a dying child, just try again */
+    if (errno == EINTR)
+      continue;
+    else
+      syserr(1, "getConnections",
+	     "accept() on listen socket failed. [%s]", dateTime());
+  }
+
+  /* get the host name */
+  he = gethostbyaddr((char *) &addr.sin_addr.s_addr,
+		     sizeof(addr.sin_addr.s_addr), AF_INET);
+  if (he != 0)
+  {
+    strcpy(host, he->h_name);
+    pa = *((int *) he->h_addr);
+  }
+  else
+  {
+    strcpy(host, inet_ntoa(addr.sin_addr));
+    pa = (int) addr.sin_addr.s_addr;
+  }
+
+  if (pa == meta_addr)
+    metaserverflag = 1;
+  /* else */
+  warnerr(0, "connect: %-33s[%s][%d]", host, dateTime(), metaserverflag);
+
+  /* fork off a server */
+  if ((pid = fork()) == 0)
+  {
+    char *newargv[10];
+    int newargc = 0;
+    int i;
+    char *binname;
+    binname = build_path(ntserv_binary);
+    dup2(sock, 0);
+    multClose(0, 1, 2, -1);	/* close everything else */
+
+    newargv[newargc++] = "ntserv";
+    if (metaserverflag == 1)
+      newargv[newargc++] = "-M";
+    for (i = 0; i < extraargc; i++)
+      newargv[newargc++] = extraargs[i];
+    newargv[newargc++] = host;
+    newargv[newargc] = 0;
+
+    execv(binname, newargv);
+
+    syserr(1, "getConnections",
+	   "couldn't execv %s as the server. [%s]",
+	   binname, dateTime());
+    exit(1);
+  }
+  else if (pid < 0)
+    syserr(1, "getConnections", "can't fork. [%s]", dateTime());
+
+  close(sock);
+  metaserverflag = 0;
+}
+
+/***********************************************************************
+ * Adds "var=value" to environment list
+ */
+
+void
+set_env(var, value)
+  char *var, *value;
+{
+  char *buf;
+#if defined(SYSV) || defined(sparc)
+  buf = malloc(strlen(var) + strlen(value) + 2);
+  if (!buf)
+    syserr(1, "set_env", "malloc() failed. [%s]", dateTime());
+
+  strcpy(buf, var);
+  strcat(buf, "=");
+  strcat(buf, value);
+
+  putenv(buf);
+#else
+  /* don't need to malloc space, setenv() does it for us */
+  setenv(var, value, 1);
+#endif
+}
+
+
+/***********************************************************************
+ * Returns a string containing the date and time.  String area is static
+ * and reused.
+ */
+
+char *
+dateTime()
+{
+  time_t t;
+  char *s;
+
+  time(&t);
+  s = ctime(&t);
+  s[24] = '\0';			/* wipe-out the newline */
+  return s;
+}
+
+/***********************************************************************
+ * Handler for SIGTERM.  Closes and shutdowns everything.
+ */
+
+void
+terminate()
+{
+  int s;
+
+  fatlerr(1, "terminate", "killed. [%s]", dateTime());
+#ifdef SYSV
+  s = sysconf(_SC_OPEN_MAX);
+  if (s < 0)			/* value for OPEN_MAX is indeterminate, */
+    s = 32;			/* so make a guess */
+#else
+  s = getdtablesize();
+#endif
+  /* shutdown and close everything */
+  for (; s >= 0; s--)
+  {
+    shutdown(s, 2);
+    close(s);
+  }
+}
+
+/***********************************************************************
+ * Waits on zombie children.
+ */
+
+void
+reaper()
+{
+#ifndef SVR4
+  while (wait3(0, WNOHANG, 0) > 0);
+#else
+  while (waitpid(0, 0, WNOHANG) > 0);
+#endif				/* SVR4 */
+}
+
+/***********************************************************************
+ * Close all file descriptors except the ones specified in the argument list.
+ * The list of file descriptors is terminated with -1 as the last arg.
+ */
+
+void
+multClose(va_alist) va_dcl
+{
+  va_list args;
+  int fds[100], nfds, fd, ts, i, j;
+
+  /* get all descriptors to be saved into the array fds */
+  va_start(args);
+  for (nfds = 0; nfds < 99; nfds++)
+  {
+    if ((fd = va_arg(args, int)) == -1)
+      break;
+    else
+      fds[nfds] = fd;
+  }
+
+#ifdef SYSV
+  ts = sysconf(_SC_OPEN_MAX);
+  if (ts < 0)			/* value for OPEN_MAX is indeterminate, */
+    ts = 32;			/* so make a guess */
+#else
+  ts = getdtablesize();
+#endif
+
+  /*
+   * close all descriptors, but first check the fds array to see if this one
+   * is an exception
+   */
+  for (i = 0; i < ts; i++)
+  {
+    for (j = 0; j < nfds; j++)
+      if (i == fds[j])
+	break;
+    if (j == nfds)
+      close(i);
+  }
+}
+
+/***********************************************************************
+ * Error reporting functions taken from my library.
+ */
+
+extern int sys_nerr;
+#ifndef FreeBSD
+extern char *sys_errlist[];
+#endif
+extern int errno;
+
+void
+syserr(va_alist) va_dcl
+{
+  va_list args;
+  int rc;
+
+  va_start(args);
+  rc = va_arg(args, int);
+  err(args);
+  if (errno < sys_nerr)
+    fprintf(stderr, "     system message: %s\n", sys_errlist[errno]);
+
+  exit(rc);
+}
+
+void
+warnerr(va_alist) va_dcl
+{
+  va_list args;
+
+  va_start(args);
+  err(args);
+}
+
+void
+fatlerr(va_alist) va_dcl
+{
+  va_list args;
+  int rc;
+
+  va_start(args);
+  rc = va_arg(args, int);
+  err(args);
+
+  exit(rc);
+}
+
+void
+err(args)
+  va_list args;
+{
+
+  char *func, *fmt;
+
+  if (program != 0)
+    fprintf(stderr, "%s", program);
+  func = va_arg(args, char *);
+  if (func != 0 && strcmp(func, "") != 0)
+    fprintf(stderr, "(%s)", func);
+  fprintf(stderr, ": ");
+
+  fmt = va_arg(args, char *);
+  vfprintf(stderr, fmt, args);
+  fputc('\n', stderr);
+  fflush(stderr);
+}
+
+char *
+lasterr()
+{
+  if (errno < sys_nerr)
+    return sys_errlist[errno];
+  else
+    return "No message text for this error.";
+}
+
+/*---------------------[ prints the usage of listen ]---------------------*/
+
+void 
+printlistenUsage(char name[])
+{
+  int x;
+  char message[][255] = {
+    "\n\t'%s [options]'\n\n",
+    "Options:\n",
+    "\t-h      help (this usage message)\n",
+    "\t-p      port other than default port\n",
+    "\t-k n    use n as shared memory key number\n\n",
+    "Any unrecognized options are passed through to ntserv. For an up\n",
+    "to date listing of available ntserv options check the binary.\n",
+    "\nNOTE: in league play you must start up two listen processes, ",
+    "one for each port.\n\n",
+    "\0"
+  };
+
+  fprintf(stderr, "-- NetrekII (Paradise), %s --\n", PARAVERS);
+  for (x = 0; *message[x] != '\0'; x++)
+    fprintf(stderr, message[x], name);
+
+  exit(1);
+}
+
+/*--------------------------[ printlistenUsage ]--------------------------*/
+
+/* set the pid logfile (BG) */
+void 
+print_pid()
+{
+  char *fname;
+  FILE *fptr;
+
+  fname = build_path("logs/listen.pid");
+  fptr = fopen(fname, "w+");
+  fprintf(fptr, "%d", getpid());
+  fclose(fptr);
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/main.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,1211 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+char binary[] = "@(#)ntserv";
+
+#include "config.h"
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <sys/types.h>
+#include <signal.h>
+#include <setjmp.h>
+#include <pwd.h>
+#include <sys/time.h>
+#include <sys/wait.h>
+#include <sys/resource.h>
+#include <unistd.h>
+
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "packets.h"
+#include "shmem.h"
+#include "path.h"
+
+#if 0
+#define D(s) do { s } while(0)
+#else
+#define D(s)
+#endif
+
+
+#if 0
+jmp_buf env;
+#endif
+
+int startTkills, startTlosses, startTarms, startTplanets, startTticks;
+char start_login[16];		/* change 1/25/91 TC */
+char start_name[16];		/* change 1/25/91 TC */
+int goAway = 0;			/* change 4/14/91 TC */
+int ignored[MAXPLAYER];		/* change 7/24/91 TC */
+
+int overload = 0;		/* global 7/31/91 TC */
+/* overload indicates a request for a reserved slot */
+
+int indie = 0;			/* always be indie 8/28/91 TC */
+
+
+extern void load_time_access();
+void printntservUsage();
+extern pid_t getpid();
+extern int connectToClient();
+extern int checkSocket();
+extern int initClientData();
+extern int socketPause();
+extern int readFromClient();
+extern int checkVersion();
+extern int findslot();
+extern void updateSelf();
+extern void updateShips();
+extern void updatePlanets();
+extern void updateTerrain();
+extern void flushSockBuf();
+extern int getname();
+extern int sendShipCap();
+extern int logEntry();
+extern int getEntry();
+void printStats();
+void exitGame();
+#if defined(sparc) && !defined(SVR4)
+int atexitfunc( /* int, caddr_t */ );
+#else
+void atexitfunc();
+#endif
+extern void (*r_signal()) ();
+extern int enter();
+extern int input();
+extern int setitimer();
+extern int pmessage2();
+extern int savestats();
+extern void move_player();
+extern int sendMotdLine();
+extern int time_access();
+void doMotdPics();
+extern int numPlanets();
+extern int sendMotdPic();
+extern int ParseXbmFile();
+
+int
+main(argc, argv)
+  int argc;
+  char **argv;
+{
+  int intrupt();
+  char *getenv();
+  int team, s_type;
+  int pno;
+  int usage = 0;		/* Flag saying tell usage */
+  int errorc = 0;		/* Error count */
+  char *errorv[5];		/* Error Vector (cannot have more than 5) */
+  char *name, *ptr;
+  int reaper();
+  int callHost = 0;
+  long starttime;
+  enum HomeAway homeaway = NEITHER;
+  int observer = 0;
+
+  argv0 = argv[0];
+
+  pno = time(NULL);
+
+
+  /* load_time_acess - read in the hours file, Larry's. */
+
+  load_time_access();
+
+  name = *argv++;
+  argc--;
+  if ((ptr = strrchr(name, '/')) != NULL)
+    name = ptr + 1;
+  while (*argv)
+  {
+    if (**argv == '-')
+      ++* argv;
+    else
+      break;
+
+    argc--;
+    ptr = *argv++;
+    while (*ptr)
+    {
+      switch (*ptr)
+      {
+       case 'u':		/* for old times sake */
+       case '-':		/* this will help the --help people */
+       case 'h':
+	usage++;
+	break;
+       case 'i':
+	indie++;
+	break;			/* 8/28/91 TC */
+       case 'R':
+	if (getuid() == geteuid())
+	  overload++;
+	break;
+       case 's':
+	xtrekPort = atoi(*argv);
+	callHost = 1;
+	argv++;
+	argc--;
+	break;
+       case 'M':
+	blk_metaserver = 1;
+	break;
+       case 'd':
+	host = *argv;
+	argc--;
+	argv++;
+	break;
+	/* netrek league stuff */
+       case 'O':
+	observer = 1;
+	break;
+       case 'H':
+	homeaway = HOME;
+	break;
+       case 'A':
+	homeaway = AWAY;
+	break;
+       default:
+	{
+	  char buffer[100];
+	  sprintf(buffer, "Unknown option '%c'\n", *ptr);
+	  errorv[errorc++] = buffer;
+	  break;
+	}
+      }
+      if (usage)
+	break;
+      ptr++;
+    }
+  }
+
+  if (usage || errorc)
+  {
+    int x;
+    char message[][255] = {
+      "\n\t'%s [options] -s <socket number> <display address>'\n\n",
+      "Options:\n",
+      "\t-h   Help (this usage message)\n",
+      "\t-i   Team independant\n",
+      "\t-R   Reserved slot\n",
+      "\t-s   Socket number\n",
+      "\t-M   Metaserver\n",
+      "\t-d   Display\n",
+      "\t-O   Observer\n",
+      "\t-H   Home (League Play)\n",
+      "\t-A   Away (League Play)\n",
+      "\nNOTE: %s is designed to be launched by the startup process\n\n",
+      "\0"
+    };
+
+    fprintf(stderr, "-- NetrekII (Paradise), %s --\n", PARAVERS);
+    for (x = 0; x < errorc; x++)
+      fprintf(stderr, "\n%s: %s", argv0, errorv[x]);
+    for (x = 0; *message[x] != '\0'; x++)
+      fprintf(stderr, message[x], argv0);
+
+    exit(1);
+  }
+
+  openmem(1, homeaway != NEITHER);
+
+  /* compatability */
+  if (argc > 0)
+    host = argv[0];
+  srand48(getpid() + time((time_t *) 0));
+  /* this finds the shared memory information */
+
+#if 0
+  if (blk_metaserver)
+  {
+    FILE *ptr;
+    char *buf;
+    buf = build_path("logs/metaserver.log");
+    ptr = fopen(buf, "a");
+    fprintf(ptr, "Connection from meta-server\n");
+    fclose(ptr);
+  }
+#endif
+
+
+  me = NULL;			/* UDP fix (?) */
+  if (callHost)
+  {
+    if (!connectToClient(host, xtrekPort))
+    {
+      exit(0);
+    }
+  }
+  else
+  {
+    sock = 0;			/* Because we were forked by inetd! */
+    checkSocket();
+    initClientData();		/* "normally" called by connectToClient() */
+  }
+
+  starttime = time(NULL);
+  while (userVersion == 0)
+  {
+    /*
+     * Waiting for user to send his version number. We give him ten seconds
+     * to do so...
+     */
+    if (starttime + 10 < time(NULL))
+    {
+      exit(1);
+    }
+    socketPause();
+    readFromClient();
+  }
+  if (!checkVersion())
+    exit(1);
+
+  pno = findslot(overload, homeaway);
+  if (pno < 0)
+  {
+    /* print some appropriate message */
+    exit(1);
+  }
+
+#if defined(sparc) && !defined(SVR4)
+  on_exit(atexitfunc, (caddr_t) 0);
+#else
+  atexit(atexitfunc);		/* register a function to execute at exit */
+#endif
+
+  me = &players[pno];
+  me->p_no = pno;
+  me->p_team = NOBODY;
+  me->p_stats.st_royal = 0;
+#ifdef	RC_DISTRESS
+  me->gen_distress = 0;		/* default to RCD off */
+#endif
+  me->p_ntspid = getpid();
+  myship = &me->p_ship;
+  mystats = &me->p_stats;
+  lastm = mctl->mc_current;
+  me->p_lastrefit = -1;
+  me->p_spyable = 1;
+  me->p_teamspy = ~0;
+#if 0
+  me->p_planfrac = 0;		/* reset fractional parts */
+  me->p_bombfrac = 0;		/* reset fractional parts */
+#endif
+
+  /* --------------------------[ CLUECHECK stuff ]-------------------------- */
+#ifdef CLUECHECK1
+  me->p_cluedelay = 10;
+  me->p_cluecountdown = 0;
+#endif
+
+#ifdef CLUECHECK2
+  me->p_cluedelay = lrand48() % 1000;	/* so it doesn't ask them immediately */
+  me->p_cluecountdown = 0;
+#endif
+  /* ----------------------------------------------------------------------- */
+
+#ifndef AUTHORIZE
+  strcpy(RSA_client_type, "server doesn't support RSA");
+#endif
+
+  (void) r_signal(SIGINT, SIG_IGN);
+  (void) r_signal(SIGCHLD, reaper);
+
+  /*
+   * We set these so we won't bother updating him on the location of the
+   * other players in the galaxy which he is not near.  There is no real harm
+   * to doing this, except that he would then get more information than he
+   * deserves. It is kind of a hack, but should be harmless.
+   */
+  me->p_x = -100000;
+  me->p_y = -100000;
+  me->p_homeaway = homeaway;
+  me->p_observer = observer;
+
+#if 0
+  updateGameparams();
+#endif
+
+  updateSelf();			/* so he gets info on who he is */
+  updateShips();		/* put this back so maybe something will work */
+  /* with Andy's meta-server */
+
+  if (!blk_metaserver)
+  {
+    updateStatus();
+    updatePlanets();
+    updateTerrain();
+  }
+#if 1
+  updateGameparams();
+#endif
+
+  flushSockBuf();
+
+  /* Get login name */
+
+#if 0
+  if ((pwent = getpwuid(getuid())) != NULL)
+    (void) strncpy(login, pwent->pw_name, sizeof(login));
+  else
+#endif
+    (void) strncpy(login, "Bozo", sizeof(login));
+  login[sizeof(login) - 1] = '\0';
+
+  strcpy(pseudo, "Guest");
+
+  strcpy(me->p_name, pseudo);
+  me->p_team = ALLTEAM;
+  getname();
+  if (me->p_stats.st_rank >= NUMRANKS)
+    me->p_stats.st_rank = NUMRANKS - 1;
+  if (me->p_stats.st_royal >= NUMROYALRANKS)
+    me->p_stats.st_royal = NUMROYALRANKS - 1;
+  strcpy(pseudo, me->p_name);
+  strcpy(start_name, me->p_name);	/* change 1/25/91 TC */
+
+  sendShipCap();		/* KAO 1/25/93 */
+
+  keeppeace = (me->p_stats.st_flags & ST_KEEPPEACE) == ST_KEEPPEACE;
+
+  /*
+   * Set p_hostile to hostile, so if keeppeace is on, the guy starts off
+   * hating everyone (like a good fighter should)
+   */
+  me->p_hostile = (FED | ROM | KLI | ORI);
+  s_type = CRUISER;
+  me->p_planets = 0;
+  me->p_armsbomb = 0;
+  me->p_dooshes = 0;
+  me->p_resbomb = 0;
+  /* Set up a reasonable default */
+  me->p_whydead = KQUIT;
+
+  (void) strncpy(me->p_login, login, sizeof(me->p_login));
+  me->p_login[sizeof(me->p_login) - 1] = '\0';
+  strcpy(start_login, login);	/* change 1/25/91 TC */
+  {
+    int i;
+    for (i = 0; i < MAXPLAYER; i++)
+      ignored[i] = 0;
+  }
+
+  (void) strncpy(me->p_monitor, host, sizeof(me->p_monitor));
+  me->p_monitor[sizeof(me->p_monitor) - 1] = '\0';
+
+  /* assume this is only place p_monitor is set, and mirror accordingly */
+  /* 4/13/92 TC */
+  (void) strncpy(me->p_full_hostname, host, sizeof(me->p_full_hostname));
+  me->p_full_hostname[sizeof(me->p_full_hostname) - 1] = '\0';
+
+  logEntry();			/* moved down to get login/monitor 2/12/92
+				 * TMC */
+
+  me->p_avrt = -1;		/* ping stats */
+  me->p_stdv = -1;
+  me->p_pkls = -1;
+
+  startTkills = me->p_stats.st_tkills;
+  startTlosses = me->p_stats.st_tlosses;
+  startTarms = me->p_stats.st_tarmsbomb;
+  startTplanets = me->p_stats.st_tplanets;
+  startTticks = me->p_stats.st_tticks;
+
+  r_signal(SIGHUP, exitGame);	/* allows use of HUP to force a clean exit */
+
+
+
+  me->p_status = POUTFIT;
+  repCount = 0;
+
+  while (1)
+  {
+    switch (me->p_status)
+    {
+     case POUTFIT:
+     case PTQUEUE:
+      /* give the player the motd and find out which team he wants */
+      if (me->p_status != PALIVE)
+      {
+	me->p_x = -100000;
+	me->p_y = -100000;
+	updateSelf();
+	updateShips();
+	teamPick = -1;
+	flushSockBuf();
+	getEntry(&team, &s_type);
+      }
+      if (goAway)
+      {				/* change 4/14/91 TC */
+	printStats();
+	exit(0);
+      }
+      if (team == -1)
+      {
+	exitGame();
+      }
+
+      if (indie)
+	team = 4;		/* force to independent 8/28/91 TC */
+      inputMask = -1;		/* Allow all input now */
+      enter(team, 0, pno, s_type, -1);
+      /* for (i = 0; i < NSIG; i++) { r_signal(i, SIG_IGN); } */
+
+      me->p_status = me->p_observer ? POBSERVE : PALIVE;	/* Put player in game */
+      me->p_ghostbuster = 0;
+      break;
+     case PALIVE:
+     case PEXPLODE:
+     case PDEAD:
+     case POBSERVE:
+      /* Get input until the player quits or dies */
+
+
+      input();
+      break;
+     default:
+      if (tmpPick != PATROL)
+      {
+	printf("player status = %d.  exiting\n", me->p_status);
+	exitGame();
+      }
+    }
+  }
+
+  /* NOTREACHED */
+  return 1;
+}
+
+extern int setflag();		/* input.c */
+
+int interrupting = 0;
+
+void
+stop_interruptor()
+{
+  struct itimerval udt;
+
+  if (!interrupting)
+    return;
+
+  r_signal(SIGALRM, SIG_IGN);	/* set up signals */
+  udt.it_interval.tv_sec = 0;
+  udt.it_interval.tv_usec = 0;
+  udt.it_value.tv_sec = 0;
+  udt.it_value.tv_usec = 0;
+  setitimer(ITIMER_REAL, &udt, 0);
+
+  interrupting = 0;
+}
+
+void
+start_interruptor()
+{
+  struct itimerval udt;
+
+  if (interrupting)
+    return;
+
+  {
+    int min_delay = me->p_observer
+    ? configvals->min_observer_upd_delay
+    : configvals->min_upd_delay;
+
+    if (timerDelay < min_delay)
+      timerDelay = min_delay;
+  }
+
+  r_signal(SIGALRM, SIG_IGN);	/* set up signals */
+  udt.it_interval.tv_sec = 0;
+  udt.it_interval.tv_usec = timerDelay;
+  udt.it_value.tv_sec = 0;
+  udt.it_value.tv_usec = timerDelay;
+  setitimer(ITIMER_REAL, &udt, 0);
+
+  r_signal(SIGALRM, setflag);
+
+  interrupting = 1;
+}
+
+#if 0
+
+/*
+ * this is TOTALLY untested.  It probably doesn't belong in this file,
+ * either.  When I figure out all the mechanisms, this will replace the while
+ * loop in main().  RF
+ */
+
+void
+inputloop()
+{
+  static int switching = -1;	/* is the player changing teams? */
+
+  int status = me->p_status;
+
+  while (1)
+  {
+    switch (status)
+    {
+     case PFREE:
+      status = me->p_status = PDEAD;
+      me->p_explode = 600;
+      stop_interruptor();
+      break;
+
+     case POUTFIT:
+     case PTQUEUE:
+      updateSelf();
+      updateShips();
+      sendMaskPacket(tournamentMask(me->p_team));
+      briefUpdateClient();
+      teamPick = -1;
+      socketPause();
+      break;
+
+     case PEXPLODE:
+     case PDEAD:
+      inputMask = 0;
+     case PALIVE:
+      socketWait();
+      delay_interrupt();	/* don't let client read be interrupted by
+				 * the interval timer */
+      /*
+       * ^-- won't work.. delay_interrupt doesn't stop ALRMs from happening
+       * (HAK 9/21)
+       */
+      break;
+    }
+
+    /* me->p_status could change in here. */
+    readFromClient();
+
+    if (isClientDead())
+    {
+      if (!reconnect())
+      {
+	/* me->p_status=PFREE; should this be done?  RF */
+	exit(0);
+      }
+    }
+    switch (status)
+    {
+     case POUTFIT:
+     case PTQUEUE:
+      if (teamPick == -1)
+	break;
+
+      if (teamPick < 0 || teamPick > 3 /* XXX */ )
+      {
+	warning("That is not a valid team.");
+	sendPickokPacket(0);
+      }
+      else if (!(tournamentMask(me->p_team) & (1 << teamPick)))
+      {
+	warning("I cannot allow that.  Pick another team");
+	sendPickokPacket(0);
+      }
+      else if (((1 << teamPick) != me->p_team) &&
+	       (me->p_team != ALLTEAM) &&
+	       switching != teamPick &&
+	       me->p_whydead != KGENOCIDE)
+      {				/* switching teams */
+	switching = teamPick;
+	warning("Please confirm change of teams.  Select the new team again.");
+	sendPickokPacket(0);
+      }
+      else if (shipPick < 0 || shipPick >= NUM_TYPES)
+      {
+	/* His team choice is ok. */
+	warning("That is an illegal ship type.  Try again.");
+	sendPickokPacket(0);
+      }
+      else if (!allowed_ship(1 << teamPick, mystats->st_rank, mystats->st_royal, shipPick))
+      {
+	sendPickokPacket(0);
+      }
+      else
+      {
+
+	if (goAway)
+	{			/* what does this do ? */
+	  printStats();
+	  exit(0);
+	}
+	if (indie)
+	  me->p_team = 4;
+
+	inputMask = -1;
+
+	enter(teamPick, 0, me->p_no, shipPick, -1);
+
+	status = me->p_status = me->p_observer ? POBSERVER : PALIVE;
+	start_interruptor();	/* since we're alive, we need regular
+				 * interrupts now */
+
+	me->p_ghostbuster = 0;
+
+	repCount = 0;
+
+      }
+      break;
+     case PALIVE:
+     case PEXPLODE:
+     case PDEAD:
+      if (0 /* sendflag */ )
+      {				/* this is still busted, ugh */
+	check_authentication();
+	if (me->p_status == PFREE)
+	{
+	  me->p_ghostbuster = 0;
+	  me->p_status = PDEAD;
+	}
+	if ((me->p_status == PDEAD || me->p_status == POUTFIT)
+	    && (me->p_ntorp <= 0)
+	    && (me->p_nplasmatorp <= 0))
+	{
+	  stop_interruptor();
+	  death();
+	  status = POUTFIT;
+	}
+	auto_features();
+	updateClient();
+
+	/* sendflag=0;   ugh, broke */
+      }
+      reenable_interrupt();
+    }
+  }
+}
+
+#endif
+
+
+void
+exitGame()
+{
+  char buf[80];
+  char addrbuf[20];
+
+  if (me != NULL && me->p_team != ALLTEAM)
+  {
+    sprintf(buf, "%s %s (%s) leaving game (%.16s@%.32s)",
+	    ((me->p_stats.st_royal) ? royal[me->p_stats.st_royal].name
+	     : ranks[me->p_stats.st_rank].name),
+	    me->p_name,
+	    twoletters(me),
+	    me->p_login,
+	    me->p_full_hostname
+      );
+    sprintf(addrbuf, " %s->ALL", twoletters(me));
+    pmessage2(buf, 0, MALL | MLEAVE, addrbuf, me->p_no);
+    me->p_stats.st_flags &= ~ST_CYBORG;	/* clear this flag 8/27/91 TC */
+    savestats();
+    printStats();
+  }
+  me->p_status = PFREE;
+  move_player(me->p_no, -1, -1, 1);
+  exit(0);
+}
+
+#if defined(sparc) && !defined(SVR4)
+int
+atexitfunc(status, arg)
+  int status;
+  caddr_t arg;
+#else
+void
+atexitfunc()
+#endif
+{
+  me->p_ntspid = 0;
+  me->p_status = PFREE;
+}
+
+#define	PLURAL(n)	(((n)==1)?"":"s")
+
+static char *weapon_types[WP_MAX] = {
+  "Plasma torpedos",
+  "Tractors",
+  "Missiles",
+  "Fighters",
+};
+
+void
+sendSysDefs()
+{
+  char buf[200], buf2[200];
+  int i;
+
+  sendMotdLine("\t@@@");
+
+  if (!time_access())
+  {
+    sendMotdLine("** WE ARE CLOSED, CHECK HOURS **");
+    sendMotdLine("");
+  }
+
+  sendMotdLine("Available ship types:");
+  buf[0] = 0;
+  for (i = 0; i < NUM_TYPES; i++)
+  {
+    struct ship *s = &shipvals[i];
+    if (!shipsallowed[i])
+      continue;
+    sprintf(buf2, "   %c) %s/%c%c", s->s_letter, s->s_name,
+	    s->s_desig1, s->s_desig2);
+    if (strlen(buf) + strlen(buf2) > 80)
+    {
+      sendMotdLine(buf);
+      strcpy(buf, buf2);
+    }
+    else
+    {
+      strcat(buf, buf2);
+    }
+  }
+  /* guaranteed to have stuff here */
+  sendMotdLine(buf);
+  sendMotdLine("");
+  sendMotdLine(
+	       "SHIP         REQUIRED RANK   BUILD TIME    LIMIT     NUM PLYRS  NUM PLNTS");
+
+  for (i = 0; i < NUM_TYPES; i++)
+  {
+    struct ship *s = &shipvals[i];
+    if (!shipsallowed[i])
+      continue;
+    buf2[0] = 0;
+    if (s->s_rank > 0 || s->s_timer > 0 || s->s_maxnum < 16 ||
+	s->s_numdefn || s->s_numplan)
+    {
+      sprintf(buf2, "%-13s%-16s", s->s_name,
+	      s->s_rank ? ranks[s->s_rank].name : "none");
+      if (s->s_timer > 0)
+	sprintf(buf, "%d minutes", s->s_timer);
+      else
+	strcpy(buf, "none");
+      sprintf(buf2 + strlen(buf2), "%-14s", buf);
+
+      if (s->s_maxnum < 16)
+	sprintf(buf, "%d/team", s->s_maxnum);
+      else
+	strcpy(buf, "none");
+      sprintf(buf2 + strlen(buf2), "%-12s  %-3d        %-3d", buf,
+	      s->s_numdefn, s->s_numplan);
+      sendMotdLine(buf2);
+    }
+  }
+  sendMotdLine("");
+
+
+  buf2[0] = 0;
+  for (i = 0; i < WP_MAX; i++)
+  {
+    if (weaponsallowed[i])
+    {
+      if (buf2[0])
+	strcat(buf2, ", ");
+      strcat(buf2, weapon_types[i]);
+    }
+  }
+  sprintf(buf, "Special weapons enabled: %s", buf2[0] ? buf2 : "none");
+  sendMotdLine(buf);
+
+  if (weaponsallowed[WP_PLASMA])
+  {
+    sprintf(buf, "You need %.1f kill%s to get plasma torpedos",
+	    configvals->plkills, PLURAL(configvals->plkills));
+    sendMotdLine(buf);
+  }
+  if (weaponsallowed[WP_MISSILE])
+  {
+    sprintf(buf, "You need %.1f kill%s to get missiles",
+	    configvals->mskills, PLURAL(configvals->mskills));
+    sendMotdLine(buf);
+  }
+  sendMotdLine("");
+
+  sprintf(buf, "Tournament mode requires %d player%s per team",
+	  configvals->tournplayers, PLURAL(configvals->tournplayers));
+  sendMotdLine(buf);
+  /* sendMotdLine(""); */
+
+  /* We don't blab about newturn */
+#if 0
+  sendMotdLine(configvals->hiddenenemy ?
+	       "Visibility is restricted during T-mode" :
+	       "Visibility is unlimited all the time");
+#endif
+  sendMotdLine(configvals->binconfirm ?
+	       "Only authorized binaries are allowed" :
+	       "Non-authorized binaries are detected but not rejected");
+
+  if (configvals->planetlimittype)
+  {
+    sprintf(buf, "Independent planets may be taken if your team has fewer than %d planets", configvals->planetsinplay);
+    sendMotdLine(buf);
+  }
+  else
+  {
+    sprintf(buf, "Only %d planets can be in play at once",
+	    configvals->planetsinplay);
+    sendMotdLine(buf);
+  }
+
+  if (configvals->planupdspd == 0)
+  {
+    sendMotdLine("Planets do not orbit their stars");
+  }
+  else
+  {
+    sprintf(buf, "Planets orbit their stars at a rate of %g",
+	    configvals->planupdspd);
+    sendMotdLine(buf);
+  }
+  sendMotdLine(configvals->warpdecel ?
+	       "New warp deceleration code is in effect" :
+	       "Old-style instant warp deceleration is in effect");
+  sprintf(buf, "The next galaxy will be generated using method #%d",
+	  configvals->galaxygenerator);
+  sendMotdLine(buf);
+  sendMotdLine
+    (configvals->affect_shiptimers_outside_T ?
+     "Ship deaths outside tournament mode affect construction timers" :
+     "Construction timers are not affected by ship deaths outside tournament mode");
+
+  sprintf(buf, "Cloaking during warp prep is %sallowed",
+	  configvals->cloakduringwarpprep ? "" : "not ");
+  sendMotdLine(buf);
+
+  sprintf(buf, "Cloaking during warp is %sallowed",
+	  configvals->cloakwhilewarping ? "" : "not ");
+  sendMotdLine(buf);
+
+  sprintf(buf, "Variable warp speed is %sabled",
+	  configvals->variable_warp ? "en" : "dis");
+  sendMotdLine(buf);
+
+  sprintf(buf, "Warp prep suspension is %sallowed",
+	  configvals->warpprep_suspendable ? "" : "not ");
+  sendMotdLine(buf);
+
+  switch (configvals->warpprepstyle)
+  {
+   case WPS_NOTRACT:
+    sendMotdLine("Tractors do not affect warp prep");
+    break;
+   case WPS_TABORT:
+    sendMotdLine("Tractors make warp fail to engage");
+    break;
+   case WPS_TPREVENT:
+    sendMotdLine("Tractors prevent entering warp");
+    break;
+   case WPS_TABORTNOW:
+    sendMotdLine("Tractors abort warp prep countdown");
+    break;
+   case WPS_TSUSPEND:
+    sendMotdLine("Tractors suspend warp prep countdown");
+    break;
+  }
+
+  sprintf(buf, "There is a %d%% chance that you'll orbit a planet CCW",
+	  (int) ((1.0 - configvals->orbitdirprob) * 100));
+  sendMotdLine(buf);
+
+  sprintf(buf, "Army growth: %d.  Pop choice: %d.  Pop speed: %d%%.",
+	configvals->popscheme, configvals->popchoice, configvals->popspeed);
+  sendMotdLine(buf);
+
+  if (configvals->warpzone)
+  {
+    sprintf(buf, "Warp zones are enabled with radius %d.",
+	    configvals->warpzone);
+  }
+  else
+  {
+    sprintf(buf, "Warp zones are disabled.");
+  }
+  sendMotdLine(buf);
+}
+
+void 
+sendMotd()
+{
+  FILE *motd;
+  char buf[100], buf2[30];	/* big enough... */
+  char *paths;
+
+  time_t curtime;
+  struct tm *tmstruct;
+  int hour, tacc;
+
+  time(&curtime);
+  tmstruct = localtime(&curtime);
+  if (!(hour = tmstruct->tm_hour % 12))
+    hour = 12;
+  sprintf(buf, "Netrek II (Paradise) server %s, connection established at %d:%02d%s.",
+	  PARAVERS,
+	  hour,
+	  tmstruct->tm_min,
+	  tmstruct->tm_hour >= 12 ? "pm" : "am");
+
+  /*
+   * if (!(tacc = time_access())) strcat(buf, "  WE'RE CLOSED, CHECK HOURS");
+   */
+  sendMotdLine(buf);
+  sendMotdLine(" ");
+
+  if (!blk_flag)
+  {
+    paths = build_path(WCMOTD);	/* Wrong client message */
+    if ((motd = fopen(paths, "r")) != NULL)
+    {
+      while (fgets(buf, sizeof(buf), motd) != NULL)
+      {
+	buf[strlen(buf) - 1] = '\0';
+	sendMotdLine(buf);
+      }
+      fclose(motd);
+    }
+    else
+    {				/* default message */
+      sendMotdLine(" ");
+      sendMotdLine(
+		   "      ****************************************************************");
+      sendMotdLine(" ");
+
+      sendMotdLine(
+		   "       This is a Paradise server; you need a Paradise client to play!");
+      sendMotdLine(" ");
+      sendMotdLine(
+		   "       Paradise clients can be had from");
+      sendMotdLine(
+		   "          ftp.cis.ufl.edu    pub/netrek.paradise/");
+      sendMotdLine(
+		   "          ftp.reed.edu       mirrors/netrek.paradise/");
+      sendMotdLine(" ");
+      sendMotdLine(
+		   "      ****************************************************************");
+    }
+    return;
+  }
+
+   /* if (blk_flag) */ 
+  {				/* added 1/19/93 KAO */
+    int i, first = 1;
+
+    strcpy(buf, "BLK: REFIT ");
+
+    for (i = 0; i < NUM_TYPES; i++)
+    {
+      struct ship *s = &shipvals[i];
+      if (!shipsallowed[i])
+	continue;
+
+      if (!first)
+	strcat(buf, ", ");
+      else
+	first = 0;
+
+      sprintf(buf2, "%c) %c%c", s->s_letter, s->s_desig1, s->s_desig2);
+      strcat(buf, buf2);
+    }
+    sendMotdLine(buf);
+  }
+  /* the following will read a motd */
+  if (!time_access())
+  {
+    paths = build_path(CLOSEDMOTD);
+    if ((motd = fopen(paths, "r")) == NULL)
+    {
+      paths = build_path(MOTD);
+      motd = fopen(paths, "r");
+    }
+  }
+  else
+  {
+    paths = build_path(MOTD);
+    motd = fopen(paths, "r");
+  }
+  if (motd != NULL)
+  {
+#ifdef CLUECHECK1
+    init_motdbuf(paths);
+#endif
+    while (fgets(buf, sizeof(buf), motd) != NULL)
+    {
+      buf[strlen(buf) - 1] = '\0';
+      sendMotdLine(buf);
+    }
+    (void) fclose(motd);
+  }
+  sendSysDefs();
+
+  /* wait till the end for the pictures */
+  if (!blk_metaserver)
+    doMotdPics();
+}
+
+int 
+reaper(sig)
+  int sig;
+{
+#ifndef SVR4
+  while (wait3((union wait *) 0, WNOHANG, (struct rusage *) 0) > 0);
+#else
+  while (waitpid(0, NULL, WNOHANG) > 0);
+#endif				/* SVR4 */
+  return 0;
+}
+
+void 
+printStats()
+{
+  FILE *logfile;
+#if defined(SVR4) || defined(sparc)
+  time_t curtime;
+#else
+  int curtime;
+#endif				/* SVR4 */
+  char *paths;			/* added 1/18/93 KAO */
+
+  paths = build_path(LOGFILENAME);
+  logfile = fopen(paths, "a");
+  if (!logfile)
+    return;
+  curtime = time(NULL);
+
+#ifdef LOG_LONG_INFO		/*-[  Long info printed to logfiles and startup.log ]-*/
+  fprintf(logfile, "Leaving: %-16s (%s) %3dP %3dA %3dW/%3dL %3dmin %drtt %dsdv %dls %dplan <%s@%s> %s",
+	  me->p_name,
+	  twoletters(me),
+	  me->p_stats.st_tplanets - startTplanets,
+	  me->p_stats.st_tarmsbomb - startTarms,
+	  me->p_stats.st_tkills - startTkills,
+	  me->p_stats.st_tlosses - startTlosses,
+	  (me->p_stats.st_tticks - startTticks) / 600,
+	  me->p_avrt, me->p_stdv, me->p_pkls,
+	  numPlanets(me->p_team),
+
+#else
+
+  fprintf(logfile, "Leaving: %s <%s@%s> %s",
+	  me->p_name,
+#endif
+	  me->p_login,
+	  me->p_full_hostname,
+	  ctime(&curtime));
+
+  /* #endif /*-[ LOG_LONG_INFO ]- */
+
+  if (goAway)
+    fprintf(logfile, "^^^ 2 players/1 slot.  was %s (%s)\n",
+	    start_name, start_login);
+  fclose(logfile);
+}
+
+
+/*
+ * .pics file format:
+ * 
+ * name x y page name x y page etc
+ */
+
+void
+doMotdPics()
+{
+  FILE *ptr, *ftemp;
+  char buf[128], fname[128];
+  unsigned char *bits;
+  char *result;
+  int x, y, page, w, h;
+  int bytesperline;		/* pad the width to a byte */
+  int linesperblock;		/* how many lines in 1016 bytes? */
+  int i;
+  char *paths;
+
+  paths = build_path(PICS);
+  ptr = fopen(paths, "r");
+  if (ptr == NULL)
+    return;
+  while (1)
+  {
+    result = fgets(fname, 125, ptr);
+    if (result == NULL)
+      /* must fclose ptr */
+      break;
+
+    if (fname[strlen(fname) - 1] == '\n')
+      fname[strlen(fname) - 1] = 0;
+
+    paths = build_path(fname);
+    ftemp = fopen(paths, "r");
+    bits = 0;
+    if (ftemp == 0)
+    {
+      fprintf(stderr, "ntserv: couldn't open file %s.  skipping\n", paths);
+    }
+    else
+    {
+      ParseXbmFile(ftemp, &w, &h, &bits);	/* parsexbm.c */
+
+      bytesperline = (w - 1) / 8 + 1;
+      linesperblock = 1016 /* packets.h */ / bytesperline;
+    }
+
+    fgets(buf, 125, ptr);
+
+    if (3 != sscanf(buf, "%d %d %d", &x, &y, &page))
+    {
+      printf("Format error in .pics file\n");
+      if (bits)
+	free(bits);
+      bits = NULL;
+    }
+
+    if (bits)
+    {
+      if (me != 0 && (me->p_stats.st_flags & ST_NOBITMAPS))
+      {
+	sendMotdNopic(x, y, page, w, h);
+      }
+      else
+	for (i = 0; i * linesperblock < h; i++)
+	{
+	  int nlines;
+	  if ((i + 1) * linesperblock > h)
+	    nlines = h - i * linesperblock;
+	  else
+	    nlines = linesperblock;
+#if 0
+	  printf("Sending MotdPics: %s %d %d %d %d %d\n",
+		 fname, x, y + i * linesperblock, page, w, nlines);
+#endif
+	  sendMotdPic(x, y + i * linesperblock,
+		      bits + bytesperline * linesperblock * i,
+		      page, w, nlines);
+	}
+      free(bits);
+    }
+  }
+  fclose(ptr);
+}
+
+
+#undef D
Binary file src/mes has changed
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/message.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,2513 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+/* was control_mess.c, but changed to message.c for a smaller filename (BG) */
+
+/*
+ * Ugh, this code is in the middle of a rewrite.
+ * 
+ * It used to use a tokenizer with a global dictionary to split the input into
+ * words.  The tokenizer accepted abbreviations as long as these were unique.
+ * However, adding a new word to the dictionary would often cause old
+ * abbreviations to be invalidated.
+ * 
+ * I wrote a new parser that was called as each token needed to be extracted.
+ * This used a dictionary that was local to each submenu. This localizes
+ * changes to the menu structure so that effects of adding new commands are
+ * minimized.
+ * 
+ * Some of the file is converted to use this, but not all.  Eventually the
+ * entire module will use the context-sensitive tokenizer.
+ * 
+ */
+
+
+#include "config.h"
+#include <stdio.h>
+#include <string.h>
+#include <signal.h>
+#include <ctype.h>
+#include <fcntl.h>
+
+#include "defs.h"
+#include "data.h"
+#include "struct.h"
+#include "shmem.h"
+
+enum token_names_e
+{
+  HELPTOK = 128,
+  CONTROLTOK,
+  VERSIONTOK,
+  QUEUETOK,
+  LEAGUETOK,
+  PARAMTOK,
+  INFOTOK,
+  OBSERVETOK,
+  CLUECHECKTOK,
+
+  NUKEGAMETOK,
+  FREESLOTTOK,
+  ROBOTTOK,
+  TOURNTOK,
+  NEWGALAXY,
+  SHIPTIMERTOK,
+
+  REFITTOK,
+  PLAYERTOK,
+  EJECTTOK,
+  DIETOK,
+  ARMIESTOK,
+  PLASMATOK,
+  MISSILETOK,
+  PLANETTOK,
+  RANKTOK,
+  MOVETOK,
+
+  PASSWDTOK,
+  RATINGSTOK,
+
+  TIMETOK,
+  CAPTAINTOK,
+  RESTARTTOK,
+  STARTTOK,
+  PASSTOK,
+  TIMEOUTTOK,
+  TEAMNAMETOK,
+  AWAYTOK,
+  HOMETOK,
+  PAUSETOK,
+  CONTINUETOK,
+  MAXPLAYERTOK,
+
+  TEAMTOK,
+
+  INDTOK,			/* these need to be adjacent and in order */
+  FEDTOK,			/* these need to be adjacent and in order */
+  ROMTOK,			/* these need to be adjacent and in order */
+  KLITOK,			/* these need to be adjacent and in order */
+  ORITOK,			/* these need to be adjacent and in order */
+
+  SHIPTOK,
+
+  SCOUTTOK,			/* these need to be adjacent and in order */
+  DESTROYERTOK,			/* these need to be adjacent and in order */
+  CRUISERTOK,			/* these need to be adjacent and in order */
+  BATTLESHIPTOK,		/* these need to be adjacent and in order */
+  ASSAULTTOK,			/* these need to be adjacent and in order */
+  STARBASETOK,			/* these need to be adjacent and in order */
+  ATTTOK,			/* these need to be adjacent and in order */
+  JUMPSHIPTOK,			/* these need to be adjacent and in order */
+  FRIGATETOK,			/* these need to be adjacent and in order */
+  WARBASETOK,			/* these need to be adjacent and in order */
+  LIGHTCRUISERTOK,
+  CARRIERTOK,
+  UTILITYTOK,
+  PATROLTOK,
+
+  PLUSTOK,
+  MINUSTOK,
+  ROYALTOK,
+  QUIETTOK,
+  KILLSTOK,
+  HOSETOK,
+  SUPERTOK,
+  ALLOWTOK,			/* control allow [teams] */
+
+  ERRORTOK = 255
+};
+
+#ifdef GOD_CONTROLS
+static int god_silent = 0;
+#endif
+
+/* static char *con_shipnos = "0123456789abcdefghijklmnopqrstuvwxyz"; */
+
+extern int pmessage2();
+extern int pmessage();
+#ifndef linux
+extern int atoi();
+#endif				/* linux */
+extern int detourneyqueue();
+extern void compute_ratings();
+
+static void respond();
+
+#if 0
+static enum token_names_e tokstring[128];	/* list of tokens */
+static char *thestrings[128];	/* list of corresponding strings */
+static char *therest[128];	/* list of pointers to the original text
+				 * starting with token i */
+static int sizetokstring;
+
+struct token
+{
+  char string[32];
+  int token;
+};
+
+
+
+struct token tstrings[] = {
+  {"0", 0}, {"1", 1}, {"2", 2}, {"3", 3},
+  {"4", 4}, {"5", 5}, {"6", 6}, {"7", 7},
+  {"8", 8}, {"9", 9}, {"a", 10}, {"b", 11},
+  {"c", 12}, {"d", 13}, {"e", 14}, {"f", 15},
+  {"g", 16}, {"h", 17}, {"i", 18}, {"j", 19},
+  {"k", 20}, {"l", 21}, {"m", 22}, {"n", 23},
+  {"o", 24}, {"p", 25}, {"q", 26}, {"r", 27},
+  {"s", 28}, {"t", 29}, {"u", 30}, {"v", 31},
+  {"w", 32}, {"x", 33}, {"y", 34}, {"z", 35},
+  {"help", HELPTOK},
+  {"control", CONTROLTOK},
+  {"con", CONTROLTOK},		/* popular abbrev */
+  {"version", VERSIONTOK},
+  {"tq", QUEUETOK},		/* popular abbrev */
+  {"queue", QUEUETOK},
+  {"league", LEAGUETOK},
+  {"params", PARAMTOK},
+  {"observe", OBSERVETOK},
+  {"info", INFOTOK},
+  {"cluecheck", CLUECHECKTOK},
+
+  {"shiptimers", SHIPTIMERTOK},
+
+  {"refit", REFITTOK},
+  {"player", PLAYERTOK},
+  {"pl", PLAYERTOK},		/* popular abbrev */
+  {"planet", PLANETTOK},
+  {"eject", EJECTTOK},
+  {"die", DIETOK},
+  {"rank", RANKTOK},
+  {"royalty", ROYALTOK},
+  {"move", MOVETOK},
+  {"armies", ARMIESTOK},
+  {"plasma", PLASMATOK},
+  {"missile", MISSILETOK},
+  {"team", TEAMTOK},
+  {"ship", SHIPTOK},
+  {"quiet", QUIETTOK},
+  {"kills", KILLSTOK},
+  {"hose", HOSETOK},
+  {"super", SUPERTOK},
+  {"allow", ALLOWTOK},
+  {"+", PLUSTOK},
+  {"-", MINUSTOK},
+
+  {"robot", ROBOTTOK},
+  {"nukegame", NUKEGAMETOK},
+  {"freeslot", FREESLOTTOK},
+  {"tourn", TOURNTOK},
+  {"galaxy", NEWGALAXY},
+
+  /* player commands */
+  {"password", PASSWDTOK},
+  {"ratings", RATINGSTOK},
+
+  /* league commands */
+  {"time", TIMETOK},
+  {"captain", CAPTAINTOK},
+  {"restart", RESTARTTOK},
+  {"start", STARTTOK},
+  {"pass", PASSTOK},
+  {"timeout", TIMEOUTTOK},
+  {"teamname", TEAMNAMETOK},
+  {"away", AWAYTOK},
+  {"home", HOMETOK},
+  {"pause", PAUSETOK},
+  {"continue", CONTINUETOK},
+  {"maxplayers", MAXPLAYERTOK},
+
+  {"independent", INDTOK},
+  {"federation", FEDTOK}, {"romulan", ROMTOK},
+  {"klingon", KLITOK}, {"orion", ORITOK},
+
+  {"sc", SCOUTTOK}, {"dd", DESTROYERTOK},
+  {"ca", CRUISERTOK}, {"bb", BATTLESHIPTOK},
+  {"as", ASSAULTTOK}, {"sb", STARBASETOK},
+  {"at", ATTTOK}, {"js", JUMPSHIPTOK},
+  {"fr", FRIGATETOK}, {"wb", WARBASETOK},
+  {"cl", LIGHTCRUISERTOK}, {"cv", CARRIERTOK},
+  {"ut", UTILITYTOK}, {"pt", PATROLTOK},
+
+  {"", ERRORTOK}		/* I'm last. Keep it that way. */
+};
+
+static void
+dotokenize(input)
+  char *input;
+{
+  char *temp0, *temp1;
+  int done = 0;
+  static char internal[120];
+
+  sizetokstring = 0;
+
+  /* Convert string to lowercase. */
+
+  temp0 = strcpy(internal, input);
+  while (*temp0 != '\0')
+  {
+    if (isupper(*temp0))
+      *temp0 = tolower(*temp0);
+    temp0++;
+  }
+
+
+  temp0 = internal;
+
+  while (done == 0)
+  {
+    struct token *tptr;
+    int wordlen;
+    int potentialtok, ambiguous;
+
+    /* Eat leading spaces. */
+
+    while (*temp0 == ' ')
+      temp0++;
+
+    /* Find the next word and chop the rest of the string away. */
+
+    temp1 = temp0;
+    therest[sizetokstring] = input + (temp1 - internal);
+    while (*temp1 != '\0' && *temp1 != ' ')
+      temp1++;
+    if (*temp1 != '\0')
+      *temp1++ = '\0';
+    wordlen = strlen(temp0);
+
+    /* Go find out what they've typed. */
+
+    tptr = tstrings;
+    potentialtok = ERRORTOK;
+    ambiguous = 0;
+    while (tptr->token != ERRORTOK)
+    {
+      if (0 == strncmp(temp0, tptr->string, wordlen))
+      {
+	if (strlen(tptr->string) == wordlen)
+	{
+	  ambiguous = 0;
+	  potentialtok = tptr->token;
+	  break;		/* exact match */
+	}
+	if (potentialtok != ERRORTOK)
+	{
+	  ambiguous = 1;	/* this isn't the only match */
+	}
+	potentialtok = tptr->token;
+      }
+      tptr++;
+    }
+
+    thestrings[sizetokstring] = temp0;
+    tokstring[sizetokstring++] = ambiguous ? ERRORTOK : potentialtok;
+    temp0 = temp1;
+
+    /* If *temp0 == 0 then we're done. */
+
+    if (*temp0 == '\0')
+      done = 1;
+  }
+  thestrings[sizetokstring] = 0;
+}
+#endif
+
+/**********************************************************************/
+
+/* New parsing method. */
+
+struct control_cmd
+{
+  char *literal;		/* the command they should type */
+  enum token_names_e tok;	/* what the parser should return */
+  char *doc;			/* documentation to print for a help command */
+};
+
+/*
+ * Scans the string cmd for the first whitespace delimited string. Tries to
+ * match this string in a case-insensitive manner against the list in legals.
+ * Returns the appropriate token.  Alters the char* pointed to by after to be
+ * the beginning of the next whitespace-delimited string.
+ * 
+ * cmd:	unmodified legals:	unmodified after:	MODIFIED
+ * 
+ */
+
+enum token_names_e
+next_token(cmd, legals, after)
+  char *cmd;
+  struct control_cmd *legals;
+  char **after;
+{
+  char buf[80];			/* space for the token */
+  char *s;
+  int i;
+  int ambiguous = 0;
+  enum token_names_e potentialtok = ERRORTOK;
+
+  while (*cmd && isspace(*cmd))
+    cmd++;
+
+  if (!*cmd)
+    return ERRORTOK;
+
+  for (s = buf; *cmd && !isspace(*cmd); s++, cmd++)
+    *s = *cmd;
+  *s = 0;
+
+  while (*cmd && isspace(*cmd))
+    cmd++;
+
+  if (after)
+    *after = cmd;		/* so they can find the next token */
+
+  for (i = 0; legals[i].literal; i++)
+  {
+    int wordlen = strlen(buf);
+    if (0 == strncasecmp(buf, legals[i].literal, wordlen))
+    {
+      if (strlen(legals[i].literal) == wordlen)
+      {
+	ambiguous = 0;
+	potentialtok = legals[i].tok;
+	break;			/* exact match */
+      }
+      if (potentialtok != ERRORTOK)
+      {
+	ambiguous = 1;		/* this isn't the only match */
+	return ERRORTOK;
+      }
+      potentialtok = legals[i].tok;
+    }
+  }
+
+  return potentialtok;
+}
+
+int
+match_token(cmd, token, after)
+  char *cmd;
+  char *token;
+  char **after;
+{
+  struct control_cmd legals[2];
+  legals[0].literal = token;
+  legals[0].tok = HELPTOK;	/* pick any token but ERRORTOK */
+  legals[1].literal = 0;
+  return HELPTOK == next_token(cmd, legals, after);
+}
+
+/*
+ * Get a player slot number. Returns -1 on failure, slot number on success.
+ * Slot number is guaranteed to be <MAXPLAYER.
+ */
+
+int 
+get_slotnum(cmd, after)
+  char *cmd;
+  char **after;
+{
+  int rval;
+  while (*cmd && isspace(*cmd))
+    cmd++;
+
+  if (!*cmd)
+    return -1;			/* no token */
+
+  if (cmd[1] && !isspace(cmd[1]))
+    return -1;			/* token too long */
+
+  if (*cmd >= '0' && *cmd <= '9')
+    rval = *cmd - '0';
+  else if (*cmd >= 'a' && *cmd <= 'z')
+    rval = *cmd - 'a' + 10;
+  else if (*cmd >= 'A' && *cmd <= 'Z')
+    rval = *cmd - 'A' + 10;
+  else
+    return -1;
+
+  if (rval >= MAXPLAYER)
+    return -1;			/* there aren't that many players */
+
+  if (after)
+  {
+    /* scan to next token */
+    cmd++;
+    while (*cmd && isspace(*cmd))
+      cmd++;
+    *after = cmd;
+  }
+
+  return rval;
+}
+
+void 
+bad_slotnum(msg)
+  char *msg;
+{
+  char buf[256];
+  sprintf(buf, "`%s' requires player slot number", msg);
+  respond(buf, 1);
+}
+
+
+/*
+ * Get a single token. Returns 0 on failure, 1 on success. Token is returned
+ * in dst.
+ */
+
+int 
+get_one_token(cmd, dst, dstsize, after)
+  char *cmd;
+  char *dst;			/* destination */
+  int dstsize;
+  char **after;
+{
+  while (*cmd && isspace(*cmd))
+    cmd++;
+
+  if (!*cmd)
+    return 0;			/* no token */
+
+  while (dstsize > 1 && *cmd && !isspace(*cmd))
+  {
+    *(dst++) = *(cmd++);
+    dstsize--;
+  }
+  *dst = 0;
+
+  if (after)
+  {
+    /* scan to next token */
+    while (*cmd && isspace(*cmd))
+      cmd++;
+    *after = cmd;
+  }
+
+  return 1;
+}
+
+/*
+ * Get an integer Integer is returned in dst. Returns 0 on failure without
+ * modifying dst.
+ */
+
+int 
+get_int(cmd, dst, after)
+  char *cmd;
+  int *dst;
+  char **after;
+{
+  int rval, offset;
+
+  if (1 != sscanf(cmd, " %i%n", &rval, &offset))
+    return 0;
+
+  cmd += offset;
+  if (*cmd && !isspace(*cmd))
+    return 0;			/* token wasn't all digits */
+
+  *dst = rval;
+
+  if (after)
+  {
+    /* scan to next token */
+    while (*cmd && isspace(*cmd))
+      cmd++;
+    *after = cmd;
+  }
+
+  return 1;
+}
+
+/*
+ * Get a double Double is returned in dst. Returns 0 on failure without
+ * modifying dst.
+ */
+
+int 
+get_double(cmd, dst, after)
+  char *cmd;
+  double *dst;
+  char **after;
+{
+  double rval;
+  int offset;
+
+  if (1 != sscanf(cmd, " %lg%n", &rval, &offset))
+    return 0;
+
+  cmd += offset;
+  if (*cmd && !isspace(*cmd))
+    return 0;			/* token wasn't all digits */
+
+  *dst = rval;
+
+  if (after)
+  {
+    /* scan to next token */
+    while (*cmd && isspace(*cmd))
+      cmd++;
+    *after = cmd;
+  }
+
+  return 1;
+}
+
+int 
+get_teamid(cmd, team, after)
+  char *cmd;
+  int *team;
+  char **after;
+{
+  int i, j;
+
+  while (*cmd && isspace(*cmd))
+  {
+    cmd++;
+  }
+  if (cmd[3] && !isspace(cmd[3]))
+    return 0;			/* too long */
+
+  *team = NOBODY;
+
+  for (i = -1; i < NUMTEAM; i++)
+  {
+    j = idx_to_mask(i);
+    if (0 == strncasecmp(cmd, teams[j].shortname, 3))
+    {
+      *team = i;
+      cmd += 3;
+      break;
+    }
+  }
+
+  if (after)
+  {
+    /* scan to next token */
+    while (*cmd && isspace(*cmd))
+      cmd++;
+    *after = cmd;
+  }
+
+  return i < NUMTEAM;
+}
+
+int 
+get_shipid(cmd, shipn, after)
+  char *cmd;
+  int *shipn;
+  char **after;
+{
+  int i;
+
+  while (*cmd && isspace(*cmd))
+  {
+    cmd++;
+  }
+  *shipn = -1;
+
+  for (i = 0; i < NUM_TYPES; i++)
+  {
+    struct ship *ship = &shipvals[i];
+    int len;
+    len = strlen(ship->s_name);
+    if (0 == strncasecmp(cmd, ship->s_name, len) &&
+	(cmd[len] == 0 || isspace(cmd[len])))
+    {
+      *shipn = i;
+      cmd += len;
+      break;
+    }
+    else if (tolower(cmd[0]) == tolower(ship->s_desig1) &&
+	     tolower(cmd[1]) == tolower(ship->s_desig2) &&
+	     (cmd[2] == 0 || isspace(cmd[2])))
+    {
+      *shipn = i;
+      cmd += 2;
+      break;
+    }
+  }
+
+  if (after)
+  {
+    /* scan to next token */
+    while (*cmd && isspace(*cmd))
+      cmd++;
+    *after = cmd;
+  }
+
+  return i < NUM_TYPES;
+}
+
+/* writes a comma-separated list of help strings into the message window */
+
+void
+respond_with_help_string(legals)
+  struct control_cmd *legals;
+{
+  int i;
+  char buf[65];			/* leave space for the message prefix */
+
+  strcpy(buf, "Available commands: ");
+  for (i = 0; legals[i].literal; i++)
+  {
+    if (!(legals[i].doc && legals[i].doc[0]))
+      continue;
+    if (strlen(buf) + 3 + strlen(legals[i].doc) > sizeof(buf))
+    {
+      respond(buf, 0);
+      strcpy(buf, "    ");
+      if (!buf[0])
+      {				/* one of the help strings was just too long */
+	respond("ACK! programmer error: help string too long", 0);
+	return;
+      }
+      i--;			/* retry */
+      continue;
+    }
+    strcat(buf, legals[i].doc);
+    if (legals[i + 1].literal)
+      strcat(buf, ", ");
+  }
+  if (buf[0])
+    respond(buf, 0);
+}
+
+/**********************************************************************/
+
+static void
+respond(msg, type)
+  char *msg;
+  int type;
+{
+  if (type == 1)
+    warning(msg);
+  else
+    pmessage2(msg, me->p_no, MINDIV, MCONTROL, 255);
+}
+
+#ifdef GOD_CONTROLS
+
+/*
+ * Here we handle the controls on players. If you add something, make sure
+ * you place it in the help. Thanks, have a nice day.
+ */
+
+static int
+parse_control_player(cmd)
+  char *cmd;
+{
+  char buf[120];
+  int pnum;
+  struct player *victim;
+  int godliness = me->p_stats.st_royal - GODLIKE + 1;
+  char *arg;
+  static struct control_cmd available_cmds[] = {
+    {"help", HELPTOK, 0},
+    {"die", DIETOK, "die"},
+    {"eject", EJECTTOK, "eject"},
+    {"armies", ARMIESTOK, "armies [%d=5]"},
+    {"plasma", PLASMATOK, "plasma [%d]"},
+    {"missiles", MISSILETOK, "missiles [%d=max]"},
+    {"team", TEAMTOK, "team <teamstr>"},
+    {"ship", SHIPTOK, "ship <shiptype>"},
+    {"rank", RANKTOK, "rank (+|-|%d)"},
+    {"royal", ROYALTOK, "royal (+|-|%d)"},
+    {"kills", KILLSTOK, "kills (+|-|%d)"},
+    {"hose", HOSETOK, "hose"},
+    {"move", MOVETOK, "move %d %d"},
+    {0}
+  };
+
+  pnum = get_slotnum(cmd, &cmd);
+  if (pnum < 0)
+  {
+    bad_slotnum("control player");
+    return 0;
+  }
+  victim = &players[pnum];
+
+  if (victim->p_status == PFREE)
+  {
+    respond("Slot is not alive.", 1);
+    return 1;
+  }
+
+  /*
+   * These would probably work better as pointers to functions instead of a
+   * giant switch, but what the hell, I'm lazy. Maybe I'll change it later.
+   */
+
+  switch (next_token(cmd, available_cmds, &arg))
+  {
+   case DIETOK:
+    victim->p_ship.s_type = STARBASE;
+    victim->p_whydead = KPROVIDENCE;
+    victim->p_explode = 10;
+    victim->p_status = PEXPLODE;
+    victim->p_whodead = 0;
+    if (!god_silent)
+    {
+      sprintf(buf, "%s (%2s) was utterly obliterated by %s (%2s).",
+	      victim->p_name, twoletters(victim),
+	      me->p_name, twoletters(me));
+      pmessage(buf, 0, MALL, MCONTROL);
+    }
+    return 1;
+
+   case EJECTTOK:
+    victim->p_ship.s_type = STARBASE;
+    victim->p_whydead = KQUIT;
+    victim->p_explode = 10;
+    victim->p_status = PEXPLODE;
+    victim->p_whodead = 0;
+    if (!god_silent)
+    {
+      sprintf(buf,
+	      "%s (%2s) has been ejected from the game by %s (%2s).",
+	      victim->p_name, twoletters(victim),
+	      me->p_name, twoletters(me));
+      pmessage(buf, 0, MALL, MCONTROL);
+    }
+    return 1;
+
+   case ARMIESTOK:
+    {
+      int armies = 5;
+      if (*arg && !get_int(arg, &armies, (char **) 0))
+      {
+	respond("optional arg to `control player <slotnum> armies` must be integer", 0);
+	return 0;
+      }
+      victim->p_armies += armies;
+      if (!god_silent)
+      {
+	sprintf(buf, "%s (%2s) has been given %d armies by %s (%2s).",
+		victim->p_name, twoletters(victim), armies,
+		me->p_name, twoletters(me));
+	pmessage(buf, 0, MALL, MCONTROL);
+      }
+      return 1;
+    }
+
+   case PLASMATOK:
+    {
+      int yes = 1;
+      if (*arg && !get_int(arg, &yes, (char **) 0))
+      {
+	respond("optional arg to `control player <slotnum> plasma` must be integer", 0);
+	return 0;
+      }
+
+      if (yes)
+	victim->p_ship.s_nflags |= SFNPLASMAARMED;
+      else
+	victim->p_ship.s_nflags &= ~SFNPLASMAARMED;
+
+      if (!god_silent)
+      {
+	sprintf(buf, "%s (%2s) has been %s plasma torps by %s (%2s).",
+		victim->p_name, twoletters(victim),
+		yes ? "given" : "denied",
+		me->p_name, twoletters(me));
+	pmessage(buf, 0, MALL, MCONTROL);
+      }
+      return 1;
+    }
+
+   case MISSILETOK:
+    {
+      int yes = shipvals[victim->p_ship.s_type].s_missilestored;
+
+      if (*arg && !get_int(arg, &yes, (char **) 0))
+      {
+	respond("optional arg to `control player <slotnum> missile` must be integer", 0);
+	return 0;
+      }
+
+      if (yes)
+      {
+	victim->p_ship.s_nflags |= SFNHASMISSILE;
+	victim->p_ship.s_missilestored = yes;
+      }
+      else
+      {
+	victim->p_ship.s_nflags &= ~SFNHASMISSILE;
+      }
+
+      if (!god_silent)
+      {
+	sprintf(buf, "%s (%2s) has been %s %d missiles by %s (%2s).",
+		victim->p_name, twoletters(victim),
+		yes ? "given" : "denied",
+		yes, me->p_name, twoletters(me));
+	pmessage(buf, 0, MALL, MCONTROL);
+      }
+      return 1;
+    }
+
+   case TEAMTOK:
+    {
+      int team;
+
+      if (!get_teamid(arg, &team, (char **) 0))
+      {
+	respond("available teams: FED ORI ROM KLI IND", 0);
+	return 0;
+      }
+      team = idx_to_mask(team);
+
+      victim->p_hostile |= victim->p_team;
+      victim->p_team = team;
+      victim->p_hostile &= ~team;
+      victim->p_swar &= ~team;
+      sprintf(buf, "%s (%2s) has been changed to a %s by %s (%2s).",
+	      victim->p_name, twoletters(victim), teams[team].nickname,
+	      me->p_name, twoletters(me));
+      if (!god_silent)
+	pmessage(buf, 0, MALL, MCONTROL);
+      return 1;
+    }
+
+   case SHIPTOK:
+    {
+      int ship;
+      if (!get_shipid(arg, &ship, (char **) 0))
+      {
+	respond("available ships: SC DD CA AS BB SB AT JS FR WB CL CV SUPER", 0);
+	return 0;
+      }
+#if 0
+      if (tokstring[4] == SUPERTOK)
+      {
+	victim->p_ship.s_maxshield = 750;
+	victim->p_shield = 750;
+	victim->p_ship.s_maxdamage = 750;
+	victim->p_ship.s_maxegntemp = 5000;
+	sprintf(buf, "%s (%2s) has been supercharged by %s (%2s).",
+		victim->p_name, twoletters(victim),
+		me->p_name, twoletters(me));
+	if (!god_silent)
+	  pmessage(buf, 0, MALL, MCONTROL);
+	return 1;
+      }
+#endif
+      /* If others are docked, then kick them off */
+      if (allows_docking(victim->p_ship))
+      {
+	int i;
+	for (i = 0; i < victim->p_ship.s_numports; i++)
+	{
+	  base_undock(victim, i);
+	}
+      }
+      get_ship_for_player(victim, ship);
+      switch_special_weapon();
+      victim->p_flags &= ~PFENG;
+      sprintf(buf, "%s (%2s) has been changed to a %c%c by %s (%2s).",
+	      victim->p_name, twoletters(victim),
+	      victim->p_ship.s_desig1, victim->p_ship.s_desig2,
+	      me->p_name, twoletters(me));
+      if (!god_silent)
+	pmessage(buf, 0, MALL, MCONTROL);
+      return 1;
+    }
+
+   case RANKTOK:
+    {
+      int rank = victim->p_stats.st_rank;
+
+      if (match_token(arg, "+", (char **) 0))
+	rank++;
+      else if (match_token(arg, "-", (char **) 0))
+	rank--;
+      else if (!get_int(arg, &rank, (char **) 0))
+      {
+	respond("Try: control player %d rank [%d]+-", 0);
+	return 0;
+      }
+
+      if (rank < 0)
+	rank = 0;
+      if (rank >= NUMRANKS)
+	rank = NUMRANKS - 1;
+
+      victim->p_stats.st_rank = rank;
+      sprintf(buf, "%s (%2s) has been given a rank of %s by %s (%2s).",
+	      victim->p_name, twoletters(victim),
+	      ranks[victim->p_stats.st_rank].name,
+	      me->p_name, twoletters(me));
+      if (!god_silent)
+	pmessage(buf, 0, MALL, MCONTROL);
+      return 1;
+    }
+
+   case ROYALTOK:
+    {
+      int rank = victim->p_stats.st_royal;
+
+      if (match_token(arg, "+", (char **) 0))
+	rank++;
+      else if (match_token(arg, "-", (char **) 0))
+	rank--;
+      else if (!get_int(arg, &rank, (char **) 0))
+      {
+	respond("Try: control player %d royal [%d]+-", 0);
+	return 0;
+      }
+
+      if (rank < 0)
+	rank = 0;
+      if (rank >= NUMROYALRANKS)
+	rank = NUMROYALRANKS - 1;
+
+      if (rank >= GODLIKE && godliness < 2)
+      {
+	respond("You aren't powerful enough to grant godlike royalty.",
+		1);
+	return 1;
+      }
+      victim->p_stats.st_royal = rank;
+      sprintf(buf, "%s (%2s) has been given a rank of %s by %s (%2s).",
+	      victim->p_name, twoletters(victim),
+	      royal[victim->p_stats.st_royal].name,
+	      me->p_name, twoletters(me));
+      if (!god_silent)
+	pmessage(buf, 0, MALL, MCONTROL);
+      return 1;
+    }
+
+   case KILLSTOK:
+    {
+      double kills = victim->p_kills;
+
+      if (match_token(arg, "+", (char **) 0))
+	kills += 1;
+      else if (match_token(arg, "-", (char **) 0))
+      {
+	kills -= 1;
+	if (kills < 0)
+	  kills = 0;
+      }
+      else if (!get_double(arg, &kills, (char **) 0))
+      {
+	respond("Try: control player %d kills [%f]+-", 0);
+	return 0;
+      }
+
+      victim->p_kills = kills;
+      sprintf(buf, "%s (%2s) has been given %f kills by %s (%2s).",
+	      victim->p_name, twoletters(victim),
+	      kills, me->p_name, twoletters(me));
+      if (!god_silent)
+	pmessage(buf, 0, MALL, MCONTROL);
+      return 1;
+    }
+
+   case HOSETOK:
+    victim->p_shield = 0;
+    victim->p_damage = victim->p_ship.s_maxdamage / 2;
+    sprintf(buf, "%s (%2s) has been hosed by %s (%2s).",
+	    victim->p_name, twoletters(victim),
+	    me->p_name, twoletters(me));
+    if (!god_silent)
+      pmessage(buf, 0, MALL, MCONTROL);
+    return 1;
+
+   case MOVETOK:
+    {
+      int x, y;
+      char *s;
+      if (!(get_int(arg, &x, &s) && get_int(s, &y, (char **) 0)))
+      {
+	respond("Try: control player %d move %d %d", 0);
+	return 0;
+      }
+
+      if (x <= 0 || y <= 0 || x >= 200000 || y >= 200000)
+      {
+	respond("You want to move him where?", 0);
+	return 0;
+      }
+      victim->p_x = x;
+      victim->p_y = y;
+      sprintf(buf, "%s (%2s) has been moved to %d %d by %s (%2s).",
+	      victim->p_name, twoletters(victim),
+	      x, y,
+	      me->p_name, twoletters(me));
+      if (!god_silent)
+	pmessage(buf, 0, MALL, MCONTROL);
+      return 1;
+    }
+
+   case HELPTOK:		/* fall through */
+   default:
+#if 1
+    respond_with_help_string(available_cmds);
+#else
+    respond("player controls: die, eject, armies [%d], plasma [%d],", 0);
+    respond("player controls: missile [%d], team [team], ship [ship],", 0);
+    respond("player controls: rank [%d]+-, royal [%d]+-, kills [%d]+-", 0);
+    respond("player controls: hose, move %d %d", 0);
+#endif
+    return 0;
+  }
+}
+
+static int
+parse_control(str)
+  char *str;
+{
+  char buf[120];
+  struct player *victim;
+  int i;
+  int godliness = me->p_stats.st_royal - GODLIKE + 1;
+
+  static struct control_cmd available_cmds[] = {
+    {"help", HELPTOK, 0},
+    {"freeslot", FREESLOTTOK, "freeslot %p"},
+    {"player", PLAYERTOK, "player ..."},
+    {"robot", ROBOTTOK, "robot [args]"},
+    {"quiet", QUIETTOK, "quiet"},
+    {"nukegame", NUKEGAMETOK, "nukegame"},
+    {"restart", RESTARTTOK, "restart"},
+    {"galaxy", NEWGALAXY, "galaxy"},
+    {"shiptimer", SHIPTIMERTOK, "shiptimer (teamstr|shipstr)*"},
+    {"allow", ALLOWTOK, "allow [teams]"},
+    {0}
+  };
+  char *nexttoken;
+
+  if (godliness <= 0)
+  {
+#if 0
+    respond("Those commands are only available to server gods");
+#endif
+    return 0;			/* "fail" silently.  Don't advertise divine
+				 * powers to peasants. */
+  }
+
+
+  switch (next_token(str, available_cmds, &nexttoken))
+  {
+   case FREESLOTTOK:
+    {
+      int slot = get_slotnum(nexttoken, (char **) 0);
+      if (slot < 0)
+      {
+	respond("\"control freeslot\" requires slot number.", 0);
+	return 1;
+      }
+      victim = &players[slot];
+      if (victim->p_ntspid)
+	kill(victim->p_ntspid, SIGHUP);
+
+      victim->p_status = PFREE;
+      victim->p_ntspid = 0;
+
+      if (!god_silent)
+      {
+	sprintf(buf, "Player slot %s has been freed by %s (%2s).",
+		nexttoken, me->p_name, twoletters(me));
+	pmessage(buf, 0, MALL, MCONTROL);
+      }
+    }
+    return 1;
+
+   case ALLOWTOK:
+    {
+      int newlock = 0;
+      int team;
+      char *s;
+      if (0 == *nexttoken)
+      {
+	newlock = ALLTEAM;
+      }
+      else
+      {
+	for (s = nexttoken; get_teamid(s, &team, &s);)
+	{
+	  newlock |= idx_to_mask(team);
+	}
+	if (*s)
+	{
+	  respond("Usage: control allow [fed] [rom] [kli] [ori]", 0);
+	  return 1;
+	}
+      }
+
+      status2->nontteamlock = newlock;
+      strcpy(buf, "Allowed teams now set to:");
+      if (status2->nontteamlock == ALLTEAM)
+      {
+	strcat(buf, " <all teams>");
+      }
+      else
+      {
+	if (status2->nontteamlock & FED)
+	  strcat(buf, " fed");
+	if (status2->nontteamlock & ROM)
+	  strcat(buf, " rom");
+	if (status2->nontteamlock & KLI)
+	  strcat(buf, " kli");
+	if (status2->nontteamlock & ORI)
+	  strcat(buf, " ori");
+      }
+      respond(buf, 0);
+    }
+    return 1;
+   case PLAYERTOK:
+    return parse_control_player(nexttoken);
+
+   case ROBOTTOK:
+    {
+      int pid;
+      char *s;
+
+      pid = fork();
+      if (pid == 0)
+      {
+	char *argv[40];
+	argv[0] = build_path(ROBOT);
+
+	s = nexttoken;
+	for (i = 1; 1; i++)
+	{
+	  int size = 80;
+	  argv[i] = malloc(size);
+	  if (!get_one_token(s, argv[i], size, &s))
+	    break;
+	  realloc(argv[i], strlen(argv[i]) + 1);
+	}
+	free(argv[i]);
+	argv[i] = 0;
+
+	execvp(argv[0], argv);
+	fprintf(stderr, "Ack! Unable to exec %s\n", argv[0]);
+	exit(1);
+      }
+      else if (pid < 0)
+      {
+	respond("Unable to fork robot", 0);
+      }
+      else
+      {
+	sprintf(buf, "Robot forked (pid %d) with arguments %s",
+		pid, nexttoken);
+	respond(buf, 1);
+      }
+    }
+    return 1;
+
+   case QUIETTOK:
+    if (godliness < 2)
+    {
+      respond("No sneaking allowed", 0);
+      return 1;
+    }
+    sprintf(buf, "Switching to %s mode.", god_silent ? "loud" : "quiet");
+    respond(buf, 0);
+    god_silent = !god_silent;
+    return 1;
+   case NUKEGAMETOK:
+    warning("Nuking game.  Have a nice day.");
+    if (!god_silent)
+    {
+      sprintf(buf, "The game has been nuked by %s (%2s).",
+	      me->p_name, twoletters(me));
+      pmessage(buf, 0, MALL, MCONTROL);
+    }
+    kill(status->nukegame, 15);
+    return 1;
+   case RESTARTTOK:
+    warning("Attempting daemon restart.");
+    startdaemon(
+#ifdef LEAGUE_SUPPORT
+		status2->league
+#else
+		0
+#endif
+		,1);
+    return 1;
+   case NEWGALAXY:
+    explode_everyone(KPROVIDENCE, 0);
+    if (!god_silent)
+    {
+      sprintf(buf, "The galaxy has been reset by %s (%2s).",
+	      me->p_name, twoletters(me));
+      pmessage(buf, 0, MALL, MCONTROL);
+    }
+    status2->newgalaxy = 1;
+    warning("Creating new galaxy");
+    return 1;
+   case SHIPTIMERTOK:
+    {
+      int teammask = 0;
+      int shipmask = 0;
+      int i, j;
+      char *s = nexttoken;
+      while (1)
+      {
+	if (get_shipid(s, &j, &s))
+	  shipmask |= 1 << j;
+	else if (get_teamid(s, &j, &s))
+	  teammask |= idx_to_mask(j);
+	else if (*s)
+	{
+	  respond("Usage:", 0);
+	  respond("control shiptimers (fed|rom|kli|ori)* (sc|dd|ca|bb|as|sb|at|js|fr|wb)*", 0);
+	  respond("  resets the ship timers.", 0);
+	  return 0;
+	}
+	else
+	  break;
+      }
+      for (i = 0; i < NUMTEAM; i++)
+      {
+	int teammask = idx_to_mask(i);
+	if (teammask && !(teammask & (1 << i)))
+	  continue;
+
+	for (j = 0; j < NUM_TYPES; j++)
+	{
+	  if (shipmask && !(shipmask & (1 << j)))
+	    continue;
+
+	  if (teams[teammask].s_turns[j])
+	  {
+	    sprintf(buf, "%s %s reset", teams[teammask].name, shipvals[j].s_name);
+	    respond(buf, 0);
+	    teams[teammask].s_turns[j] = 0;
+	  }
+	}
+      }
+    }
+    return 1;
+   case HELPTOK:		/* fall through */
+   default:
+#if 1
+    respond_with_help_string(available_cmds);
+#else
+    respond("Available controls: player, quiet, nukegame, freeslot,", 0);
+    respond("  galaxy, restart, shiptimer", 0);
+#endif
+    return 0;
+  }
+}
+#endif
+
+/*
+ */
+
+static int
+parse_info(cmd)
+  char *cmd;
+{
+  char buf[120];
+  char *nexttoken;
+
+  static struct control_cmd available_cmds[] = {
+    {"help", HELPTOK, 0},
+    {"shiptimer", SHIPTIMERTOK, "shiptimer (teamstr|shipstr)*"},
+    {0}
+  };
+
+  switch (next_token(cmd, available_cmds, &nexttoken))
+  {
+   case SHIPTIMERTOK:
+    {
+      int race = 0;
+      int i, j;
+      int anydead = 0;
+#if 0
+      if (me->p_stats.st_royal < 2)
+	race = me->p_team;
+#endif
+      for (i = 0; i < NUMTEAM; i++)
+      {
+	int teammask = idx_to_mask(i);
+	if (race && !(race & (1 << i)))
+	  continue;
+	for (j = 0; j < NUM_TYPES; j++)
+	{
+	  if (teams[teammask].s_turns[j])
+	  {
+	    sprintf(buf, "%s %s: %d minutes", teams[teammask].name,
+		    shipvals[j].s_name, teams[teammask].s_turns[j]);
+	    anydead = 1;
+	    respond(buf, 0);
+	  }
+	}
+      }
+      if (!anydead)
+	respond("All ships are available", 0);
+    }
+    return 1;
+   case HELPTOK:
+   default:
+    respond("Available subcommands: shiptimer", 0);
+    return 1;
+  }
+}
+
+#define crypt(a, b) (a)
+
+static int
+parse_player(cmd)
+  char *cmd;
+{
+  static int passver = 0;
+  char buf[80];
+#if 0
+  static char newpass[16];
+#endif
+  char *nexttoken;
+
+  static struct control_cmd available_cmds[] = {
+    {"help", HELPTOK, 0},
+    {"password", PASSWDTOK, "password %s"},
+    {"passwd", PASSWDTOK, 0},
+    {"ratings", RATINGSTOK, "ratings"},
+    {"rank", RANKTOK, "rank"},
+    {0}
+  };
+
+  if (0 == *cmd)
+  {
+    if (passver)
+      respond("Password change cancelled.", 0);
+
+    passver = 0;
+    return 1;
+  }
+
+  switch (next_token(cmd, available_cmds, &nexttoken))
+  {
+   case PASSWDTOK:
+    {
+      static char newpass[16];
+      if (me->p_pos < 0)
+      {				/* guest login */
+	respond("You don't have a password!", 0);
+      }
+      else if (*nexttoken == 0)
+      {
+	respond("\"player password\" requires new password as argument.", 0);
+	respond("  example: \"player password lh4ern\"", 0);
+      }
+      else if (!passver)
+      {
+	strcpy(newpass, crypt(nexttoken, me->p_name));
+	respond("Repeat \"player password\" command to verify new password.", 0);
+	respond(" or send \"player\" (no arguments) to cancel.", 0);
+	passver = 1;
+      }
+      else
+      {
+	char *paths;
+	int fd;
+	if (!strcmp(newpass, crypt(nexttoken, me->p_name)))
+	{
+
+	  /* perhaps it'd be better to put this part in */
+	  /* a different place */
+	  paths = build_path(PLAYERFILE);
+	  fd = open(paths, O_WRONLY, 0644);
+	  if (fd >= 0)
+	  {
+	    lseek(fd, 16 + me->p_pos * sizeof(struct statentry), 0);
+	    write(fd, newpass, 16);
+	    close(fd);
+	    respond("Password changed.", 0);
+	  }
+	  else
+	  {
+	    respond("open() of playerfile failed, password not changed.", 0);
+	  }
+	}
+	else
+	  respond("Passwords did not match, password unchanged.", 0);
+	passver = 0;
+      }
+    }
+    return 1;
+   case RATINGSTOK:		/* print your ratings */
+    {
+      struct rating r;
+      compute_ratings(me, &r);
+
+      sprintf(buf, "Bomb:%5.2f   Plnts:%5.2f   Rsrcs:%5.2f    Dshs:%5.2f   Offns:%5.2f", r.bombrat, r.planetrat, r.resrat, r.dooshrat, r.offrat);
+      respond(buf, 0);
+
+      sprintf(buf, "  JS:%5.2f      SB:%5.2f      WB:%5.2f   Ratio:%5.2f", r.jsrat, r.sbrat, r.wbrat, r.ratio);
+      respond(buf, 0);
+
+      sprintf(buf, "Overall Ratings:  Battle:%5.2f   Strat:%5.2f   Spec. Ship:%5.2f", r.battle, r.strategy, r.special);
+      respond(buf, 0);
+    }
+    return 1;
+
+   case RANKTOK:		/* print the requirements for the next rank */
+    {
+      int rank;
+      rank = me->p_stats.st_rank;
+      strcpy(buf, "Your current rank is ");
+      strcat(buf, ranks[rank].name);
+      respond(buf, 0);
+      if (rank == NUMRANKS - 1)
+	return 1;
+
+      sprintf(buf, "To make the next rank (%s) you need:",
+	      ranks[rank + 1].name);
+      respond(buf, 0);
+
+      sprintf(buf, "    Genocides: %d   DI: %.2f   Battle: %.2f",
+	      ranks[rank + 1].genocides, ranks[rank + 1].di,
+	      ranks[rank + 1].battle);
+      respond(buf, 0);
+
+      sprintf(buf, "    Strategy: %.2f   Spec. Ships: %.2f",
+	      ranks[rank + 1].strategy, ranks[rank + 1].specship);
+      respond(buf, 0);
+    }
+    return 1;
+
+   case HELPTOK:
+   default:
+    respond_with_help_string(available_cmds);
+    return 1;
+  }
+}
+
+/*
+ */
+
+#ifdef LEAGUE_SUPPORT
+
+static void
+umpire_speak(msg)
+  char *msg;
+{
+  pmessage(msg, -1, MALL, UMPIRE);
+}
+
+static void
+talk_about_team(team, type)
+  struct league_team *team;
+  char *type;
+{
+  char buf[120];
+  struct player *captain;
+
+  if (team->captain >= 0)
+    captain = &players[team->captain];
+  else
+    captain = 0;
+
+  sprintf(buf, "The %s team is named `%s'.", type, team->name);
+  respond(buf, 0);
+
+  if (captain)
+    sprintf(buf, "  %s (%s) is their captain.", captain->p_name,
+	    twoletters(captain));
+  else
+    strcpy(buf, "  They have not chosen a captain yet.");
+  respond(buf, 0);
+
+  if (team->index >= 0)
+  {
+    sprintf(buf, "  They have chosen the %s",
+	    teams[idx_to_mask(team->index)].name);
+  }
+  else
+  {
+    strcpy(buf, "  They have not chosen an empire yet.");
+  }
+  respond(buf, 0);
+}
+
+static int
+team_really_ready(team)
+  struct league_team *team;
+{
+  if (team->index < 0)
+  {
+    respond("You haven't chosen an empire", 1);
+    return 0;
+  }
+  if (team->name[0] == 0)
+  {
+    respond("You haven't chosen a name", 1);
+    return 0;
+  }
+  return 1;
+}
+
+void
+trydefect(victim, dest, destname, from, fromname, actor)
+  struct player *victim;
+  enum HomeAway dest;
+  char *destname;
+  enum HomeAway from;
+  char *fromname;
+  struct player *actor;
+{
+  char buf[120];
+  struct league_team *fromteam =
+  (from == AWAY) ? &status2->away : &status2->home;
+
+  if (victim->p_status == PFREE)
+  {
+    respond("Uh, he's not in the game.", 1);
+    return;
+  }
+
+  if (victim->p_homeaway == dest)
+  {
+    sprintf(buf, "%s already belong to the %s team",
+	    actor == victim ? "You" : "They", destname);
+    respond(buf, 1);
+    return;
+  }
+  if (actor->p_homeaway != from)
+  {
+    sprintf(buf, "You don't belong to the %s team.  You can't kick him off.",
+	    fromname);
+    respond(buf, 1);
+    return;
+  }
+  if (fromteam->captain == actor->p_no)
+  {
+    if (victim == actor)
+    {
+      if (status2->league > 1 || status2->home.ready || status2->away.ready)
+      {
+	respond("You can't defect in the middle of the game.  You're the captain!", 1);
+	return;
+      }
+      sprintf(buf, "%s (%s), the captain of the %s team, has defected!",
+	      victim->p_name, twoletters(victim), fromname);
+      umpire_speak(buf);
+    }
+    else
+    {
+      sprintf(buf, "%s (%s) has kicked %s (%s) off his team.",
+	      actor->p_name, twoletters(actor),
+	      victim->p_name, twoletters(victim));
+      umpire_speak(buf);
+    }
+  }
+  else
+  {
+    if (victim == actor)
+    {
+      if (status2->league > 1 || status2->home.ready || status2->away.ready)
+      {
+	respond("Only the captain can kick you off now.", 1);
+	return;
+      }
+      sprintf(buf, "%s (%s) has defected to the %s team!",
+	      victim->p_name, twoletters(victim),
+	      destname);
+      umpire_speak(buf);
+    }
+    else
+    {
+      respond("Only the captain can kick other people off the team.", 1);
+      return;
+    }
+  }
+  victim->p_homeaway = dest;
+  victim->p_status = PEXPLODE;
+  victim->p_whydead = KPROVIDENCE;
+  victim->p_explode = 1;
+}
+
+static int
+parse_league(subcommand)
+  char *subcommand;
+{
+  struct league_team *myteam, *otherteam;
+  char *teamtype;
+  char buf[120];
+  int i;
+  int iscaptain;
+  char *nexttoken;
+  static char captain_only[] = "That command is reserved for team captains.";
+  static struct control_cmd available_cmds[] = {
+    {"help", HELPTOK, 0},
+    {"captain", CAPTAINTOK, "captain [%d]"},
+    {"time", TIMETOK, "time [%d %d]"},
+    {"pass", PASSTOK, "pass"},
+    {"start", STARTTOK, "start [%d]"},
+    {"restart", RESTARTTOK, "restart [%d]"},
+#if 0
+    {"timeout", TIMEOUTTOK, "timeout [%d]"},
+#endif
+    {"teamname", TEAMNAMETOK, "teamname %s"},
+    {"information", INFOTOK, "information"},
+#if 0
+    /* damn, these should be initialized from the current race list */
+    {"federation", FEDTOK, "fed"},
+    {"romulan", ROMTOK, "rom"},
+    {"klingon", KLITOK, "kli"},
+    {"orion", ORITOK, "ori"},
+#endif
+
+    {"away", AWAYTOK, "away [%d]"},
+    {"home", HOMETOK, "home [%d]"},
+    {"newgalaxy", NEWGALAXY, "newgalaxy [%d]"},
+    {"pause", PAUSETOK, "pause"},
+    {"continue", CONTINUETOK, "continue"},
+    {"maxplayer", MAXPLAYERTOK, "maxplayer [%d]"},
+    {"freeslot", FREESLOTTOK, "freeslot %d"},
+    {0}
+  };
+
+  switch (me->p_homeaway)
+  {
+   case HOME:
+    myteam = &status2->home;
+    otherteam = &status2->away;
+    teamtype = "home";
+    break;
+   case AWAY:
+    myteam = &status2->away;
+    otherteam = &status2->home;
+    teamtype = "away";
+    break;
+   default:
+    respond("WHOA! internal error.  You aren't on a team!", 0);
+    respond("I'm afraid I'm going to have to kill you", 0);
+    me->p_status = PEXPLODE;
+    me->p_explode = 1;
+    return 0;
+  }
+
+  iscaptain = (myteam->captain == me->p_no);
+
+  /********************/
+
+  if (get_teamid(subcommand, &i, (char **) 0))
+  {
+    if (!iscaptain)
+    {
+      respond(captain_only, 1);
+      return 1;
+    }
+    if (status2->league != 1)
+    {
+      respond("The game has started.  You can't change your mind now.", 1);
+      return 1;
+    }
+    if ((myteam->ready || otherteam->ready) && myteam->index >= 0)
+    {
+      respond("One of the teams is ready.  You can't change your mind now.", 1);
+      return 1;
+    }
+    if (otherteam->index >= 0)
+    {
+      if (i == otherteam->index)
+      {
+	respond("The other team has already chosen that empire", 1);
+	return 1;
+      }
+    }
+    else
+    {
+      if (me->p_homeaway == HOME)
+      {
+	if (!status2->awaypassed)
+	{
+	  respond("Away team gets first choice of empire.", 1);
+	  return 1;
+	}
+      }
+      else /* away */ if (myteam->index >= 0 && 0 == status2->awaypassed)
+      {
+	respond("Give the other team a chance to choose a side, will ya?", 1);
+	return 1;
+      }
+      else if (status2->awaypassed == 1)
+      {
+	respond("You passed the choice of empire.  You have to wait for their choice.", 1);
+	return 1;
+      }
+    }
+
+    if (i == myteam->index)
+    {
+      respond("That already IS your empire.", 1);
+      return 1;
+    }
+    if (i < 0)
+    {
+#if 0
+      sprintf(buf, "The %s team no longer wishes the %s for their empire.",
+	      teamtype, teams[idx_to_mask(myteam->index)].name);
+#else
+      respond("You can't change your mind without a reset.  Ask for one", 1);
+      return 1;
+#endif
+    }
+    else
+      sprintf(buf, "The %s team has chosen the %s for their empire.",
+	      teamtype, teams[idx_to_mask(i)].name);
+    umpire_speak(buf);
+
+    myteam->index = i;
+
+    return 1;
+  }
+  else
+    switch (next_token(subcommand, available_cmds, &nexttoken))
+    {
+     default:
+     case HELPTOK:		/********************/
+      if (iscaptain)
+      {
+	respond_with_help_string(available_cmds);
+      }
+      else
+      {
+	respond("Available commands: captain [ %d ], time, information, maxplayer.", 0);
+      }
+      return 1;
+
+     case CAPTAINTOK:		/********************/
+      {
+	int j;
+	i = !get_int(nexttoken, &j, (char **) 0) || j;
+      }
+      if (i)
+      {
+	if (myteam->captain < 0 ||
+	    players[myteam->captain].p_status != PALIVE ||
+	    players[myteam->captain].p_team != me->p_team)
+	{
+	  if (myteam->captain >= 0)
+	  {
+	    /*
+	     * safety valve in case the person is ghostbusted or on another
+	     * team
+	     */
+	    sprintf(buf, "%s has been overthrown as captain of the %s team",
+		    players[myteam->captain].p_name, teamtype);
+	    umpire_speak(buf);
+	  }
+	  respond("OK.  *POOF* you're a captain!", 1);
+	  sprintf(buf, "%s (%s) is now fearless leader of the %s team!",
+		  me->p_name, twoletters(me), teamtype);
+	  umpire_speak(buf);
+	  myteam->captain = me->p_no;
+	}
+	else if (iscaptain)
+	{
+	  respond("Listen, silly.  You already are captain.  No point in rubbing it in", 1);
+	}
+	else
+	{
+	  /* if myteam->captain were <0, we wouldn't get here */
+	  struct player *capn = &players[myteam->captain];
+	  sprintf(buf, "Sorry, %s (%s) is already captain of your team.",
+		  capn->p_name, twoletters(capn));
+	  respond(buf, 1);
+	}
+      }
+      else
+      {
+	if (iscaptain)
+	{
+	  respond("Wimp.  We didn't want you for captain anyway.", 1);
+	  sprintf(buf, "%s (%s) has chickened out.",
+		  me->p_name, twoletters(me));
+	  umpire_speak(buf);
+	  sprintf(buf, "Who now will lead the %s team?", teamtype);
+	  umpire_speak(buf);
+	  myteam->captain = -1;
+	}
+	else
+	{
+	  respond("You can't quit being a captain.  You weren't one in the first place.", 1);
+	}
+      }
+      return 1;
+
+     case TIMETOK:		/********************/
+      if (0 == *nexttoken)
+      {
+	switch (status2->league)
+	{
+	 case 2:
+	  sprintf(buf, "%d seconds left in pre-tourney warm-up.",
+		  status2->leagueticksleft / SECONDS(1));
+	  break;
+	 case 3:
+	  sprintf(buf, "%d minutes left in regulation play.",
+		  status2->leagueticksleft / MINUTES(1));
+	  break;
+	 case 4:
+	  sprintf(buf, "%d minutes left in overtime.",
+		  status2->leagueticksleft / MINUTES(1));
+	  break;
+	 default:
+	  sprintf(buf, "game is configured for %d minutes (%d overtime).",
+	      configvals->regulation_minutes, configvals->overtime_minutes);
+	  break;
+	}
+	respond(buf, 0);
+	return 1;
+      }
+      else if (!iscaptain)
+      {
+	respond(captain_only, 1);
+	return 1;
+      }
+      else if (status2->league != 1)
+      {
+	respond("You can only adjust the time parameters during the configuration phase", 1);
+	return 1;
+      }
+      else if (otherteam->ready)
+      {
+	respond("The other team is ready to start.  You can't change the game params NOW.", 1);
+	return 1;
+      }
+      else if (!get_int(nexttoken, &myteam->desired.regulation, &nexttoken)
+	     || !get_int(nexttoken, &myteam->desired.overtime, (char **) 0))
+      {
+	respond("Usage: time [ %d %d ]", 1);
+	return 1;
+      }
+
+      if (status2->home.desired.regulation == status2->away.desired.regulation
+	&& status2->home.desired.overtime == status2->away.desired.overtime)
+      {
+	configvals->regulation_minutes = status2->home.desired.regulation;
+	configvals->overtime_minutes = status2->home.desired.overtime;
+	sprintf(buf, "The captains have agreed to a %d minute game (%d overtime).", configvals->regulation_minutes, configvals->overtime_minutes);
+	umpire_speak(buf);
+      }
+      else
+      {
+	sprintf(buf, "The %s team wishes a game of %d minutes (%d overtime)",
+	    teamtype, myteam->desired.regulation, myteam->desired.overtime);
+	umpire_speak(buf);
+      }
+
+      return 1;
+
+     case PASSTOK:		/********************/
+      if (status2->league != 1)
+      {
+	respond("The time for that is long past.", 1);
+	return 1;
+      }
+      if (!iscaptain)
+      {
+	respond(captain_only, 1);
+	return 1;
+      }
+      if (me->p_homeaway == AWAY && status2->awaypassed)
+      {
+	respond("You already passed the choice of empire.", 1);
+	return 1;
+      }
+      else if (me->p_homeaway == HOME && status2->awaypassed == 0)
+      {
+	respond("You can't possibly pass the choice of empire.  You don't HAVE it!", 1);
+	return 1;
+      }
+      else if (status2->awaypassed > 1)
+      {
+	respond("You both passed already, so get on with it.  (indecisive wishy-washy cretins)", 1);
+	return 1;
+      }
+      status2->awaypassed++;
+
+      sprintf(buf, "The %s team has passed the choice of empire", teamtype);
+      umpire_speak(buf);
+
+      if (status2->awaypassed > 1)
+      {
+	umpire_speak("Computer choosing randomly for both teams");
+	if (status2->home.index < 0)
+	{
+	  status2->home.index = lrand48() % ((status2->away.index < 0) ? 4 : 3);
+	  if (status2->away.index >= 0 &&
+	      status2->home.index >= status2->away.index)
+	    status2->home.index++;
+	}
+	if (status2->away.index < 0)
+	{
+	  status2->away.index = lrand48() % 3;
+	  if (status2->away.index >= status2->home.index)
+	    status2->away.index++;
+	}
+      }
+      return 1;
+
+     case STARTTOK:		/********************/
+      if (status2->league != 1)
+      {
+	respond("The game has already started.", 1);
+	return 1;
+      }
+      if (!iscaptain)
+      {
+	respond(captain_only, 1);
+	return 1;
+      }
+      if (get_int(nexttoken, &myteam->ready, (char **) 0) && !myteam->ready)
+      {
+	sprintf(buf, "The %s team is not ready.", teamtype);
+	umpire_speak(buf);
+	return 0;
+      }
+      myteam->ready = 1;
+      if (!team_really_ready(myteam))
+      {
+	respond("Your team is not really ready.  You need a name and an empire.", 0);
+	myteam->ready = 0;
+	return 0;
+      }
+
+      if (otherteam->ready && !team_really_ready(otherteam))
+      {
+	otherteam->ready = 0;
+	sprintf(buf, "The %s team was ready but the other wasn't.", teamtype);
+      }
+      else
+      {
+	sprintf(buf, "The %s team is ready to start now.", teamtype);
+      }
+      umpire_speak(buf);
+
+
+      if (otherteam->ready)
+      {
+	/* shit! we're good to go! */
+
+	umpire_speak("Both sides are ready.  Let the carnage begin!");
+	umpire_speak("Everybody dies.  T-mode starts in 1 minute.");
+	status2->league = 2;
+	status2->leagueticksleft = 60 * TICKSPERSEC;
+
+	/* version */
+
+	explode_everyone(KTOURNSTART, 0);
+      }
+      return 1;
+
+     case RESTARTTOK:		/********************/
+      if (status2->league != 1)
+      {
+	respond("The game has started.  You can't change your mind now.", 1);
+	return 1;
+      }
+
+      if (!iscaptain)
+      {
+	respond(captain_only, 1);
+	return 1;
+      }
+
+      myteam->desired.restart = !get_int(nexttoken, &i, (char **) 0) || i;
+
+      sprintf(buf, myteam->desired.restart ?
+	      "%s (%s) would like to restart team selection." :
+	      "%s (%s) is satisfied with the teams.",
+	      me->p_name, twoletters(me));
+      umpire_speak(buf);
+
+      if (status2->home.desired.restart && status2->away.desired.restart)
+      {
+	umpire_speak("Both captains have agreed to restart team selection.");
+
+	status2->awaypassed = 0;
+	status2->home.index = status2->away.index = -1;
+
+	status2->home.ready = status2->away.ready = 0;
+
+	status2->home.desired.restart = status2->away.desired.restart = 0;
+      }
+
+      return 1;
+
+     case TIMEOUTTOK:
+      respond("NYI", 0);
+      return 1;
+
+     case TEAMNAMETOK:		/********************/
+      if (status2->league != 1)
+      {
+	respond("The game has started.  You can't change your mind now.", 1);
+	return 1;
+      }
+      if (!iscaptain)
+      {
+	respond(captain_only, 1);
+	return 1;
+      }
+      if (0 == *nexttoken)
+      {
+	respond("What do you want to call your team?\n", 1);
+	return 1;
+      }
+      strncpy(myteam->name, nexttoken, sizeof(myteam->name));
+      myteam->name[sizeof(myteam->name) - 1] = 0;
+
+      sprintf(buf, "Henceforth let the %s team be known as `%s'!",
+	      teamtype, myteam->name);
+      umpire_speak(buf);
+
+      return 1;
+
+     case INFOTOK:		/********************/
+      sprintf(buf, "The game will last for %d minutes (%d overtime)",
+	      configvals->regulation_minutes,
+	      configvals->overtime_minutes);
+      respond(buf, 0);
+      sprintf(buf, "Teams are limited to %d players on the field at once",
+	      configvals->playersperteam);
+      respond(buf, 0);
+      sprintf(buf, "You are on the %s team.", teamtype);
+      respond(buf, 0);
+      talk_about_team(&status2->home, "home");
+      talk_about_team(&status2->away, "away");
+
+      if (status2->awaypassed > 1)
+	umpire_speak("Both teams passed empire choice.  Computer assigned.");
+      else if (status2->awaypassed)
+	umpire_speak("Away has passed empire choice to the Home team");
+      else
+	umpire_speak("Away chooses empire first");
+
+      return 1;
+
+
+     case AWAYTOK:		/********************/
+      {
+	struct player *victim;
+	if (!*nexttoken)
+	  victim = me;
+	else
+	{
+	  int idx = get_slotnum(nexttoken, (char **) 0);
+	  if (idx < 0)
+	  {
+	    respond("`league away' requires a valid slot number", 0);
+	    return 1;
+	  }
+	  else
+	  {
+	    victim = &players[idx];
+	  }
+	}
+	trydefect(victim, AWAY, "away", HOME, "home", me);
+      }
+      return 1;
+
+     case HOMETOK:		/********************/
+      {
+	struct player *victim;
+	if (!*nexttoken)
+	  victim = me;
+	else
+	{
+	  int idx = get_slotnum(nexttoken, (char **) 0);
+	  if (idx < 0)
+	  {
+	    respond("`league away' requires a valid slot number", 0);
+	    return 1;
+	  }
+	  else
+	  {
+	    victim = &players[idx];
+	  }
+	}
+	trydefect(victim, HOME, "home", AWAY, "away", me);
+      }
+
+      return 1;
+     case NEWGALAXY:		/********************/
+
+      if (status2->league != 1)
+      {
+	respond("The game has started.  You can't change your mind now.", 1);
+	return 1;
+      }
+
+      if (myteam->ready || otherteam->ready)
+      {
+	respond("You can't reset the galaxy now.  We're almost ready!", 1);
+	return 1;
+      }
+      if (!iscaptain)
+      {
+	respond(captain_only, 1);
+	return 1;
+      }
+
+      {
+	int j;
+	myteam->desired.galaxyreset =
+	  !get_int(nexttoken, &j, (char **) 0) || j;
+      }
+
+      if (myteam->desired.galaxyreset)
+      {
+	sprintf(buf, "%s (%s) is dissatisfied with the galaxy",
+		me->p_name, twoletters(me));
+      }
+      else
+      {
+	sprintf(buf, "%s (%s) thinks the galaxy is just fine, thank you.",
+		me->p_name, twoletters(me));
+      }
+      umpire_speak(buf);
+
+      if (status2->home.desired.galaxyreset &&
+	  status2->away.desired.galaxyreset)
+      {
+	umpire_speak("Both captains have agreed that the galaxy sucks.");
+	status2->newgalaxy = 1;
+	warning("Creating new galaxy");
+
+	status2->home.desired.galaxyreset = status2->away.desired.galaxyreset = 0;
+
+	status2->awaypassed = 0;
+	status2->home.index = status2->away.index = -1;
+	status2->home.ready = status2->away.ready = 0;
+	status2->home.desired.restart = status2->away.desired.restart = 0;
+      }
+
+      return 1;
+
+     case PAUSETOK:		/********************/
+      if (!iscaptain)
+      {
+	respond(captain_only, 1);
+	return 1;
+      }
+
+      myteam->desirepause = 1;
+
+      if (status2->home.desirepause && status2->away.desirepause)
+      {
+	/* well, it's unanimous! */
+	status2->paused = SECONDS(10);
+	umpire_speak("The game has been paused");
+      }
+      else
+      {
+	sprintf(buf, "The %s team wishes to PAUSE the game.", teamtype);
+	umpire_speak(buf);
+      }
+
+      status2->pausemsgfuse = 0;
+
+      return 1;
+
+     case CONTINUETOK:		/********************/
+      if (!iscaptain)
+      {
+	respond(captain_only, 1);
+	return 0;
+      }
+
+      myteam->desirepause = 0;
+
+      sprintf(buf, "The %s team wishes to CONTINUE.", teamtype);
+      umpire_speak(buf);
+
+      status2->pausemsgfuse = 0;
+
+      return 1;
+
+     case MAXPLAYERTOK:
+      {
+	int mp;
+	if (!get_int(nexttoken, &mp, (char **) 0))
+	{
+	  sprintf(buf, "The game is currently configured for a maximum of %d players on each", configvals->playersperteam);
+	  respond(buf, 0);
+	  respond("team.", 0);
+	  if (configvals->playersperteam != myteam->desired.maxplayers)
+	  {
+	    sprintf(buf, "You, however, want it to be %d.",
+		    myteam->desired.maxplayers);
+	    respond(buf, 0);
+	  }
+	}
+	else
+	{
+	  if (!iscaptain)
+	  {
+	    respond(captain_only, 1);
+	    return 1;
+	  }
+
+	  if (mp < 1)
+	  {
+	    respond("That's a stupid value for players-per-team.", 1);
+	    return 1;
+	  }
+
+	  myteam->desired.maxplayers = mp;
+	  if (status2->away.desired.maxplayers
+	      == status2->home.desired.maxplayers)
+	  {
+	    configvals->playersperteam = mp;
+	    sprintf(buf, "Captains have agreed to limit players per team to %d", mp);
+	    umpire_speak(buf);
+	  }
+	  else
+	  {
+	    sprintf(buf, "The %s team would like to limit the number of ",
+		    teamtype);
+	    umpire_speak(buf);
+	    sprintf(buf, "players per team to %d", mp);
+	    umpire_speak(buf);
+	  }
+	}
+      }
+      return 1;
+     case FREESLOTTOK:
+      {
+	int slotnum;
+	struct player *victim;
+	slotnum = get_slotnum(nexttoken, (char **) 0);
+	if (slotnum < 0)
+	{
+	  respond("freeslot requires slot number", 1);
+	  return 0;
+	}
+
+	victim = &players[slotnum];
+	if (victim->p_ntspid)
+	  kill(victim->p_ntspid, SIGHUP);
+
+	victim->p_status = PFREE;
+	victim->p_ntspid = 0;
+
+	sprintf(buf, "Player slot %s has been freed by %s (%2s).",
+		nexttoken, me->p_name, twoletters(me));
+	umpire_speak(buf);
+      }
+      return 1;
+    }
+}
+#endif
+/*
+ */
+
+
+
+/*
+ * Parse command messages.
+ */
+
+int
+parse_command_mess(str, who)
+  char *str;
+  unsigned char who;
+{
+  char buf[120];
+  char *nexttoken;
+  int godlike = me->p_stats.st_royal - GODLIKE + 1;
+
+  static struct control_cmd available_cmds[] = {
+    {"help", HELPTOK, 0},
+    {"control", CONTROLTOK, "control ..."},
+    {"league", LEAGUETOK, "league ..."},
+    {"version", VERSIONTOK, "version"},
+    {"player", PLAYERTOK, "player ..."},
+    {"queue", QUEUETOK, "queue"}, {"tq", QUEUETOK, 0},
+    {"information", INFOTOK, "information ..."},
+    {"observe", OBSERVETOK, "observe"},
+    {"parameters", PARAMTOK, "parameters"}, {"params", PARAMTOK, 0},
+    {"cluecheck", CLUECHECKTOK, "cluecheck [%s]"},
+    {0}
+  };
+
+  if (godlike < 0)
+    godlike = 0;
+
+
+
+#if 0
+  /* so we won't damage the original string */
+  strcpy(buf, str);
+
+  dotokenize(buf);
+
+  if (sizetokstring == 0)	/* nothing?  ok, then let them send it :) */
+    return 0;
+#endif
+
+  switch (next_token(str, available_cmds, &nexttoken))
+  {
+   case HELPTOK:
+    sprintf(buf, "Available commands: %s%s%s",
+	    godlike ? "control ..., " : "",
+#ifdef LEAGUE_SUPPORT
+	    status2->league ? "league ..., " :
+#endif
+	    "",
+	    "version, player ..., queue,");
+    respond(buf, 0);
+    respond("   information ..., observe [%d], parameters, cluecheck [%s], help", 0);
+    return 1;
+   case CONTROLTOK:
+#ifdef GOD_CONTROLS
+#ifdef LEAGUE_SUPPORT
+    if (status2->league)
+    {
+      respond("God controls are disabled during league play.", 1);
+    }
+    else
+#endif
+      return parse_control(nexttoken);
+#else
+    respond("This ntserv is not compiled with god controls.", 1);
+    return 1;
+#endif
+
+   case VERSIONTOK:
+    sprintf(buf, "NetrekII (Paradise), %s", PARAVERS);
+    respond(buf, 0);
+    return 1;
+
+   case PLAYERTOK:
+    return parse_player(nexttoken);
+
+   case QUEUETOK:
+    if (me->p_status == PTQUEUE)
+    {
+      detourneyqueue();
+    }
+    else if (status->tourn)
+      respond("Dude, what are you waiting for!?  It's T-mode.  GO FIGHT!", 1);
+    else if (!(me->p_flags & PFGREEN))
+      respond("Can not enter the tourney queue while at alert", 1);
+#ifndef AEDILE
+    else if (me->p_damage != 0 || me->p_shield < me->p_ship.s_maxshield)
+      respond("Can not enter the tourney queue while damaged", 1);
+#endif
+    else if (me->p_armies > 0)
+      respond("Can not take armies into the tourney queue", 1);
+    else
+    {
+      /*
+       * well, stick them on the queue.  They will be awoken when T mode
+       * arrives
+       */
+
+      evaporate(me);
+
+      /* need code to blab about this */
+      me->p_status = PTQUEUE;
+      sprintf(buf, "%s has entered the tournament queue to wait for T-mode", me->p_name);
+      pmessage(buf, -1, MALL, "GOD->ALL");
+    }
+    return 1;
+
+   case LEAGUETOK:
+#ifdef LEAGUE_SUPPORT
+    if (status2->league)
+    {
+      return parse_league(nexttoken);
+    }
+    else
+    {
+      respond("League commands are disabled during non-league play.", 1);
+      return 1;
+    }
+#else
+    respond("Server is not compiled with league support.", 1);
+    respond("Edit config.h, recompile, install, and nuke the game.", 1);
+    return 1;
+#endif
+
+   case INFOTOK:
+    return parse_info(nexttoken);
+
+   case OBSERVETOK:
+    {
+      int i;
+      i = !get_int(nexttoken, &i, (char **) 0) || i;
+
+      if (i)
+      {
+	if (me->p_observer && me->p_status == POBSERVE)
+	{
+	  respond("You are already an observer.", 1);
+	}
+	else
+	{
+	  if (!(me->p_flags & PFGREEN))
+	    respond("Can not become an observer while at alert", 1);
+	  else if (me->p_damage != 0 || me->p_shield < me->p_ship.s_maxshield)
+	    respond("Can not become an observer while damaged", 1);
+	  else if (me->p_armies > 0)
+	    respond("Can not become an observer while carrying armies", 1);
+	  else
+	  {
+	    evaporate(me);
+	    me->p_status = POBSERVE;
+	  }
+	  me->p_observer = 1;
+	}
+      }
+      else
+      {
+	if (me->p_observer && me->p_status == POBSERVE)
+	{
+	  me->p_observer = 0;
+#if 1
+#ifdef LEAGUE_SUPPORT
+	  if (!(status2->league && status->tourn))
+#endif
+	    evaporate(me);
+#else
+	  respond("You may now self-destruct and reenter the game as a player", 0);
+	  respond("(assuming there's room for you).", 0);
+#endif
+	}
+      }
+    }
+    return 1;
+
+   case PARAMTOK:
+    warning("Transmitting new game parameters");
+    updateGameparams();
+    return 1;
+   case CLUECHECKTOK:
+#if defined(CLUECHECK1) || defined(CLUECHECK2)
+    return accept_cluecheck(nexttoken);
+#else
+    return 0;
+#endif
+   default:
+    return 0;
+  }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/misc.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,562 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <math.h>
+#include <setjmp.h>
+
+#include "defs.h"
+#include "misc.h"
+#include "struct.h"
+#include "data.h"
+#include "daemonII.h"
+#include "planets.h"
+#include "shmem.h"
+
+/*------------------------------MODULE VARIABLES--------------------------*/
+
+#define	NMESSAGES	16
+/* The list of messages that are sent when t-mode begins.  */
+char *warmessages[] = {
+  "A dark mood settles upon the galaxy as galactic superpowers seek",
+  "to colonize new territory.  Border disputes errupt developing",
+  "into all out intergalactic WAR!!!",
+
+  "Slick Willy is elected president, the bottom falls out of the",
+  "economy, and the pressure to expand rises.",
+
+  "Clinton found with foreign emperor's wife!",
+  "All out war ensues.",
+
+  "Taxes on the rich raised to 80%.",
+  "The rich flee to neigboring empire.",
+  "Democrats demand extradition.",
+  "Foreign emporer refuses.",
+  "Hillary 'Rob Them' Clinton declares war!",
+  "Bill hides out in London.",
+
+  "INTERGALACTIC BEER SHORTAGE!",
+  "CHAOS LIKE IT HAS NEVER BEEN SEEN BEFORE ENSUES!",
+
+  "The American Gladiators hit as the number 1 show on the vid.",
+  "After a gruesome episode, the masses storm out their homes,",
+  "jump in their ships, make a quick stop at their local 7-11",
+  "to grab a few beers, then head out into the galaxy to ",
+  "KICK SOME ASS.  WAR!!!!!!!!!!!!!!!!!!",
+
+  "Khan is resurrected in a bizzare experiment.  Elected as head of",
+  "the Starfleet Council, he removes all pajama-wearing new-age",
+  "hippies from Starfleet.  (And you thought the borg were bad news)",
+
+  "Several members of the computing support staff of the Galactic",
+  "Council are severely injured in a freak skydiving accident.",
+  "The rest are killed.  The network collapses without their",
+  "guidance.  Galactic chaos breaks out!",
+
+  /* oldies */
+  "A dark mood settles upon the galaxy",
+  "Political pressure to expand is rising",
+  "Border disputes break out as political tensions increase!",
+  "Galactic superpowers seek to colonize new territory!",
+  "'Manifest Destiny' becomes motive in galactic superpower conflict!",
+  "Diplomat insults foriegn emperor's mother and fighting breaks out!",
+  "Dan Quayle declares self as galactic emperor and chaos breaks out!",
+  "Peace parties have been demobilized, and fighting ensues.",
+};
+
+
+/*
+ * The starting index of the message series and the number of messages in
+ * series.
+ */
+int warm[NMESSAGES][2] = {
+  {0, 3},
+  {3, 2},
+  {5, 2},
+  {7, 6},
+  {13, 2},
+  {15, 5},
+  {20, 3},
+  {23, 4},
+  {27, 1},
+  {28, 1},
+  {29, 1},
+  {30, 1},
+  {31, 1},
+  {32, 1},
+  {33, 1},
+  {34, 1},
+};
+
+
+/* The set of message series that are displayed when t-mode ends. */
+char *peacemessages[] = {
+  "The expansionist pressure subsides, and temporary agreements are",
+  "reached in local border disputes.",
+  "Peace returns to the galaxy...",
+
+  "Wild mob storms the White House!",
+  "Clinton flees.",
+  "Order returns to the galaxy.",
+
+  "Slick Willy apologizes about incident with foreign emperor's wife.",
+  "Claims he did not penetrate.",
+  "Peace ensues.",
+
+  "The economy goes belly up.  The Democrats are kicked",
+  "out of office, tarred and feathered, and sent to the",
+  "zoo on Rigel 4.  Capitalism is implemented and order",
+  "returns to the empire.",
+
+  "Officials sieze hidden beer stockpiles at the Mackenzie brother's",
+  "house!  The beer shortage is over.  Peace breaks out.",
+
+  "The people decide they would rather sit home and watch",
+  "Al Bundy than fight.  The war comes to an end.",
+
+  "Khan takes a fatal blow to the kidneys when Kirk returns from",
+  "retirement and whacks him in the back with his walker.",
+  "It looks like the hippies are back in control.",
+
+  "Sole survivors of the skydiving accident that took out the",
+  "rest of the computing support staff, Michael McLean and Brad",
+  "Spatz are released from intensive care and rebuild the",
+  "network.  Peace is restored to the Galaxy.",
+
+  /* oldies */
+  "A new day dawns as the oppressive mood lifts",
+  "The expansionist pressure subsides",
+  "Temporary agreement is reached in local border disputes.",
+  "Galactic governments reduce colonization efforts.",
+  "'Manifest Destiny is no longer a fad.' says influential philosopher.",
+  "Diplomat apologizes to foreign emperor's mother and invasion is stopped!",
+  "Dan Quayle is locked up and order returns to the galaxy!",
+  "The peace party has reformed, and is rallying for peace",
+};
+
+
+/*
+ * The starting index of each message series and the number of messages in
+ * the series.
+ */
+int peacem[NMESSAGES][2] = {
+  {0, 3},
+  {3, 3},
+  {6, 3},
+  {9, 4},
+  {13, 2},
+  {15, 2},
+  {17, 3},
+  {20, 4},
+  {24, 1},
+  {25, 1},
+  {26, 1},
+  {27, 1},
+  {28, 1},
+  {29, 1},
+  {30, 1},
+  {31, 1},
+};
+
+
+static int series = 0;		/* the message series that was printed */
+/* when t-mode started.  */
+
+/*------------------------------------------------------------------------*/
+
+
+
+
+
+
+
+
+/*------------------------------VISIBLE FUNCTIONS-------------------------*/
+
+/*---------------------------------WARMESSAGE----------------------------*/
+/*
+ * This function is called when t-mode ensues.  It chooses a message series
+ * and prints it out.  It records the message series in the module's series
+ * variable so that the corresponding peace message series can be printed.
+ */
+
+
+void
+warmessage()
+{
+  int i;			/* to hold index */
+  int n;			/* number of messages in a series */
+
+  i = lrand48() % NMESSAGES;	/* choose message series */
+  series = i;			/* record series number */
+  n = warm[i][1];		/* get number of messages in series */
+  i = warm[i][0];		/* get index of first message */
+  while (n > 0)
+  {				/* print all messages */
+    pmessage(warmessages[i], 0, MALL, "  WAR->  ");
+    n--;			/* dec messages that need to be printed */
+    i++;			/* on to next message */
+  }
+}
+
+
+
+
+/*-------------------------------PEACEMESSAGE-----------------------------*/
+/*
+ * This function prints a peace message series.  It uses the module variable
+ * series to decide which message series to print.
+ */
+
+void
+peacemessage()
+{
+  int i;			/* to hold index */
+  int n;			/* number of messages in a series */
+
+  n = peacem[series][1];	/* get # messages in series */
+  i = peacem[series][0];	/* get starting index */
+  while (n > 0)
+  {				/* print all messages */
+    pmessage(peacemessages[i], 0, MALL, " PEACE-> ");
+    n--;			/* dec messages that need to be printed */
+    i++;			/* on to next message */
+  }
+}
+
+
+
+
+/*-------------------------------REALNUMSHIPS-----------------------------*/
+/*
+ * This function counts the number of ships on a team.  If the race is the
+ * same and the slot is not free, then the ship is counted.
+ */
+
+int
+realNumShips(owner)
+  int owner;
+{
+  int i, num;			/* for looping and counting */
+  struct player *p;		/* to point to a player */
+
+  num = 0;			/* zero the count of ships */
+  for (i = 0, p = players; i < MAXPLAYER; i++, p++)
+    if ((p->p_status != PFREE) && (p->p_team == owner))
+      num++;			/* found a ship on this team, inc count */
+  return (num);			/* return number of ships */
+}
+
+
+
+
+/*------------------------------TOURNAMENTMODE------------------------------*/
+/*
+ * This function counts the number of players on each team and sees if two or
+ * more teams have enough players for t-mode.  It returns a 0 for no team
+ * mode.
+ */
+
+int
+tournamentMode()
+{
+#ifdef LEAGUE_SUPPORT
+  if (status2->league)
+  {
+    return status2->league > 2;
+  }
+  else
+#endif
+  {
+    static int wt = -1;		/* warning time due to team unbalance */
+    static int lct = -1;	/* last check time (ie last pmessage) */
+    char buf[80];		/* temp buffer for tmode halt warning */
+    int i;			/* looping vars */
+    int t[16];			/* to count players with */
+    int counts[4];
+    /* Even though teams go from 0-8, leave */
+    /* this at 16...or bad things might happen */
+    struct player *p;		/* to point to players */
+
+    for (i = 0; i < 16; i++)	/* clear the team players count array */
+      t[i] = 0;
+    for (i = 0, p = players; i < MAXPLAYER; i++, p++)
+    {
+      /* through all players */
+      if (p->p_flags & PFROBOT)
+	continue;		/* robots don't count */
+      switch (p->p_status)
+      {
+       case PFREE:		/* free slots obviously don't count */
+       case POBSERVE:		/* and neither do observers */
+	break;
+
+       case PEXPLODE:
+       case PDEAD:
+       case PTQUEUE:
+	if (p->p_observer)
+	  break;		/* observers in these modes don't count */
+	/* we can fall through here */
+       case POUTFIT:
+       case PALIVE:
+	t[p->p_team]++;
+	break;
+      }
+    }
+
+#if 0
+    /*
+     * If there hasnt been tmode recently (5 mins) clear all kills, and place
+     * any carried armies on the most valuble planet not yet finished.
+     */
+    if ((tourntimestamp - ticks) > 300)
+    {
+      int x;
+      for (x = 0; x < MAXPLAYER; x++)
+      {
+	if (players[x].p_status == PALIVE)
+	{
+	  if (players[x].p_armies > 0)
+	    PlaceLostArmies(players[x]);
+	  players[x].p_kills = 0.00;
+	}
+      }
+    }
+#endif
+
+    i = 0;			/* zero count of # teams with tournplayers */
+    if (t[FED] >= configvals->tournplayers)
+      counts[i++] = t[FED];
+    if (t[ROM] >= configvals->tournplayers)
+      counts[i++] = t[ROM];
+    if (t[KLI] >= configvals->tournplayers)
+      counts[i++] = t[KLI];
+    if (t[ORI] >= configvals->tournplayers)
+      counts[i++] = t[ORI];
+    if (i > 1)
+    {
+      i = counts[0] - counts[1];
+      if (i < 0)
+	i = -i;
+      if (i > 2)
+      {				/* Too big a team imbalance */
+	if (wt == -1)
+	{
+	  wt = status->clock;	/* make a timestamp */
+	  lct = status->clock;	/* used every time it sends the message */
+	}
+	if (((status->clock - wt) < 3) && (lct < status->clock))
+	{
+	  /*
+	   * teams are unblananced, and we havn't said anything for 1 minute,
+	   * warn everybody  Yes these are noisy strings, but I want people
+	   * to notice.
+	   */
+	  pmessage("**************************!! Teams are unbalanced !!**************************", 0, MALL, "");
+	  sprintf(buf, "********************* TMODE WILL BE HALTED IN %li MINUTES **********************", (wt - status->clock) + 3);
+	  pmessage(buf, 0, MALL, "");
+	  pmessage("************************ if this problem isn't fixed *************************", 0, MALL, "");
+	  lct = status->clock;
+	  return (1);
+	}
+	else if ((status->clock - wt) >= 3)
+	{
+	  if (lct < status->clock)
+	  {
+	    pmessage("******************!! TMODE HALTED DUE TO TEAM IMBALANCE !!********************", 0, MALL, "");
+	    pmessage("*********************** Balance teams to resume tmode ************************", 0, MALL, "");
+	    lct = status->clock;
+	  }
+	  return (0);		/* stop tmode */
+	}
+      }
+      else
+      {
+	wt = -1;
+	lct = -1;
+      }
+      return (1);		/* return that we are in team mode */
+    }
+    return (0);			/* we are NOT in T-mode */
+  }
+}
+
+
+
+
+/*------------------------------PMESSAGE----------------------------------*/
+/*
+ * This function sens a message to the message board.  It places the message
+ * in the next position of the array of messages.  The message will ahve a
+ * header attached to the front of it.
+ */
+
+void
+pmessage(str, recip, group, address)
+  char *str;			/* the message */
+  int recip;			/* who is the recipient */
+  int group;			/* group sent to and other flags */
+  char *address;		/* the header attached to front */
+{
+  struct message *cur;		/* to pnt to where to put message */
+  int mesgnum;			/* to hold index into array of messgs */
+
+  if ((mesgnum = ++(mctl->mc_current)) >= MAXMESSAGE)
+  {
+    mctl->mc_current = 0;	/* move to next index in array and */
+    mesgnum = 0;		/* warp around if necessart */
+  }
+  cur = &messages[mesgnum];	/* get addr of message struct */
+  cur->m_no = mesgnum;		/* set the message number */
+  cur->m_flags = group;		/* set the group and flags */
+  cur->m_recpt = recip;		/* set the recipient */
+  cur->m_from = 255;		/* message is from God */
+  (void) sprintf(cur->m_data, "%s %s", address, str);
+  cur->m_flags |= MVALID;	/* mark message as valid */
+}
+
+/*
+ * send a message from god to a particular player number
+ */
+
+void
+god2player(str, pno)
+  char *str;
+  int pno;
+{
+  struct message *cur;		/* to pnt to where to put message */
+
+  if (++(mctl->mc_current) >= MAXMESSAGE)
+    mctl->mc_current = 0;
+  cur = &messages[mctl->mc_current];
+
+  cur->m_no = mctl->mc_current;
+  cur->m_flags = MINDIV;
+  cur->m_recpt = pno;
+  cur->m_from = 255;
+  sprintf(cur->m_data, "%s->%s %s", SERVNAME, twoletters(&players[pno]), str);
+  cur->m_flags |= MVALID;
+}
+
+
+
+
+/*-------------------------------LOG_MESSAGE------------------------------*/
+/*
+ * This function writes a message into the logfile so the server god can look
+ * at it later.  It is used to record messages to god.
+ */
+
+void
+log_message(msg)
+  char *msg;			/* the string to record */
+{
+  char *path;			/* to hold path and name */
+  FILE *logfile;		/* to open logfile with */
+
+  path = build_path(GODLOGFILE);
+  logfile = fopen(path, "a");	/* let's try to open this baby */
+  if (!logfile)			/* Oops, could not open */
+    return;			/* OUt of Dodge */
+  fputs(msg, logfile);		/* write the string to log file */
+  fclose(logfile);		/* close the sucker up */
+}
+
+
+
+
+/*-----------------------------PARSE_GODMESSAGES----------------------------*/
+/*
+ * This function checks all the new messages in the message buffer to see if
+ * the keyword 'GOD' is at the first of the message.  If it is, then the
+ * message is written to the logfile so the server god can read it.
+ */
+
+void
+parse_godmessages()
+{
+  char buf[100];		/* bigger than message::m_data */
+  static int lastparsed = 0;	/* keeps track of last message */
+
+  while (mctl->mc_current != lastparsed)
+  {				/* Is there a new message? */
+    struct message *cur;	/* to point to message struct */
+    char *s;			/* to point to string */
+
+    if (++lastparsed >= MAXMESSAGE)	/* rollover after end of */
+      lastparsed = 0;		/* message buffer */
+    cur = &messages[lastparsed];/* get new message's struct */
+
+    s = cur->m_data + 1;	/* get new message */
+    while ((*s != ' ') && (*s != 0))	/* go until first space */
+      s++;
+    while ((*s == ' ') && (*s != 0))	/* go past spaces */
+      s++;
+
+    if (!*s)
+      continue;
+    if (cur->m_flags & MGOD)
+    {				/* if it's to god */
+      sprintf(buf, "%s says: %s\n",
+	      (cur->m_from >= 0) ? players[cur->m_from].p_name : "who", s);
+    }
+    else
+    {				/* or labeled to god */
+      if (strncmp(s, "GOD:", 4) != 0)	/* check for 'GOD' keyword */
+	continue;		/* not there, then next message */
+      sprintf(buf, "GOD: %s\n", s + 4);
+    }
+    log_message(buf);		/* go record message */
+  }
+}
+
+/*------------------------------------------------------------------------*/
+#if 0
+/*--------------------------[ PLACE LOST ARMIES ]--------------------------*/
+/*
+ * If for some reason or another the player looses armies they may be
+ * carrying (ghost busts, etc), this function is called, to place the armies
+ * safely on a planet.
+ */
+
+void
+PlaceLostArmies(victim)
+  struct player *victim;
+{
+  char buf[80];
+  int i, startplanet;
+
+  ((startplanet = find_start_planet(me->p_team, PLSHIPYARD)) != -1
+   || (startplanet = find_start_planet(me->p_team, PLREPAIR)) != -1
+   || (startplanet = find_start_planet(me->p_team, PLAGRI)) != -1
+   || (startplanet = find_start_planet(me->p_team, PLFUEL)) != -1
+   || (startplanet = find_start_planet(me->p_team, PLHOME)) != -1
+   || (startplanet = find_start_planet(me->p_team, 0)) != -1);
+  planets[startplanet].pl_armies += victim->p_armies;
+  (void) sprintf(buf, "%s's %d armies placed on %s",
+	    victim->p_name, victim->p_armies, planets[startplanet].pl_name);
+  pmessage(buf, 0, MALL | MGHOST, MSERVA);
+}
+#endif
+/*------------------------------------------------------------------------*/
+
+/*----------END OF FILE--------*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/misc.h	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,36 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+
+
+/*----------------------------FUNCTION PROTOTYPES--------------------------*/
+void warmessage();		/* print war message */
+void peacemessag();		/* print peace message */
+int realNumShips();		/* number of ships on a team */
+int tournamentMode();		/* are we in tourn mode */
+void pmessage( /* char *str, int recip, int group, char addr */ );
+void parse_godmesssages();	/* detect and log messages to god */
+void PlaceLostArmies();		/* places armies from a lost/GB'd player */
+/*--------------------------------------------------------------------------*/
+
+
+
+
+
+
+/*--------END OF FILE------*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/orbit.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,235 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <stdio.h>
+#include <math.h>
+#include <sys/types.h>
+#include <sys/ipc.h>
+#include <sys/shm.h>
+
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "shmem.h"
+
+
+void newdock();
+extern void move_player();
+
+static void
+deactivate_friendly_tractors()
+{
+  int i;
+  struct player *j;
+  for (i = 0; i < MAXPLAYER; i++)
+  {
+    j = &players[i];
+    if (i == me->p_no ||
+	j->p_team != me->p_team ||
+	j->p_tractor != me->p_no)
+      continue;
+    /* j is a teammate with a tractor on me */
+    j->p_flags &= ~(PFTRACT | PFPRESS);
+  }
+}
+
+/*------------------------------VISIBLE PROCEDURES-------------------------*/
+
+/*-----------------------------------ORBIT---------------------------------*/
+/*
+ * This function is called when the player presses the orbit key.  It check
+ * to make sure the player can orbit or dock.  The function first checks
+ * through the list of players to see whether there is a dockable ship
+ * nearby. If that fails, then the planets are checked.  The orbit key acts
+ * as a toggle for docking and undocking at ships.
+ */
+
+void
+orbit()
+{
+  register int i;		/* looping var */
+  register struct planet *l;	/* to point to planet being orbited */
+  unsigned char dir;		/* to hold course direction */
+  int dx, dy;			/* used to get distances */
+
+  if (me->p_flags & PFORBIT)	/* if not orbiting then we do not */
+    return;			/* need to do this section of code */
+  if (me->p_speed > ORBSPEED)
+  {				/* going to fast to orbit/dock? */
+    warning("Helmsman: Captain, the maximum safe speed for docking or orbiting is warp 2!");
+    return;
+  }				/* turn off flags */
+  me->p_flags &= ~(PFORBIT | PFTRACT | PFPRESS |
+		   PFWARP | PFAFTER | PFWARPPREP | PFWPSUSPENDED);
+  me->p_warptime = 0;
+  if ((me->p_flags & PFDOCK) && (players[me->p_docked].p_speed > 4))
+  {
+    warning("It's unsafe to disengage from your base when it's moving faster then warp 4.");	/* cannot disengage from
+												 * travelling */
+    return;			/* ships if they are going too fast */
+  }
+  else
+    undock_player(me);		/* make sure I'm not docked */
+
+  if (!(me->p_flags & PFPLLOCK) &&	/* if we aren't locked onto a planet */
+      can_dock(me->p_ship))
+  {				/* and player can dock then */
+    for (i = 0; i < MAXPLAYER; i++)
+    {				/* go through all players */
+      if (me->p_no == i)	/* and look for ships that */
+	continue;		/* allow docking */
+      if (!allows_docking(players[i].p_ship))
+	continue;
+      if (!isAlive(&players[i]))/* disregard players not */
+	continue;		/* alive */
+      if (!friendlyPlayer(&players[i]))	/* cannot dock on enemy */
+	continue;		/* players */
+      dx = ABS(players[i].p_x - me->p_x);	/* get distance */
+      dy = ABS(players[i].p_y - me->p_y);
+      if (dx > DOCKDIST || dy > DOCKDIST)	/* if too far away thenwe */
+	continue;		/* cannot dock here */
+      newdock(i);		/* find a port and dock */
+
+      deactivate_friendly_tractors();
+
+      me->p_flags &= ~(PFPLOCK | PFPLLOCK);
+      return;
+    }
+  }				/* not docking with a player try a planet */
+  if (!(me->p_flags & PFPLOCK))
+  {				/* aren't locked onto a player */
+    for (i = 0, l = &planets[i]; i < NUMPLANETS; i++, l++)
+    {
+      switch (PL_TYPE(*l))
+      {
+       case PLSTAR:
+       case PLNEB:
+       case PLBHOLE:
+       case PLPULSAR:
+	continue;
+      }
+
+      dx = ABS(l->pl_x - me->p_x);	/* go through all planets and */
+      dy = ABS(l->pl_y - me->p_y);	/* check their distance */
+      if (dx > ENTORBDIST || dy > ENTORBDIST)
+	continue;		/* too far away?  */
+      if (dx * dx + dy * dy > ENTORBDIST * ENTORBDIST)
+	continue;
+
+      if (!(me->p_team & planets[i].pl_owner) &&	/* can player orbit? */
+	  !(me->p_ship.s_nflags & SFNCANORBIT))
+      {
+	warning("Central Command regulations prohibits you from orbiting foreign planets");
+	return;
+      }
+      if (!(me->p_team & planets[i].pl_owner) &&
+	  !status->tourn)
+      {
+	warning("No one is allowed to orbit alien planets outside of T-mode");
+	return;
+      }
+      dir = (unsigned char) (int) (atan2((double) (me->p_x - l->pl_x),
+			    (double) (l->pl_y - me->p_y)) / 3.14159 * 128.);
+      scout_planet(me->p_no, l->pl_no);
+#if 0
+      l->pl_torbit |= me->p_team;	/* place team in orbit */
+#endif
+      me->p_orbitdir = drand48() < configvals->orbitdirprob;
+      me->p_dir = dir + (me->p_orbitdir ? 64 : -64);	/* get direction for
+							 * player */
+      me->p_flags |= PFORBIT;	/* set his orbiting flag */
+
+      deactivate_friendly_tractors();
+
+      move_player(me->p_no, (int) (l->pl_x + ORBDIST * Cos[dir]),
+		  (int) (l->pl_y + ORBDIST * Sin[dir]), 1);
+
+      me->p_speed = me->p_desspeed = 0;
+      me->p_planet = l->pl_no;	/* set the planet number */
+      me->p_flags &= ~(PFPLOCK | PFPLLOCK);
+      return;
+    }
+  }
+  warning("Helmsman:  Sensors read no valid targets in range to dock or orbit sir!");
+}
+
+
+
+/*--------------------------------NEWDOCK--------------------------------*/
+
+
+void
+newdock(base_num)
+  int base_num;
+{
+  char buf[80];			/* to sprintf into */
+  struct player *base = &players[base_num];
+  int port_id, i;
+  int numports;
+  long distsq = 0;		/* will be set by the first matching port */
+  int angle;
+
+  if (!(base->p_flags & PFDOCKOK))
+  {				/* docking allowed? */
+    sprintf(buf, "Starbase %s refusing us docking permission captain.",
+	    base->p_name);
+    warning(buf);		/* if not say so then */
+    return;			/* get out of here */
+  }
+  port_id = -1;
+  numports = base->p_ship.s_numports;
+
+  for (i = 0; i < numports; i++)
+  {
+    long ds;
+    int x, y;
+    angle = (2 * i + 1) * 128 / numports;
+
+    if (base->p_port[i] != VACANT)
+      continue;
+
+    x = base->p_x + DOCKDIST * Cos[angle];
+    y = base->p_y + DOCKDIST * Sin[angle];
+    ds = (me->p_x - x) * (me->p_x - x) + (me->p_y - y) * (me->p_y - y);
+    if (port_id == -1 || ds < distsq)
+    {
+      port_id = i;
+      distsq = ds;
+    }
+  }
+
+  if (port_id < 0)
+  {
+    sprintf(buf, "Base %s: Permission to dock denied, all ports currently occupied.", base->p_name);
+    warning(buf);		/* print message to tell */
+    return;
+  }				/* player */
+  dock_to(me, base_num, port_id);
+
+  sprintf(buf, "Helmsman:  Docking manuever completed Captain.  All moorings secured at port %d.", port_id);
+  warning(buf);			/* tell user he's docked */
+}
+
+#ifdef HAVE_RAND48
+
+double drand48();
+
+#endif
+
+/*------------END OF FILE--------*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/packets.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,359 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <stddef.h>
+#include <stdio.h>
+
+#include "defs.h"
+#include "packets.h"
+#include "gppackets.h"
+#include "wtext.h"
+
+size_t client_packet_sizes[] = {
+  0,
+  sizeof(struct mesg_cpacket),
+  sizeof(struct speed_cpacket),
+  sizeof(struct dir_cpacket),
+  sizeof(struct phaser_cpacket),
+  sizeof(struct plasma_cpacket),
+  sizeof(struct torp_cpacket),
+  sizeof(struct quit_cpacket),
+  sizeof(struct login_cpacket),
+  sizeof(struct outfit_cpacket),
+  /* 10 v */
+  sizeof(struct war_cpacket),
+  sizeof(struct practr_cpacket),
+  sizeof(struct shield_cpacket),
+  sizeof(struct repair_cpacket),
+  sizeof(struct orbit_cpacket),
+  sizeof(struct planlock_cpacket),
+  sizeof(struct playlock_cpacket),
+  sizeof(struct bomb_cpacket),
+  sizeof(struct beam_cpacket),
+  sizeof(struct cloak_cpacket),
+  /* 20 v */
+  sizeof(struct det_torps_cpacket),
+  sizeof(struct det_mytorp_cpacket),
+  sizeof(struct copilot_cpacket),
+  sizeof(struct refit_cpacket),
+  sizeof(struct tractor_cpacket),
+  sizeof(struct repress_cpacket),
+  sizeof(struct coup_cpacket),
+  sizeof(struct socket_cpacket),
+  sizeof(struct options_cpacket),
+  sizeof(struct bye_cpacket),
+  /* 30 v */
+  sizeof(struct dockperm_cpacket),
+  sizeof(struct updates_cpacket),
+  sizeof(struct resetstats_cpacket),
+  sizeof(struct reserved_cpacket),
+  sizeof(struct scan_cpacket),
+  sizeof(struct udp_req_cpacket),
+  sizeof(struct sequence_cpacket),
+  sizeof(struct rsa_key_cpacket),
+  sizeof(struct obvious_packet),
+  0,
+  /* 40 v */
+  0,
+  0,
+  sizeof(struct ping_cpacket),
+#ifdef SHORT_PACKETS
+  sizeof(struct shortreq_cpacket),
+  sizeof(struct threshold_cpacket),
+  0,				/* CP_S_MESSAGE */
+  0,				/* CP_S_RESERVED */
+  0,				/* CP_S_DUMMY */
+#else
+  0,
+  0,
+  0,
+  0,
+  0,
+#endif
+  0,
+  0,
+  /* 50 v */
+  0,
+  0,
+  0,
+  0,
+  0,
+  0,
+  0,
+  0,
+  0,
+  0,
+  /* 60 v */
+#ifdef FEATURE
+  sizeof(struct feature_cpacket)/* CP_FEATURE */
+#else
+  0
+#endif
+};
+
+#define num_cpacket_sizes	sizeof(client_packet_sizes)/sizeof(*client_packet_sizes)
+
+int 
+size_of_cpacket(pkt)
+  void *pkt;
+{
+  CARD8 type;
+  CARD8 subtype;
+
+  type = ((CARD8 *) pkt)[0];
+  subtype = ((CARD8 *) pkt)[1];
+
+  if (type < num_cpacket_sizes && client_packet_sizes[type] > 0)
+    return client_packet_sizes[type];
+
+  switch (type)
+  {
+#ifdef CP_FIRE_WEAPON
+   case CP_FIRE_WEAPON:
+    {
+      struct fire_weapon_cpacket *fwp = pkt;
+      return (fwp->mech == TM_POSITION) ? 12 : 4;
+    }
+#endif
+
+#ifdef SHORT_PACKETS
+   case CP_S_MESSAGE:
+    return ((unsigned char *) pkt)[3];
+   case CP_S_RESERVED:
+   case CP_S_DUMMY:
+    /* hmm? good question */
+    return 0;
+#endif				/* SHORT_PACKETS */
+
+   default:
+    return 0;
+  }
+}
+
+
+int server_packet_sizes[] = {
+  0,				/* record 0 */
+  sizeof(struct mesg_spacket),	/* SP_MESSAGE */
+  sizeof(struct plyr_info_spacket),	/* SP_PLAYER_INFO */
+  sizeof(struct kills_spacket),	/* SP_KILLS */
+  sizeof(struct player_spacket),/* SP_PLAYER */
+  sizeof(struct torp_info_spacket),	/* SP_TORP_INFO */
+  sizeof(struct torp_spacket),	/* SP_TORP */
+  sizeof(struct phaser_spacket),/* SP_PHASER */
+  sizeof(struct plasma_info_spacket),	/* SP_PLASMA_INFO */
+  sizeof(struct plasma_spacket),/* SP_PLASMA */
+  /* 10 v */
+  sizeof(struct warning_spacket),	/* SP_WARNING */
+  sizeof(struct motd_spacket),	/* SP_MOTD */
+  sizeof(struct you_spacket),	/* SP_YOU */
+  sizeof(struct queue_spacket),	/* SP_QUEUE */
+  sizeof(struct status_spacket),/* SP_STATUS */
+  sizeof(struct planet_spacket),/* SP_PLANET */
+  sizeof(struct pickok_spacket),/* SP_PICKOK */
+  sizeof(struct login_spacket),	/* SP_LOGIN */
+  sizeof(struct flags_spacket),	/* SP_FLAGS */
+  sizeof(struct mask_spacket),	/* SP_MASK */
+  /* 20 v */
+  sizeof(struct pstatus_spacket),	/* SP_PSTATUS */
+  sizeof(struct badversion_spacket),	/* SP_BADVERSION */
+  sizeof(struct hostile_spacket),	/* SP_HOSTILE */
+  sizeof(struct stats_spacket),	/* SP_STATS */
+  sizeof(struct plyr_login_spacket),	/* SP_PL_LOGIN */
+  sizeof(struct reserved_spacket),	/* SP_RESERVED */
+  sizeof(struct planet_loc_spacket),	/* SP_PLANET_LOC */
+  sizeof(struct scan_spacket),	/* SP_SCAN (ATM) */
+  sizeof(struct udp_reply_spacket),	/* SP_UDP_REPLY */
+  sizeof(struct sequence_spacket),	/* SP_SEQUENCE */
+  /* 30 v */
+  sizeof(struct sc_sequence_spacket),	/* SP_SC_SEQUENCE */
+  sizeof(struct rsa_key_spacket),	/* SP_RSA_KEY */
+  sizeof(struct motd_pic_spacket),	/* SP_MOTD_PIC */
+  sizeof(struct stats_spacket2),/* SP_STATS2 */
+  sizeof(struct status_spacket2),	/* SP_STATUS2 */
+  sizeof(struct planet_spacket2),	/* SP_PLANET2 */
+  sizeof(struct obvious_packet),/* SP_NEW_MOTD */
+  sizeof(struct thingy_spacket),/* SP_THINGY */
+  sizeof(struct thingy_info_spacket),	/* SP_THINGY_INFO */
+  sizeof(struct ship_cap_spacket),	/* SP_SHIP_CAP */
+  /* 40 v */
+#ifdef SHORT_PACKETS
+  sizeof(struct shortreply_spacket),	/* SP_S_REPLY */
+  -1,				/* SP_S_MESSAGE */
+  -1,				/* SP_S_WARNING */
+  sizeof(struct youshort_spacket),	/* SP_S_YOU */
+  sizeof(struct youss_spacket),	/* SP_S_YOU_SS */
+  -1,				/* SP_S_PLAYER */
+#else
+  -1,
+  -1,
+  -1,
+  -1,
+  -1,
+  -1,
+#endif
+  sizeof(struct ping_spacket),	/* SP_PING */
+  -1,				/* SP_S_TORP */
+  -1,				/* SP_S_TORP_INFO */
+  20,				/* SP_S_8_TORP */
+  /* 50 v */
+  -1,				/* SP_S_PLANET */
+  -1,				/* SP_GPARAM */
+  -1,				/* SP_PARADISE_EXT1 */
+  sizeof(struct terrain_packet2),	/* SP_TERRAIN2 */
+  sizeof(struct terrain_info_packet2),	/* SP_TERRAIN_INFO2 */
+  -1,
+  -1,
+  -1,
+  -1,
+  -1,
+  /* 60 v */
+#ifdef FEATURE
+  sizeof(struct feature_spacket),	/* SP_FEATURE */
+#else
+  -1,
+#endif
+  -1				/* The fools!  I'm pretty sure there is an */
+  /* off-by-one in the test in the last fnc */
+  /* but I'm not sure and this WILL fix it. */
+};
+
+#define num_spacket_sizes (sizeof(server_packet_sizes) / sizeof(server_packet_sizes[0]) - 1)
+
+#ifdef SHORT_PACKETS
+unsigned char numofbits[256] =
+{0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4, 1,
+  2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 1,
+  2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2,
+  3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 1,
+  2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2,
+  3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 2,
+  3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3,
+  4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 1,
+  2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2,
+  3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 2,
+  3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3,
+  4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 2,
+  3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3,
+  4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 3,
+  4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 4,
+5, 5, 6, 5, 6, 6, 7, 5, 6, 6, 7, 6, 7, 7, 8};
+
+static int vtsize[9] =
+{4, 8, 8, 12, 12, 16, 20, 20, 24};	/* How big is the torppacket */
+int vtisize[9] =
+{4, 7, 9, 11, 13, 16, 18, 20, 22};	/* 4 byte Header + torpdata */
+
+
+static int 
+padto4(sz)
+  int sz;
+{
+  return (sz % 4) ? (sz / 4 + 1) * 4 : sz;
+
+}
+#endif
+
+int
+size_of_spacket(pkt)
+  unsigned char *pkt;
+{
+  switch (pkt[0])
+  {
+   case SP_GPARAM:
+    switch (pkt[1])
+    {
+     case 0:
+      return sizeof(struct gp_sizes_spacket);
+     case 1:
+      return sizeof(struct gp_team_spacket);
+     case 2:
+      return sizeof(struct gp_teamlogo_spacket);
+     case 3:
+      return sizeof(struct gp_shipshape_spacket);
+     case 4:
+      return sizeof(struct gp_shipbitmap_spacket);
+     case 5:
+      return sizeof(struct gp_rank_spacket);
+     case 6:
+      return sizeof(struct gp_royal_spacket);
+     case 7:
+      return sizeof(struct gp_teamplanet_spacket);
+     default:
+      return 0;
+    }
+#ifdef SHORT_PACKETS
+   case SP_S_MESSAGE:
+    return padto4(pkt[4]);	/* IMPORTANT  Changed */
+   case SP_S_WARNING:
+    if (pkt[1] == STEXTE_STRING ||
+	pkt[1] == SHORT_WARNING)
+    {
+      return padto4(pkt[3]);
+    }
+    else
+      return 4;			/* Normal Packet */
+   case SP_S_PLAYER:
+    if (pkt[1] & 128)
+    {				/* Small +extended Header */
+      return padto4(((pkt[1] & 0x3f) * 4) + 4);
+    }
+    else if (pkt[1] & 64)
+    {				/* Small Header */
+      return padto4(((pkt[1] & 0x3f) * 4) + 4);
+    }
+    else
+    {				/* Big Header */
+      return padto4((pkt[1] * 4 + 12));
+    }
+   case SP_S_TORP:
+    return padto4(vtsize[numofbits[pkt[1]]]);
+   case SP_S_TORP_INFO:
+    return padto4((vtisize[numofbits[pkt[1]]] + numofbits[pkt[3]]));
+   case SP_S_PLANET:
+    return padto4((pkt[1] * VPLANET_SIZE) + 2);
+#endif
+   case SP_PARADISE_EXT1:
+    switch (pkt[1])
+    {
+     case SP_PE1_MISSING_BITMAP:
+      return sizeof(struct pe1_missing_bitmap_spacket);
+     case SP_PE1_NUM_MISSILES:
+      return sizeof(struct pe1_num_missiles_spacket);
+     default:
+      return 0;
+    }
+   default:
+#ifdef FEATURE_DIAG
+    if (*pkt >= num_spacket_sizes)
+    {
+      fprintf(stderr, "Packet type %d out of range (%d)\n", (*pkt), num_spacket_sizes);
+      return (0);
+    }
+    if (server_packet_sizes[*pkt] < 0)
+    {
+      fprintf(stderr, "Packet type %d has no size\n", (*pkt));
+      return (0);
+    }
+    return (server_packet_sizes[*pkt]);
+#else
+    return (*pkt < num_spacket_sizes && server_packet_sizes[*pkt] >= 0)
+      ? server_packet_sizes[*pkt] : 0;
+#endif
+  }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/packets.h	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,1201 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+/*
+ * Include file for socket I/O xtrek.
+ * 
+ * Kevin P. Smith 1/29/89
+ */
+
+#ifndef packets_h_
+#define packets_h_
+
+#define STATUS_TOKEN	"\t@@@"	/* ATM */
+
+/*
+ * the following typedefs allow portability to machines without the
+ * ubiquitous 32-bit architecture (KSR1, Cray, DEC Alpha)
+ */
+
+typedef unsigned int CARD32;
+typedef unsigned short CARD16;
+typedef unsigned char CARD8;
+
+typedef int INT32;
+typedef short INT16;
+#if __STDC__ || defined(sgi) || defined(AIXV3)
+typedef signed char INT8;
+#else
+/* stupid compilers */
+typedef char INT8;
+#endif
+
+/*
+ * TCP and UDP use identical packet formats; the only difference is that,
+ * when in UDP mode, all packets sent from server to client have a sequence
+ * number appended to them.
+ * 
+ * (note: ALL packets, whether sent on the TCP or UDP channel, will have the
+ * sequence number.  Thus it's important that client & server agree on when
+ * to switch.  This was done to keep critical and non-critical data in sync.)
+ */
+
+/*
+ * the various pad members of the structures are used for explicit data
+ * alignment.  They do not contain any data.  All structures are aligned to 4
+ * bytes.  If your compiler forces 8 byte alignment, you will not be adhering
+ * to the netrek protocol.
+ */
+
+/* packets sent from xtrek server to remote client */
+#define SP_MESSAGE 	1
+#define SP_PLAYER_INFO 	2	/* general player info not elsewhere */
+#define SP_KILLS	3	/* # kills a player has */
+#define SP_PLAYER	4	/* x,y for player */
+#define SP_TORP_INFO	5	/* torp status */
+#define SP_TORP		6	/* torp location */
+#define SP_PHASER	7	/* phaser status and direction */
+#define SP_PLASMA_INFO	8	/* player login information */
+#define SP_PLASMA	9	/* like SP_TORP */
+#define SP_WARNING	10	/* like SP_MESG */
+#define SP_MOTD		11	/* line from .motd screen */
+#define SP_YOU		12	/* info on you? */
+#define SP_QUEUE	13	/* estimated loc in queue? */
+#define SP_STATUS	14	/* galaxy status numbers */
+#define SP_PLANET 	15	/* planet armies & facilities */
+#define SP_PICKOK	16	/* your team & ship was accepted */
+#define SP_LOGIN	17	/* login response */
+#define SP_FLAGS	18	/* give flags for a player */
+#define SP_MASK		19	/* tournament mode mask */
+#define SP_PSTATUS	20	/* give status for a player */
+#define SP_BADVERSION   21	/* invalid version number */
+#define SP_HOSTILE	22	/* hostility settings for a player */
+#define SP_STATS	23	/* a player's statistics */
+#define SP_PL_LOGIN	24	/* new player logs in */
+#define SP_RESERVED	25	/* for future use */
+#define SP_PLANET_LOC	26	/* planet name, x, y */
+
+#define SP_SCAN		27	/* scan packet */
+#define SP_UDP_REPLY	28	/* notify client of UDP status */
+#define SP_SEQUENCE	29	/* sequence # packet */
+#define SP_SC_SEQUENCE	30	/* this trans is semi-critical info */
+#define SP_RSA_KEY	31	/* RSA data packet */
+
+#define SP_MOTD_PIC     32	/* motd bitmap pictures */
+#define SP_STATS2	33	/* new stats packet */
+#define SP_STATUS2	34	/* new status packet */
+#define SP_PLANET2	35	/* new planet packet */
+#define SP_NEW_MOTD     36	/* New MOTD info notification uses */
+#define SP_THINGY	37	/* thingy location */
+#define SP_THINGY_INFO	38	/* thingy status */
+#define SP_SHIP_CAP	39	/* ship capabilities */
+
+#ifdef SHORT_PACKETS
+#define SP_S_REPLY      40	/* reply to send-short request */
+#define SP_S_MESSAGE    41	/* var. Message Packet */
+#define SP_S_WARNING    42	/* Warnings with 4  Bytes */
+#define SP_S_YOU        43	/* hostile,armies,whydead,etc .. */
+#define SP_S_YOU_SS     44	/* your ship status */
+#define SP_S_PLAYER     45	/* variable length player packet */
+#endif
+
+#define SP_PING         46	/* ping packet */
+
+#ifdef SHORT_PACKETS
+#define SP_S_TORP       47	/* variable length torp packet */
+#define SP_S_TORP_INFO  48	/* SP_S_TORP with TorpInfo */
+#define SP_S_8_TORP     49	/* optimized SP_S_TORP */
+#define SP_S_PLANET     50	/* see SP_PLANET */
+
+/* variable length packets */
+#define VPLAYER_SIZE    4
+#define SHORTVERSION    10	/* other number blocks, like UDP Version */
+#endif
+
+#define SP_GPARAM	51	/* game params packet */
+
+/*
+ * the following is a family of packets with the same type, but a
+ * discriminating subtype
+ */
+#define SP_PARADISE_EXT1	52
+#define SP_PE1_MISSING_BITMAP	0
+#define SP_PE1_NUM_MISSILES	1
+/* end of packet 52 subtypes */
+#define SP_TERRAIN2	53	/* GZipped terrain packet - 5/16/95 rpg */
+#define SP_TERRAIN_INFO2 54	/* header for terrain info */
+#ifdef FEATURE
+#define SP_FEATURE	60	/* type 60 is FEATURE packets */
+#endif
+
+
+/* packets sent from remote client to xtrek server */
+#define CP_MESSAGE      1	/* send a message */
+#define CP_SPEED	2	/* set speed */
+#define CP_DIRECTION	3	/* change direction */
+#define CP_PHASER	4	/* phaser in a direction */
+#define CP_PLASMA	5	/* plasma (in a direction) */
+#define CP_TORP		6	/* fire torp in a direction */
+#define CP_QUIT		7	/* self destruct */
+#define CP_LOGIN	8	/* log in (name, password) */
+#define CP_OUTFIT	9	/* outfit to new ship */
+#define CP_WAR		10	/* change war status */
+#define CP_PRACTR	11	/* create practice robot? */
+#define CP_SHIELD	12	/* raise/lower sheilds */
+#define CP_REPAIR	13	/* enter repair mode */
+#define CP_ORBIT	14	/* orbit planet/starbase */
+#define CP_PLANLOCK	15	/* lock on planet */
+#define CP_PLAYLOCK	16	/* lock on player */
+#define CP_BOMB		17	/* bomb a planet */
+#define CP_BEAM		18	/* beam armies up/down */
+#define CP_CLOAK	19	/* cloak on/off */
+#define CP_DET_TORPS	20	/* detonate enemy torps */
+#define CP_DET_MYTORP	21	/* detonate one of my torps */
+#define CP_COPILOT	22	/* toggle copilot mode */
+#define CP_REFIT	23	/* refit to different ship type */
+#define CP_TRACTOR	24	/* tractor on/off */
+#define CP_REPRESS	25	/* pressor on/off */
+#define CP_COUP		26	/* coup home planet */
+#define CP_SOCKET	27	/* new socket for reconnection */
+#define CP_OPTIONS	28	/* send my options to be saved */
+#define CP_BYE		29	/* I'm done! */
+#define CP_DOCKPERM	30	/* set docking permissions */
+#define CP_UPDATES	31	/* set number of usecs per update */
+#define CP_RESETSTATS	32	/* reset my stats packet */
+#define CP_RESERVED	33	/* for future use */
+
+#define CP_SCAN		34	/* ATM: request for player scan */
+
+#define CP_UDP_REQ	35	/* request UDP on/off */
+#define CP_SEQUENCE	36	/* sequence # packet */
+#define CP_RSA_KEY	37	/* request MOTD */
+#define CP_ASK_MOTD	38	/* request MOTD */
+
+#define CP_PING_RESPONSE 42	/* client response */
+
+#ifdef SHORT_PACKETS
+#define CP_S_REQ                43
+#define CP_S_THRS               44
+#define CP_S_MESSAGE    45	/* vari. Message Packet */
+#define CP_S_RESERVED       46
+#define CP_S_DUMMY      47
+#endif
+
+#ifdef FEATURE
+#define CP_FEATURE	60	/* Feature packets */
+#endif
+
+#define SOCKVERSION 	4
+#define UDPVERSION	10	/* changing this blocks other */
+/* versions */
+
+struct packet_handler
+{
+  void (*handler) ();
+};
+
+
+/*
+ * These are server --> client packets
+ */
+
+struct mesg_spacket
+{
+  INT8 type;			/* SP_MESSAGE */
+  CARD8 m_flags;
+  CARD8 m_recpt;
+  CARD8 m_from;
+  char mesg[80];
+};
+
+struct plyr_info_spacket
+{
+  INT8 type;			/* SP_PLAYER_INFO */
+  INT8 pnum;
+  INT8 shiptype;
+  INT8 team;
+};
+
+struct kills_spacket
+{
+  INT8 type;			/* SP_KILLS */
+  INT8 pnum;
+  INT8 pad1;
+  INT8 pad2;
+  CARD32 kills;			/* where 1234=12.34 kills and 0=0.00 kills */
+};
+
+struct player_spacket
+{
+  INT8 type;			/* SP_PLAYER */
+  INT8 pnum;
+  CARD8 dir;
+  INT8 speed;
+  INT32 x, y;
+};
+
+struct torp_info_spacket
+{
+  INT8 type;			/* SP_TORP_INFO */
+  INT8 war;
+  INT8 status;			/* TFREE, TDET, etc... */
+  INT8 pad1;			/* pad needed for cross cpu compatibility */
+  INT16 tnum;
+  INT16 pad2;
+};
+
+struct torp_spacket
+{
+  INT8 type;			/* SP_TORP */
+  CARD8 dir;
+  INT16 tnum;
+  INT32 x, y;
+};
+
+/*
+ * Shapes of thingys.  It would be best to add to the end of this list and
+ * try to coordinate your additions with other hackers.
+ */
+enum thingy_types
+{
+  SHP_BLANK, SHP_MISSILE, SHP_BOOM, SHP_TORP, SHP_PLASMA, SHP_MINE,
+  SHP_PBOOM, SHP_FIGHTER, SHP_WARP_BEACON
+};
+
+struct thingy_info_spacket
+{
+  INT8 type;			/* SP_THINGY_INFO */
+  INT8 war;
+  INT16 shape;			/* a thingy_types */
+  INT16 tnum;
+  INT16 owner;
+};
+
+struct thingy_spacket
+{
+  INT8 type;			/* SP_THINGY */
+  CARD8 dir;
+  INT16 tnum;
+  INT32 x, y;
+};
+
+struct phaser_spacket
+{
+  INT8 type;			/* SP_PHASER */
+  INT8 pnum;
+  INT8 status;			/* PH_HIT, etc... */
+  CARD8 dir;
+  INT32 x, y;
+  INT32 target;
+};
+
+struct plasma_info_spacket
+{
+  INT8 type;			/* SP_PLASMA_INFO */
+  INT8 war;
+  INT8 status;			/* TFREE, TDET, etc... */
+  INT8 pad1;			/* pad needed for cross cpu compatibility */
+  INT16 pnum;
+  INT16 pad2;
+};
+
+struct plasma_spacket
+{
+  INT8 type;			/* SP_PLASMA */
+  INT8 pad1;
+  INT16 pnum;
+  INT32 x, y;
+};
+
+struct warning_spacket
+{
+  INT8 type;			/* SP_WARNING */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+  char mesg[80];
+};
+
+struct motd_spacket
+{
+  INT8 type;			/* SP_MOTD */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+  char line[80];
+};
+
+struct you_spacket
+{
+  INT8 type;			/* SP_YOU */
+  INT8 pnum;			/* Guy needs to know this... */
+  INT8 hostile;
+  INT8 swar;
+  INT8 armies;
+  INT8 tractor;			/* ATM - visible tractor (was pad1) */
+  CARD8 pad2;
+  CARD8 pad3;
+  CARD32 flags;
+  INT32 damage;
+  INT32 shield;
+  INT32 fuel;
+  INT16 etemp;
+  INT16 wtemp;
+  INT16 whydead;
+  INT16 whodead;
+};
+
+struct queue_spacket
+{
+  INT8 type;			/* SP_QUEUE */
+  INT8 pad1;
+  INT16 pos;
+};
+
+struct status_spacket
+{
+  INT8 type;			/* SP_STATUS */
+  INT8 tourn;
+  INT8 pad1;
+  INT8 pad2;
+  CARD32 armsbomb;
+  CARD32 planets;
+  CARD32 kills;
+  CARD32 losses;
+  CARD32 time;
+  CARD32 timeprod;
+};
+
+struct planet_spacket
+{
+  INT8 type;			/* SP_PLANET */
+  INT8 pnum;
+  INT8 owner;
+  INT8 info;
+  INT16 flags;
+  INT16 pad2;
+  INT32 armies;
+};
+
+struct pickok_spacket
+{
+  INT8 type;			/* SP_PICKOK */
+  INT8 state;
+  INT8 pad2;
+  INT8 pad3;
+};
+
+struct login_spacket
+{
+  INT8 type;			/* SP_LOGIN */
+  INT8 accept;			/* 1/0 */
+  INT8 pad2;
+  INT8 pad3;
+  INT32 flags;
+  char keymap[96];
+};
+
+struct flags_spacket
+{
+  INT8 type;			/* SP_FLAGS */
+  INT8 pnum;			/* whose flags are they? */
+  INT8 tractor;			/* ATM - visible tractors */
+  INT8 pad2;
+  CARD32 flags;
+};
+
+struct mask_spacket
+{
+  INT8 type;			/* SP_MASK */
+  INT8 mask;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct pstatus_spacket
+{
+  INT8 type;			/* SP_PSTATUS */
+  INT8 pnum;
+  INT8 status;
+  INT8 pad1;
+};
+
+struct badversion_spacket
+{
+  INT8 type;			/* SP_BADVERSION */
+  INT8 why;
+  INT8 pad2;
+  INT8 pad3;
+};
+
+struct hostile_spacket
+{
+  INT8 type;			/* SP_HOSTILE */
+  INT8 pnum;
+  INT8 war;
+  INT8 hostile;
+};
+
+struct stats_spacket
+{
+  INT8 type;			/* SP_STATS */
+  INT8 pnum;
+  INT8 pad1;
+  INT8 pad2;
+  INT32 tkills;			/* Tournament kills */
+  INT32 tlosses;		/* Tournament losses */
+  INT32 kills;			/* overall */
+  INT32 losses;			/* overall */
+  INT32 tticks;			/* ticks of tournament play time */
+  INT32 tplanets;		/* Tournament planets */
+  INT32 tarmies;		/* Tournament armies */
+  INT32 sbkills;		/* Starbase kills */
+  INT32 sblosses;		/* Starbase losses */
+  INT32 armies;			/* non-tourn armies */
+  INT32 planets;		/* non-tourn planets */
+  INT32 maxkills;		/* max kills as player * 100 */
+  INT32 sbmaxkills;		/* max kills as sb * 100 */
+};
+
+struct plyr_login_spacket
+{
+  INT8 type;			/* SP_PL_LOGIN */
+  INT8 pnum;
+  INT8 rank;
+  INT8 pad1;
+  INT8 name[16];
+  INT8 monitor[16];
+  INT8 login[16];
+};
+
+struct reserved_spacket
+{
+  INT8 type;			/* SP_RESERVED */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+  INT8 data[16];
+};
+
+struct planet_loc_spacket
+{
+  INT8 type;			/* SP_PLANET_LOC */
+  INT8 pnum;
+  INT8 pad2;
+  INT8 pad3;
+  INT32 x;
+  INT32 y;
+  INT8 name[16];
+};
+
+struct scan_spacket
+{				/* ATM */
+  INT8 type;			/* SP_SCAN */
+  INT8 pnum;
+  INT8 success;
+  INT8 pad1;
+  INT32 p_fuel;
+  INT32 p_armies;
+  INT32 p_shield;
+  INT32 p_damage;
+  INT32 p_etemp;
+  INT32 p_wtemp;
+};
+
+struct udp_reply_spacket
+{				/* UDP */
+  INT8 type;			/* SP_UDP_REPLY */
+  INT8 reply;
+  INT8 pad1;
+  INT8 pad2;
+  INT32 port;
+};
+
+struct sequence_spacket
+{				/* UDP */
+  INT8 type;			/* SP_SEQUENCE */
+  INT8 pad1;
+  CARD16 sequence;
+};
+struct sc_sequence_spacket
+{				/* UDP */
+  INT8 type;			/* SP_CP_SEQUENCE */
+  INT8 pad1;
+  CARD16 sequence;
+};
+
+/*
+ * Game configuration. KAO 1/23/93
+ */
+
+struct ship_cap_spacket
+{				/* Server configuration of client */
+  INT8 type;			/* screw motd method */
+  INT8 operation;		/* 0 = add/change a ship, 1 = remove a ship */
+  INT16 s_type;			/* SP_SHIP_CAP */
+  INT16 s_torpspeed;
+  INT16 s_phaserrange;
+  INT32 s_maxspeed;
+  INT32 s_maxfuel;
+  INT32 s_maxshield;
+  INT32 s_maxdamage;
+  INT32 s_maxwpntemp;
+  INT32 s_maxegntemp;
+  INT16 s_width;
+  INT16 s_height;
+  INT16 s_maxarmies;
+  INT8 s_letter;
+  INT8 pad2;
+  INT8 s_name[16];
+  INT8 s_desig1;
+  INT8 s_desig2;
+  INT16 s_bitmap;
+};
+
+struct motd_pic_spacket
+{
+  INT8 type;			/* SP_MOTD_PIC */
+  INT8 pad1;
+  INT16 x, y, page;
+  INT16 width, height;
+  INT8 bits[1016];
+};
+
+
+/* This is used to send paradise style stats */
+struct stats_spacket2
+{
+  INT8 type;			/* SP_STATS2 */
+  INT8 pnum;
+  INT8 pad1;
+  INT8 pad2;
+
+  INT32 genocides;		/* number of genocides participated in */
+  INT32 maxkills;		/* max kills ever * 100  */
+  INT32 di;			/* destruction inflicted for all time * 100 */
+  INT32 kills;			/* Kills in tournament play */
+  INT32 losses;			/* Losses in tournament play */
+  INT32 armsbomb;		/* Tournament armies bombed */
+  INT32 resbomb;		/* resources bombed off */
+  INT32 dooshes;		/* armies killed while being carried */
+  INT32 planets;		/* Tournament planets conquered */
+  INT32 tticks;			/* Tournament ticks */
+  /* SB/WB/JS stats are entirely separate */
+  INT32 sbkills;		/* Kills as starbase */
+  INT32 sblosses;		/* Losses as starbase */
+  INT32 sbticks;		/* Time as starbase */
+  INT32 sbmaxkills;		/* Max kills as starbase * 100 */
+  INT32 wbkills;		/* Kills as warbase */
+  INT32 wblosses;		/* Losses as warbase */
+  INT32 wbticks;		/* Time as warbase */
+  INT32 wbmaxkills;		/* Max kills as warbase * 100 */
+  INT32 jsplanets;		/* planets assisted with in JS */
+  INT32 jsticks;		/* ticks played as a JS */
+  INT32 rank;			/* Ranking of the player */
+  INT32 royal;			/* royaly, specialty, rank */
+};
+
+/* status info for paradise stats */
+struct status_spacket2
+{
+  INT8 type;			/* SP_STATUS2 */
+  INT8 tourn;
+  INT8 pad1;
+  INT8 pad2;
+  CARD32 dooshes;		/* total number of armies dooshed */
+  CARD32 armsbomb;		/* all t-mode armies bombed */
+  CARD32 resbomb;		/* resources bombed */
+  CARD32 planets;		/* all t-mode planets taken */
+  CARD32 kills;			/* all t-mode kills made */
+  CARD32 losses;		/* all t-mode losses */
+  CARD32 sbkills;		/* total kills in SB's */
+  CARD32 sblosses;		/* total losses in Sb's */
+  CARD32 sbtime;		/* total time in SB's */
+  CARD32 wbkills;		/* kills in warbases */
+  CARD32 wblosses;		/* losses in warbases */
+  CARD32 wbtime;		/* total time played in wb's */
+  CARD32 jsplanets;		/* total planets taken by jump ships */
+  CARD32 jstime;		/* total time in a jump ship */
+  CARD32 time;			/* t mode time in this game */
+  CARD32 timeprod;		/* t-mode ship ticks--sort of like */
+};
+
+
+/* planet info for a paradise planet */
+struct planet_spacket2
+{
+  INT8 type;			/* SP_PLANET2 */
+  INT8 pnum;			/* planet number */
+  INT8 owner;			/* owner of the planet */
+  INT8 info;			/* who has touched planet */
+  INT32 flags;			/* planet's flags */
+  INT32 timestamp;		/* timestamp for info on planet */
+  INT32 armies;			/* armies on the planet */
+};
+
+/* terrain info for Paradise terrain */
+/* 5/16/95 rpg */
+
+struct terrain_info_packet2
+{
+  CARD8 type;			/* SP_TERRAIN_INFO2 */
+  CARD8 pad;
+  CARD16 pad2;
+  CARD16 xdim;			/* x dimension of grid */
+  CARD16 ydim;			/* y dimension of grid */
+};
+
+struct terrain_packet2
+{
+  CARD8 type;			/* SP_TERRAIN2 */
+  CARD8 sequence;		/* Sequence number - may need multiple
+				 * packets */
+  CARD8 total_pkts;		/* since terrain is gzipped, we need length
+				 * (may have) */
+  CARD8 length;			/* multiple packets */
+  CARD8 terrain_type[NTERRAIN];	/* send up to 128 gzipped bytes at once */
+  /* CARD16   terrain_alt1[128]; *//* Note - these values should be *changed* */
+  /* CARD16   terrain_atl2[128]; *//* depending on if alt1/alt2 are used. */
+};
+
+struct obvious_packet
+{
+  INT8 type;			/* SP_NEW_MOTD */
+  INT8 pad1;			/* CP_ASK_MOTD */
+};
+
+struct rsa_key_spacket
+{
+  INT8 type;			/* SP_RSA_KEY */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+  CARD8 data[KEY_SIZE];
+};
+
+
+struct ping_spacket
+{
+  INT8 type;			/* SP_PING */
+  CARD8 number;			/* id (ok to wrap) */
+  CARD16 lag;			/* delay of last ping in ms */
+
+  CARD8 tloss_sc;		/* total loss server-client 0-100% */
+  CARD8 tloss_cs;		/* total loss client-server 0-100% */
+
+  CARD8 iloss_sc;		/* inc. loss server-client 0-100% */
+  CARD8 iloss_cs;		/* inc. loss client-server 0-100% */
+};
+
+struct paradiseext1_spacket
+{
+  INT8 type;
+  CARD8 subtype;
+  INT16 pad;
+};
+
+struct pe1_missing_bitmap_spacket
+{
+  INT8 type;
+  CARD8 subtype;
+
+  INT16 page;
+
+  INT16 x, y;
+  INT16 width, height;
+};
+
+struct pe1_num_missiles_spacket
+{
+  INT8 type;			/* SP_PARADISE_EXT1 */
+  CARD8 subtype;		/* SP_PE1_NUM_MISSILES */
+
+  INT16 num;			/* number of missiles */
+};
+
+/*
+ * These are the client --> server packets
+ */
+
+struct mesg_cpacket
+{
+  INT8 type;			/* CP_MESSAGE */
+  INT8 group;
+  INT8 indiv;
+  INT8 pad1;
+  INT8 mesg[80];
+};
+
+struct speed_cpacket
+{
+  INT8 type;			/* CP_SPEED */
+  INT8 speed;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct dir_cpacket
+{
+  INT8 type;			/* CP_DIRECTION */
+  CARD8 dir;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct phaser_cpacket
+{
+  INT8 type;			/* CP_PHASER */
+  CARD8 dir;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct plasma_cpacket
+{
+  INT8 type;			/* CP_PLASMA */
+  CARD8 dir;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct torp_cpacket
+{
+  INT8 type;			/* CP_TORP */
+  CARD8 dir;			/* direction to fire torp */
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct quit_cpacket
+{
+  INT8 type;			/* CP_QUIT */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+};
+
+struct login_cpacket
+{
+  INT8 type;			/* CP_LOGIN */
+  INT8 query;
+  INT8 pad2;
+  INT8 pad3;
+  INT8 name[16];
+  INT8 password[16];
+  INT8 login[16];
+};
+
+struct outfit_cpacket
+{
+  INT8 type;			/* CP_OUTFIT */
+  INT8 team;
+  INT8 ship;
+  INT8 pad1;
+};
+
+struct war_cpacket
+{
+  INT8 type;			/* CP_WAR */
+  INT8 newmask;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct practr_cpacket
+{
+  INT8 type;			/* CP_PRACTR */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+};
+
+struct shield_cpacket
+{
+  INT8 type;			/* CP_SHIELD */
+  INT8 state;			/* up/down */
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct repair_cpacket
+{
+  INT8 type;			/* CP_REPAIR */
+  INT8 state;			/* on/off */
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct orbit_cpacket
+{
+  INT8 type;			/* CP_ORBIT */
+  INT8 state;			/* on/off */
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct planlock_cpacket
+{
+  INT8 type;			/* CP_PLANLOCK */
+  INT8 pnum;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct playlock_cpacket
+{
+  INT8 type;			/* CP_PLAYLOCK */
+  INT8 pnum;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct bomb_cpacket
+{
+  INT8 type;			/* CP_BOMB */
+  INT8 state;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct beam_cpacket
+{
+  INT8 type;			/* CP_BEAM */
+  INT8 state;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct cloak_cpacket
+{
+  INT8 type;			/* CP_CLOAK */
+  INT8 state;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct det_torps_cpacket
+{
+  INT8 type;			/* CP_DET_TORPS */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+};
+
+struct det_mytorp_cpacket
+{
+  INT8 type;			/* CP_DET_MYTORP */
+  INT8 pad1;
+  INT16 tnum;
+};
+
+struct copilot_cpacket
+{
+  INT8 type;			/* CP_COPLIOT */
+  INT8 state;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct refit_cpacket
+{
+  INT8 type;			/* CP_REFIT */
+  INT8 ship;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct tractor_cpacket
+{
+  INT8 type;			/* CP_TRACTOR */
+  INT8 state;
+  INT8 pnum;
+  INT8 pad2;
+};
+
+struct repress_cpacket
+{
+  INT8 type;			/* CP_REPRESS */
+  INT8 state;
+  INT8 pnum;
+  INT8 pad2;
+};
+
+struct coup_cpacket
+{
+  INT8 type;			/* CP_COUP */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+};
+
+struct socket_cpacket
+{
+  INT8 type;			/* CP_SOCKET */
+  INT8 version;
+  INT8 udp_version;		/* was pad2 */
+  INT8 pad3;
+  CARD32 socket;
+};
+
+struct options_cpacket
+{
+  INT8 type;			/* CP_OPTIONS */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+  CARD32 flags;
+  INT8 keymap[96];
+};
+
+struct bye_cpacket
+{
+  INT8 type;			/* CP_BYE */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+};
+
+struct dockperm_cpacket
+{
+  INT8 type;			/* CP_DOCKPERM */
+  INT8 state;
+  INT8 pad2;
+  INT8 pad3;
+};
+
+struct updates_cpacket
+{
+  INT8 type;			/* CP_UPDATES */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+  CARD32 usecs;
+};
+
+struct resetstats_cpacket
+{
+  INT8 type;			/* CP_RESETSTATS */
+  INT8 verify;			/* 'Y' - just to make sure he meant it */
+  INT8 pad2;
+  INT8 pad3;
+};
+
+struct reserved_cpacket
+{
+  INT8 type;			/* CP_RESERVED */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+  INT8 data[16];
+  INT8 resp[16];
+};
+
+struct scan_cpacket
+{				/* ATM */
+  INT8 type;			/* CP_SCAN */
+  INT8 pnum;
+  INT8 pad1;
+  INT8 pad2;
+};
+
+struct udp_req_cpacket
+{				/* UDP */
+  INT8 type;			/* CP_UDP_REQ */
+  INT8 request;
+  INT8 connmode;		/* respond with port # or just send UDP
+				 * packet? */
+  INT8 pad2;
+  INT32 port;			/* compensate for hosed recvfrom() */
+};
+
+struct sequence_cpacket
+{				/* UDP */
+  INT8 type;			/* CP_SEQUENCE */
+  INT8 pad1;
+  CARD16 sequence;
+};
+
+struct rsa_key_cpacket
+{
+  INT8 type;			/* CP_RSA_KEY */
+  INT8 pad1;
+  INT8 pad2;
+  INT8 pad3;
+  CARD8 global[KEY_SIZE];
+  CARD8 public[KEY_SIZE];
+  CARD8 resp[KEY_SIZE];
+};
+
+/* the CP_ASK_MOTD packet is the same as temp_spacket */
+
+struct ping_cpacket
+{
+  INT8 type;			/* CP_PING_RESPONSE */
+  CARD8 number;			/* id */
+  INT8 pingme;			/* if client wants server to ping */
+  INT8 pad1;
+
+  INT32 cp_sent;		/* # packets sent to server */
+  INT32 cp_recv;		/* # packets recv from server */
+};
+
+#ifdef FEATURE
+struct feature_cpacket
+{
+  char type;
+  char feature_type;
+  char arg1, arg2;
+  int value;
+  char name[80];
+};
+
+struct feature_spacket
+{
+  char type;
+  char feature_type;
+  char arg1, arg2;
+  int value;
+  char name[80];
+};
+#endif
+
+/*
+ * short stuff
+ */
+
+#ifdef SHORT_PACKETS
+struct shortreq_cpacket
+{				/* CP_S_REQ */
+  INT8 type;
+  INT8 req;
+  INT8 version;
+  INT8 pad2;
+};
+
+struct threshold_cpacket
+{				/* CP_S_THRS */
+  INT8 type;
+  INT8 pad1;
+  CARD16 thresh;
+};
+
+struct shortreply_spacket
+{				/* SP_S_REPLY */
+  INT8 type;
+  INT8 repl;
+  CARD16 winside;
+  INT32 gwidth;
+};
+
+struct youshort_spacket
+{				/* SP_S_YOU */
+  INT8 type;
+
+  INT8 pnum;
+  INT8 hostile;
+  INT8 swar;
+
+  INT8 armies;
+  INT8 whydead;
+  INT8 whodead;
+
+  INT8 pad1;
+
+  CARD32 flags;
+};
+
+struct youss_spacket
+{				/* SP_S_YOU_SS */
+  INT8 type;
+  INT8 pad1;
+
+  CARD16 damage;
+  CARD16 shield;
+  CARD16 fuel;
+  CARD16 etemp;
+  CARD16 wtemp;
+};
+
+#define VPLANET_SIZE 6
+
+struct planet_s_spacket
+{				/* body of SP_S_PLANET  */
+  INT8 pnum;
+  INT8 owner;
+  INT8 info;
+  CARD8 armies;			/* more than 255 Armies ? ...  */
+  INT16 flags;
+};
+struct warning_s_spacket
+{				/* SP_S_WARNING */
+  INT8 type;
+  CARD8 whichmessage;
+  INT8 argument, argument2;	/* for phaser  etc ... */
+};
+
+struct player_s_spacket
+{
+  INT8 type;			/* SP_S_PLAYER Header */
+  INT8 packets;			/* How many player-packets are in this packet
+				 * ( only the first 6 bits are relevant ) */
+  CARD8 dir;
+  INT8 speed;
+  INT32 x, y;			/* To get the absolute Position */
+};
+
+/*
+ * The format of the body: struct player_s_body_spacket {	Body of new
+ * Player Packet CARD8 pnum;	 0-4 = pnum, 5 local or galactic, 6 = 9.
+ * x-bit, 7 9. y-bit CARD8 speeddir;	 0-3 = speed , 4-7 direction of ship
+ * CARD8 x;	 low 8 bits from X-Pixelcoordinate CARD8 y;	 low 8 bits
+ * from Y-Pixelcoordinate };
+ */
+
+struct torp_s_spacket
+{
+  INT8 type;			/* SP_S_TORP */
+  CARD8 bitset;			/* bit=1 that torp is in packet */
+  CARD8 whichtorps;		/* Torpnumber of first torp / 8 */
+  CARD8 data[21];		/* For every torp 2*9 bit coordinates */
+};
+
+struct mesg_s_spacket
+{
+  INT8 type;			/* SP_S_MESSAGE */
+  CARD8 m_flags;
+  CARD8 m_recpt;
+  CARD8 m_from;
+  CARD8 length;			/* Length of whole packet */
+  INT8 mesg;
+  INT8 pad2;
+  INT8 pad3;
+  INT8 pad[76];
+};
+
+struct mesg_s_cpacket
+{
+  INT8 type;			/* CP_S__MESSAGE */
+  INT8 group;
+  INT8 indiv;
+  INT8 length;			/* Size of whole packet   */
+  INT8 mesg[80];
+};
+
+#endif
+
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/parsexbm.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,159 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#define pm_error(str) {\
+fprintf(stderr, "Error in ParseXbmFile(): %s\n", str);\
+if(*dataP) { free(*dataP); *dataP = 0; }\
+return -1; }
+
+/*
+ * most of the following function came from xbmtopbm.c in the PBMPLUS library
+ */
+
+#define MAX_LINE 500
+
+int
+ParseXbmFile(stream, widthP, heightP, dataP)
+  FILE *stream;
+  int *widthP;
+  int *heightP;
+  char **dataP;
+{
+  char line[MAX_LINE], name_and_type[MAX_LINE];
+  char *ptr;
+  char *t;
+  int raster_length, v;
+  register int bytes, bytes_per_line;
+  register int c1, c2, value1, value2;
+  int hex_table[256];
+
+  *widthP = *heightP = -1;
+  *dataP = 0;
+
+  for (;;)
+  {
+    if (fgets(line, MAX_LINE, stream) == NULL)
+      pm_error("EOF / read error");
+    if (strlen(line) == MAX_LINE - 1)
+      pm_error("line too long");
+
+    if (sscanf(line, "#define %s %d", name_and_type, &v) == 2)
+    {
+      if ((t = strrchr(name_and_type, '_')) == NULL)
+	t = name_and_type;
+      else
+	++t;
+      if (!strcmp("width", t))
+	*widthP = v;
+      else if (!strcmp("height", t))
+	*heightP = v;
+      continue;
+    }
+
+    if (sscanf(line, "static char %s = {", name_and_type) == 1 ||
+	sscanf(line, "static unsigned char %s = {", name_and_type) == 1)
+      break;
+  }
+
+  if (*widthP == -1)
+    pm_error("invalid width");
+  if (*heightP == -1)
+    pm_error("invalid height");
+
+  bytes_per_line = (*widthP + 7) / 8;
+
+  raster_length = bytes_per_line * *heightP;
+  *dataP = (char *) malloc(raster_length);
+  if (*dataP == (char *) 0)
+    pm_error("out of memory");
+
+  /* Initialize hex_table. */
+  for (c1 = 0; c1 < 256; ++c1)
+    hex_table[c1] = 256;
+
+  hex_table['0'] = 0;
+  hex_table['1'] = 1;
+  hex_table['2'] = 2;
+  hex_table['3'] = 3;
+  hex_table['4'] = 4;
+  hex_table['5'] = 5;
+  hex_table['6'] = 6;
+  hex_table['7'] = 7;
+  hex_table['8'] = 8;
+  hex_table['9'] = 9;
+  hex_table['A'] = 10;
+  hex_table['B'] = 11;
+  hex_table['C'] = 12;
+  hex_table['D'] = 13;
+  hex_table['E'] = 14;
+  hex_table['F'] = 15;
+  hex_table['a'] = 10;
+  hex_table['b'] = 11;
+  hex_table['c'] = 12;
+  hex_table['d'] = 13;
+  hex_table['e'] = 14;
+  hex_table['f'] = 15;
+
+  for (bytes = 0, ptr = *dataP; bytes < raster_length; ++bytes)
+  {
+    /*
+     * Skip until digit is found.
+     */
+    for (;;)
+    {
+      c1 = getc(stream);
+      if (c1 == EOF)
+	pm_error("EOF / read error");
+      value1 = hex_table[c1];
+      if (value1 != 256)
+	break;
+    }
+    /*
+     * Loop on digits.
+     */
+    for (;;)
+    {
+      c2 = getc(stream);
+      if (c2 == EOF)
+	pm_error("EOF / read error");
+      value2 = hex_table[c2];
+      if (value2 != 256)
+      {
+	value1 = (value1 << 4) | value2;
+	if (value1 >= 256)
+	  pm_error("syntax error");
+      }
+      else if (c2 == 'x' || c2 == 'X')
+      {
+	if (value1 == 0)
+	  continue;
+	else
+	  pm_error("syntax error");
+      }
+      else
+	break;
+    }
+    *ptr++ = (char) (value1 ^ 0xFF);
+  }
+  return (raster_length);
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/path.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,77 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+
+/*
+ * This is the default path, you don't have to change this as long as you set
+ * the NETREKDIR env
+ */
+char netrekpath[256] = "/local/lib/paradise";
+
+
+static char buf[512];
+static int initted = 0;
+
+char *
+build_path(suffix)
+  char *suffix;
+{
+  int len;
+  if (!initted)
+  {
+    char *ptr;
+    initted = 1;
+    ptr = (char *) getenv("NETREKDIR");
+    if (ptr == NULL)
+    {
+      fprintf(stderr, "Warning, no NETREKDIR envariable.  Using default of %s\n", netrekpath);
+    }
+    else
+    {
+      strncpy(netrekpath, ptr, 255);
+      netrekpath[255] = '\0';
+    }
+    len = strlen(netrekpath);
+    if (len > 230)
+    {
+      fprintf(stderr, "NETREKDIR enviroment variable too long.\n");
+      fprintf(stderr, "Please change it.\n");
+      exit(1);
+    }
+    if (netrekpath[len - 1] != '/')
+    {
+      netrekpath[len] = '/';
+      netrekpath[len + 1] = '\0';
+    }
+  }
+  if (*suffix == '/')
+  {				/* absolute path... don't prepend anything */
+    strcpy(buf, suffix);
+  }
+  else
+  {
+    strcpy(buf, netrekpath);
+    strcat(buf, suffix);
+  }
+
+  return buf;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/path.h	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,19 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+char *build_path();
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/phaser.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,247 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <stdio.h>
+#include <math.h>
+#include <signal.h>
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "weapons.h"
+#include "shmem.h"
+
+/*-----------------------------VISIBLE FUNCTIONS---------------------------*/
+
+/*----------------------------------PHASER---------------------------------*/
+/*
+ * This function shoots a phaser for a player.  Various conditions are
+ * checked to see if the phaser should be allowed to fire.  If the player can
+ * fire then, a player is found in the direction the phaser was fired.
+ */
+
+
+extern int angdist();
+
+void
+phaser(course)
+  unsigned char course;
+{
+  int i;			/* looping var */
+  struct player *j, *target = 0;/* to hold player found */
+  struct plasmatorp *k, *target2 = 0;	/* to hold plasma torp found */
+  struct missile *d, *target3 = 0;
+  struct phaser *mine;		/* to pnt to player's phaser */
+  int whichtarget;		/* to hold type of target */
+  int target_x = 0, target_y = 0;	/* target's x and y coords */
+  unsigned char dir;		/* to get direction of phasr */
+  int range, trange;		/* range of target */
+  int maxangle;			/* potential target range */
+  int myphrange;		/* angle to hit potentl targ */
+  char buf[80];			/* to sprintf warnings into */
+
+  mine = &phasers[me->p_no];	/* get phaser struct */
+  if (!(myship->s_nflags & SFNHASPHASERS))
+  {				/* do we have phasers? */
+    warning("Weapons Officer:  This ship is not armed with phasers, captain!");
+    return;
+  }
+  if (mine->ph_status != PHFREE)
+  {				/* is phaser currently being */
+    warning("Phasers have not recharged");	/* fired */
+    return;
+  }
+  if (me->p_fuel < myship->s_phaser.cost)
+  {				/* do we have enough fuel */
+    warning("Not enough fuel for phaser");
+    return;
+  }
+  if (me->p_flags & PFREPAIR)
+  {				/* cannot fire while in */
+    warning("Can't fire while repairing");	/* repair mode */
+    return;
+  }
+  if (me->p_flags & PFWEP)
+  {				/* cannot fire while weapon */
+    warning("Weapons overheated");	/* temped */
+    return;
+  }
+  if ((me->p_cloakphase) && (me->p_ship.s_type != ATT))
+  {
+    warning("Cannot fire while cloaked");	/* cannot fire while cloaked */
+    return;
+  }
+  if (!check_fire_warp()
+      || !check_fire_warpprep()
+      || !check_fire_docked())
+    return;
+
+  me->p_fuel -= myship->s_phaser.cost;	/* subtract off fuel cost */
+  me->p_wtemp += myship->s_phaser.wtemp;	/* add to w temp */
+  mine->ph_dir = course;	/* get direction of phaser */
+  whichtarget = 0;		/* no target fount yet */
+  range = 1000000;		/* Sufficiently big. */
+  /* check the players */
+  for (i = 0, j = &players[i]; i < MAXPLAYER; i++, j++)
+  {				/* loop all players */
+    if (((j->p_status != PALIVE)
+#ifdef PFBIRD
+	 && !(j->p_flags & PFBIRD)
+#endif
+	 ) || (j == me))	/* only check alive players */
+      continue;
+    if ((!((j->p_swar | j->p_hostile) & me->p_team)) &&
+	(!((me->p_swar | me->p_hostile) & j->p_team)) &&
+#ifdef PFBIRD
+	!(j->p_flags & PFBIRD)
+#endif
+      )
+      continue;			/* only check at war with */
+    dir = (unsigned char) (int) (atan2((double) (j->p_x - me->p_x),
+			     (double) (me->p_y - j->p_y)) / 3.14159 * 128.);
+    /* get range of target */
+    trange = ihypot(j->p_x - me->p_x, j->p_y - me->p_y);
+    if (trange == 0)		/* don't want zero in atan */
+      trange = 1;
+    maxangle = atan((float) EXPDIST / trange) / 3.14159 * 128.0;
+    if (angdist(dir, course) <= maxangle)
+    {				/* if angle within tolerance */
+      if (range > trange)
+      {				/* then check to see if */
+	whichtarget = 1;	/* this is the closest target */
+	target = j;		/* found yet */
+	range = trange;		/* record if it is */
+      }
+    }
+  }				/* check the plasmas */
+  for (i = 0, k = &plasmatorps[i]; i < MAXPLASMA * MAXPLAYER; i++, k++)
+  {
+    if ((k->pt_status != PTMOVE) || (k->pt_owner == me->p_no))
+      continue;			/* only check live plasmas */
+    if ((!(k->pt_war & me->p_team)) &&	/* and unfriendly ones */
+	(!((me->p_swar | me->p_hostile) & k->pt_team)))
+      continue;
+    dir = (unsigned char) (int) (atan2((double) (k->pt_x - me->p_x),
+			    (double) (me->p_y - k->pt_y)) / 3.14159 * 128.);	/* find direction */
+    trange = ihypot(k->pt_x - me->p_x, k->pt_y - me->p_y);
+    if (trange == 0)		/* no zeroes in math funcs */
+      trange = 1;
+    maxangle = atan((float) EXPDIST / 4 / trange) / 3.14159 * 128.0;
+    if (angdist(dir, course) <= (maxangle + 1))
+    {				/* if we can hit it */
+      if (range > trange)
+      {				/* then check to see if this */
+	target_x = k->pt_x;	/* is the closest plasma */
+	target_y = k->pt_y;	/* found yet */
+	whichtarget = 2;	/* and record it if it is */
+	target2 = k;
+	range = trange;
+      }
+    }
+  }
+  /* check the fighters */
+  for (i = 0, d = &missiles[i]; i < NPTHINGIES * MAXPLAYER; i++, d++)
+  {
+    if ((d->ms_owner == me->p_no) || (d->ms_type != FIGHTERTHINGY))
+      continue;			/* only check live fighters */
+    if ((!(d->ms_war & me->p_team)) &&	/* and unfriendly ones */
+	(!((me->p_swar | me->p_hostile) & d->ms_team)))
+      continue;
+    dir = (unsigned char) (atan2((double) (d->ms_x - me->p_x),
+				 (double) (me->p_y - d->ms_y))
+			   / 3.14159 * 128.);	/* find direction */
+    trange = ihypot(d->ms_x - me->p_x, d->ms_y - me->p_y);
+    if (trange == 0)		/* no zeroes in math funcs */
+      trange = 1;
+    maxangle = atan((float) EXPDIST / 8 / trange) / 3.14159 * 128.0;
+    if (angdist(dir, course) <= (maxangle + 1))
+    {				/* if we can hit it */
+      if (range > trange)
+      {				/* then check to see if this */
+	target_x = d->ms_x;	/* is the closest fighter */
+	target_y = d->ms_y;	/* found yet */
+	whichtarget = 3;	/* and record it if it is */
+	target3 = d;
+	range = trange;
+      }
+    }
+  }
+
+  mine->ph_fuse = me->p_ship.s_phaser.fuse;	/* set phaser fuse */
+  myphrange = me->p_ship.s_phaser.speed;	/* phaser range */
+  if ((whichtarget == 0) ||	/* if no target found or all */
+      (range > myphrange))
+  {				/* targets too long */
+    mine->ph_status = PHMISS;	/* then we missed */
+    warning("Phaser missed!!!");
+  }
+  else if (whichtarget == 2)
+  {				/* if we hit a plasma then */
+    warning("You destroyed the plasma torpedo!");
+    mine->ph_x = target_x;	/* the set point to shoot at */
+    mine->ph_y = target_y;
+    mine->ph_status = PHHIT2;	/* we hit a plasma */
+    target2->pt_status = PTEXPLODE;	/* Plasmas hurt everyone */
+    target2->pt_whodet = me->p_no;
+  }
+  else if (whichtarget == 3)
+  {				/* if we hit a fighter then */
+    warning("You shot a fighter!");
+    mine->ph_x = target_x;	/* the set point to shoot at */
+    mine->ph_y = target_y;
+    mine->ph_status = PHHIT2;	/* we hit the fighter */
+    target3->ms_status = TDET;	/* det the fighter */
+    target3->ms_whodet = me->p_no;
+  }
+  else
+  {				/* else if we hit player */
+    mine->ph_target = target->p_no;	/* set the player number */
+    mine->ph_damage = me->p_ship.s_phaser.damage *	/* get damage */
+      (1.0 - (range / (float) myphrange));
+    if (mine->ph_damage < 0)	/* if damage inflicted */
+      mine->ph_damage = -mine->ph_damage;	/* set the phaser damage */
+    mine->ph_status = PHHIT;	/* status is a hit */
+
+#ifdef PFBIRD
+    if (target->p_flags & PFBIRD)
+    {
+      /* change to PHHIT2 so phaser won't follow bird */
+      mine->ph_status = PHHIT2;
+      mine->ph_x = target->p_x;
+      mine->ph_y = target->p_y;
+      /* xx: slight misuse of fields here */
+      target->p_damage = mine->ph_damage;
+      target->p_whodead = me->p_no;
+
+      (void) sprintf(buf, "\"AAWWK!\"");
+    }
+#endif
+
+    (void) sprintf(buf, "Phaser burst hit %s for %d points",
+		   target->p_name, mine->ph_damage);
+    warning(buf);
+  }
+}
+
+/*------------------------------------------------------------------------*/
+
+
+
+
+
+/*-------END OF FILE-------*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/ping.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,374 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <stdio.h>
+#include <signal.h>
+#include <sys/types.h>
+#include <sys/time.h>
+#include <math.h>
+#include <errno.h>
+#include <netinet/in.h>
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "packets.h"
+#include "shmem.h"
+
+#define PING_DEBUG				0	/* debugging */
+
+/* special fields in stats record for rountrip delay and packet loss */
+#define INL_STATS
+
+
+static unsigned char ping_id;	/* wraparound expected */
+static int ping_lag;		/* ping roundtrip delay */
+
+static int tloss_sc,		/* total packet loss s-c */
+    tloss_cs,			/* total packet loss c-s */
+    iloss_sc,			/* inc. packet loss s-c */
+    iloss_cs;			/* inc. packet loss c-s */
+
+/*
+ * This structure allows us to send several pings before any response is
+ * received without losing information -- as would be the case for roundtrip
+ * times equal to or larger then the ping interval times. HASHSIZE * ping
+ * interval must be greater then the largest expected roundtrip time.
+ */
+
+#define HASHSIZE		32
+#define PITH(i)			(((int)i) & (HASHSIZE-1))
+
+static
+struct
+{
+  long time;
+  long packets_sent_at_ping;
+}   ping_sent[HASHSIZE];
+
+
+void calc_loss();
+void update_lag_stats();
+void update_loss_stats();
+int mstime();
+int msetime();
+
+/*
+ * response from client
+ */
+
+void
+pingResponse(packet)
+  struct ping_cpacket *packet;
+{
+  register i;
+  static int last_num;
+
+  if (!ping || packet->pingme != 1)
+    return;			/* oops, garbage */
+
+  ping_ghostbust = 0;		/* don't ghostbust, client is alive */
+
+  /* throw out out-of-order packets */
+  i = uchar_diff((int) packet->number, last_num);
+  if (i < 0)
+  {
+#if PING_DEBUG >= 1
+    fprintf(stderr, "out-of-order response ignored: %d (last: %d)\n",
+	    packet->number, last_num);
+    fflush(stderr);
+#endif
+    return;
+  }
+  last_num = packet->number;
+  i = PITH(last_num);
+
+  /* calculate roundtrip */
+  ping_lag = mstime() - ping_sent[i].time;
+
+#ifdef INL_STATS
+  /* fill in lag stats fields */
+  update_lag_stats();
+#endif
+
+  /* watch out for div by 0 */
+  if (!packets_received || !ping_sent[i].packets_sent_at_ping)
+    return;
+
+  /* calculate total packet loss */
+  calc_loss(i, packet);
+
+#ifdef INL_STATS
+  update_loss_stats();
+#endif
+}
+
+/*
+ * request from server
+ */
+
+void
+sendClientPing()
+{
+  struct ping_spacket packet;
+
+  ping_ghostbust++;
+  ping_id++;			/* ok to wrap */
+
+  packet.type = SP_PING;
+  packet.number = (unsigned char) ping_id;
+  packet.lag = htons((unsigned short) ping_lag);
+  packet.tloss_sc = tloss_sc;
+  packet.tloss_cs = tloss_cs;
+  packet.iloss_sc = iloss_sc;
+  packet.iloss_cs = iloss_cs;
+
+  ping_sent[PITH(ping_id)].time = mstime();
+  /*
+   * printf("ping sent at %d\n", msetime());
+   */
+
+  sendClientPacket(&packet);
+
+  ping_sent[PITH(ping_id)].packets_sent_at_ping = packets_sent;
+}
+
+void
+calc_loss(i, packet)
+  int i;
+  struct ping_cpacket *packet;
+{
+  /* tloss vars */
+  register cp_recv,		/* client packets recv */
+      cp_sent;			/* client packets sent */
+  int s_to_c_dropped,		/* server to client */
+      c_to_s_dropped;		/* client to server */
+  static int old_s_to_c_dropped,/* previous update */
+      old_c_to_s_dropped;	/* "" */
+  /* iloss vars */
+  int p_sent, p_recv;
+
+  static
+  int timer;
+
+  static int inc_packets_sent,	/* packets sent start-point */
+      inc_packets_received,	/* packets recvd start-pt  */
+      inc_s_to_c_dropped,	/* dropped s-to-c start-pt */
+      inc_c_to_s_dropped;	/* dropped c-to-s start-pt */
+
+  if (!timer)
+    timer = configvals->ping_iloss_interval;
+
+  cp_recv = ntohl(packet->cp_recv);
+  cp_sent = ntohl(packet->cp_sent);
+
+  /* at ping time, total packets dropped from server to client */
+  s_to_c_dropped = ping_sent[i].packets_sent_at_ping - cp_recv;
+
+  if (s_to_c_dropped < old_s_to_c_dropped)
+  {
+    /*
+     * The network may duplicate or send out-of-order packets. Both are
+     * detected and thrown out by the client if sequence checking is on. If
+     * not there's not much we can do -- there's no way to distinguish a
+     * duplicated packet from a series of out of order packets.  While the
+     * latter case cancels itself out eventually in terms of packet loss, the
+     * former hides real packet loss by adding extra packets. We'll have to
+     * kludge it by adding the extra packets the client thinks it got to
+     * packets_sent
+     */
+    packets_sent += old_s_to_c_dropped - s_to_c_dropped;
+    /* and adjust s_to_c_dropped so we don't get a negative packet loss */
+    s_to_c_dropped = old_s_to_c_dropped;
+  }
+
+  /* total loss server-to-client since start of connection */
+  tloss_sc = 100 -
+    (100 * (ping_sent[i].packets_sent_at_ping - s_to_c_dropped)) /
+    ping_sent[i].packets_sent_at_ping;
+
+  /*
+   * at ping time, total packets dropped from client to server NOTE: not
+   * packets_received_at_ping since the client may have sent any amount of
+   * packets between the time we sent the ping and the time the client
+   * received it.
+   */
+  c_to_s_dropped = cp_sent - packets_received;
+
+#if PING_DEBUG >= 2
+  printf("cp_sent: %d, packets_received: %d\n",
+	 cp_sent, packets_received);
+#endif
+
+  if (c_to_s_dropped < old_c_to_s_dropped)
+  {
+    /*
+     * The network may duplicate or send out-of-order packets. Since no
+     * sequence checking is done by the server, there's not much we can do --
+     * there's no way to distinguish a duplicated packet from a series of out
+     * of order packets.  While the latter case cancels itself out eventually
+     * in terms of packet loss, the former hides real packet loss by adding
+     * extra packets. We'll have to kludge it by subtracting the extra
+     * packets we think we got from the client from packets_received.
+     */
+    packets_received -= old_c_to_s_dropped - c_to_s_dropped;
+    /* and adjust c_to_s_dropped so we don't get a negative packet loss */
+    c_to_s_dropped = old_c_to_s_dropped;
+  }
+
+  /* total loss client-to-server since start of connection */
+  tloss_cs = 100 -
+    (100 * (packets_received - c_to_s_dropped)) / (packets_received ? packets_received : 1);
+
+  old_s_to_c_dropped = s_to_c_dropped;
+  old_c_to_s_dropped = c_to_s_dropped;
+
+  /* Incremental packet loss */
+
+  /* packets sent since last ping response */
+  p_sent = ping_sent[i].packets_sent_at_ping - inc_packets_sent;
+
+  /* packets received since last ping response */
+  p_recv = packets_received - inc_packets_received;
+
+  if (!p_sent || !p_recv)
+  {
+    /* just in case */
+    return;
+  }
+
+  /* percent loss server-to-client since PACKET_LOSS_INTERVAL */
+  iloss_sc = 100 -
+    (100 * (p_sent - (s_to_c_dropped - inc_s_to_c_dropped))) / p_sent;
+  /*
+   * we're not going to do any of the adjustments we did in tloss
+   * calculations since this starts fresh every PACKET_LOSS_INTERVAL
+   */
+  if (iloss_sc < 0)
+    iloss_sc = 0;
+
+  /* total percent loss client-to-server since PACKET_LOSS_INTERVAL */
+  iloss_cs = 100 -
+    (100 * (p_recv - (c_to_s_dropped - inc_c_to_s_dropped))) / p_recv;
+  /*
+   * we're not going to do any of the adjustments we did in tloss
+   * calculations since this starts fresh every PACKET_LOSS_INTERVAL
+   */
+  if (iloss_cs < 0)
+    iloss_cs = 0;
+
+  /*
+   * we update these variables every PACKET_LOSS_INTERVAL seconds to start a
+   * fresh increment
+   */
+  if ((timer % configvals->ping_iloss_interval) == 0)
+  {
+    inc_s_to_c_dropped = s_to_c_dropped;
+    inc_c_to_s_dropped = c_to_s_dropped;
+
+    inc_packets_sent = ping_sent[i].packets_sent_at_ping;
+    inc_packets_received = packets_received;
+  }
+
+  timer++;
+}
+
+#ifdef INL_STATS
+
+/*
+ * Lag stats struct player .p_avrt -	average round trip time ms struct
+ * player .p_stdv - 	standard deviation in rt time struct player .p_pkls -
+ * input/output packet loss
+ */
+
+static int sum, n, s2;
+static int M, var;
+
+void
+update_lag_stats()
+{
+  n++;
+  sum += ping_lag;
+  s2 += (ping_lag * ping_lag);
+  if (n == 1)
+    return;
+
+  M = sum / n;
+  var = (s2 - M * sum) / (n - 1);
+
+  /* average round trip time */
+  me->p_avrt = M;
+  /* standard deviation */
+  if (var > 0)
+    me->p_stdv = (int) isqrt(var);
+}
+
+void
+update_loss_stats()
+{
+  /*
+   * packet loss (as average of server-to-client, client-to-server loss),
+   * give tloss_sc extra weight (or should we?)
+   */
+  me->p_pkls = (2 * tloss_sc + tloss_cs) / 3;
+}
+#endif				/* INL_STATS */
+
+/* utilities */
+
+/* ms time from start */
+int
+mstime()
+{
+  static struct timeval tv_base = {0, 0};
+  struct timeval tv;
+
+  if (!tv_base.tv_sec)
+  {
+    gettimeofday(&tv_base, NULL);
+    return 0;
+  }
+  gettimeofday(&tv, NULL);
+  return (tv.tv_sec - tv_base.tv_sec) * 1000 +
+    (tv.tv_usec - tv_base.tv_usec) / 1000;
+}
+
+/* debugging */
+int
+msetime()
+{
+  struct timeval tv;
+  gettimeofday(&tv, NULL);
+  return (tv.tv_sec - 732737182) * 1000 + tv.tv_usec / 1000;
+}
+
+int
+uchar_diff(x, y)
+  int x, y;
+{
+  register res;
+
+  res = x - y;
+
+  if (res > 128)
+    return res - 256;
+  else if (res < -128)
+    return res + 256;
+  else
+    return res;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/pl_gen0.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,336 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+
+#include <math.h>
+#include <stdio.h>
+
+#include "defs.h"
+#include "struct.h"
+#include "shmem.h"
+#include "planets.h"
+
+/*
+ * This file contains utility procedures useful when laying out the galaxy.
+ * 
+ */
+
+/* #define SLOWER */
+
+int 
+place_stars(first, count, border, minpad, maxpad,
+	    othercheck, ocount)
+  struct planet *first;
+  int count;
+  int border, minpad, maxpad;
+  struct planet *othercheck;
+  int ocount;
+{
+  int i;
+  double x, y;
+
+  for (i = 0; i < count; i++)
+  {
+    int done, attempts;
+    int j;
+
+    attempts = 0;
+    done = 0;
+#ifndef SLOWER
+    x = drand48() * (configvals->gwidth - 2 * border) + border;
+    y = drand48() * (configvals->gwidth - 2 * border) + border;
+#endif
+    do
+    {
+
+      attempts++;
+
+#ifdef SLOWER
+      x = drand48() * (configvals->gwidth - 2 * border) + border;
+      y = drand48() * (configvals->gwidth - 2 * border) + border;
+#else
+      x = border + fmod(x + 3574 - border,
+			(configvals->gwidth - 2.0 * border));
+      y = border + fmod(y + 1034 - border,
+			(configvals->gwidth - 2.0 * border));
+#endif
+      done = 1;
+      /* check to make sure we aren't too close to other stars */
+      for (j = 0; j < ocount; j++)
+      {
+	double dist = hypot(x - othercheck[j].pl_x,
+			    y - othercheck[j].pl_y);
+	if (dist < minpad || dist > maxpad)
+	{
+	  done = 0;
+	  break;
+	}
+      }
+      /*
+       * check to make sure we're not too close to the current set of stars
+       */
+      if (done)
+	for (j = 0; j < i; j++)
+	{
+	  double dist = hypot(x - first[j].pl_x,
+			      y - first[j].pl_y);
+	  if (dist < minpad || dist > maxpad)
+	  {
+	    done = 0;
+	    break;
+	  }
+	}
+    } while (!done && attempts < 1000);
+    if (!done)
+      return 0;
+
+    first[i].pl_owner = NOBODY;
+    first[i].pl_system = (&first[i] - planets) + 1;
+    first[i].pl_flags |= PLSTAR;
+    move_planet(&first[i] - planets, (int) x, (int) y, 0);
+    first[i].pl_hinfo = ALLTEAM;/* all teams know its a star */
+    for (j = 0; j < MAXTEAM + 1; j++)
+    {				/* go put in info for teams */
+      first[i].pl_tinfo[j].owner = NOBODY;	/* nobody owns it */
+      first[i].pl_tinfo[j].armies = 0;
+      first[i].pl_tinfo[j].flags = first[i].pl_flags;
+    }
+  }
+  return 1;
+}
+
+void
+zero_plflags(first, count)
+  struct planet *first;
+  int count;
+{
+  int i;
+  for (i = 0; i < count; i++)
+  {
+    first[i].pl_flags = 0;
+  }
+}
+
+void
+randomize_atmospheres(first, count, p1, p2, p3, p4)
+  struct planet *first;
+  int count;
+  int p1, p2, p3, p4;
+{
+  int i;
+  int sum = p1 + p2 + p3 + p4;
+  for (i = 0; i < count; i++)
+  {
+    int val;
+    int atmosphere;
+    val = lrand48() % sum;
+    if ((val -= p1) < 0)
+      atmosphere = PLATYPE1;
+    else if ((val -= p2) < 0)
+      atmosphere = PLATYPE2;
+    else if ((val -= p3) < 0)
+      atmosphere = PLATYPE3;
+    else
+      atmosphere = PLPOISON;
+    first[i].pl_flags &= !PLATMASK;
+    first[i].pl_flags |= atmosphere;
+  }
+}
+
+/*
+ * special note.
+ * 
+ * It looks like this function originally wanted to make all Dilithium planets
+ * toxic, but the code was buggy and if an arable happened to be placed on
+ * the same planet later, you would get an STND DA.
+ * 
+ * I am loath to fix this bug, because it would noticably alter the galactic
+ * mix.  This must be brought before the PLC.
+ * 
+ * -RF
+ * 
+ */
+void
+randomize_resources(first, count, nm, nd, na)
+  struct planet *first;
+  int count;
+  int nm, nd, na;
+{
+  for (; count > 0; count--, first++)
+  {
+    int val;
+
+    val = lrand48() % count;
+    if (val < nm)
+    {
+      nm--;
+      first->pl_flags |= PLMETAL;
+      if (!configvals->resource_bombing)
+	first->pl_flags |= PLREPAIR;
+    }
+
+    val = lrand48() % count;
+    if (val < nd)
+    {
+      nd--;
+      first->pl_flags |= PLDILYTH;
+      first->pl_flags &= ~(PLATMASK | PLARABLE);
+      first->pl_flags |= PLPOISON;
+      if (!configvals->resource_bombing)
+	first->pl_flags |= PLFUEL;
+    }
+
+    val = lrand48() % count;
+    if (val < na)
+    {
+      na--;
+      first->pl_flags |= PLARABLE | PLATYPE1;
+      if (!configvals->resource_bombing)
+	first->pl_flags |= PLAGRI;
+    }
+  }
+}
+
+static int
+count_planets_in_system(sysnum)
+  int sysnum;
+{
+  int rval = 0;
+  int i;
+
+  for (i = 0; i < NUMPLANETS; i++)
+  {
+    if (PL_TYPE(planets[i]) == PLPLANET &&
+	planets[i].pl_system == sysnum)
+      rval++;
+  }
+  return rval;
+}
+
+static int
+pick_metal_planet_from_system(sysnum, nplanets)
+  int sysnum, nplanets;
+{
+  int i;
+
+  for (i = 0; i < NUMPLANETS; i++)
+  {
+    if (PL_TYPE(planets[i]) == PLPLANET &&
+	planets[i].pl_system == sysnum &&
+	(planets[i].pl_flags & PLMETAL))
+    {
+      if (lrand48() % nplanets == 0)
+	return i;
+      nplanets--;
+    }
+  }
+  return -1;
+}
+
+static int
+pick_planet_from_system(sysnum, nplanets)
+  int sysnum, nplanets;
+{
+  int i;
+
+  if (nplanets < 0)
+    nplanets = count_planets_in_system(sysnum);
+
+  for (i = 0; i < NUMPLANETS; i++)
+  {
+    if (PL_TYPE(planets[i]) == PLPLANET &&
+	planets[i].pl_system == sysnum)
+    {
+      if (lrand48() % nplanets == 0)
+	return i;
+      nplanets--;
+    }
+  }
+  return -1;
+}
+
+
+void
+justify_galaxy(numsystems)
+  int numsystems;
+/*
+ * Balances the galaxy to be "fair".  Currently ensures that: -> One metal
+ * planet exists within each system.
+ */
+{
+  int i, j;
+  int *metalcount;
+
+  metalcount = malloc(sizeof(*metalcount) * (numsystems + 1));
+
+  for (i = 0; i <= numsystems; i++)
+    metalcount[i] = 0;
+
+  for (i = 0; i < NUMPLANETS; i++)
+  {
+    switch (PL_TYPE(planets[i]))
+    {
+     case PLPLANET:
+      {
+	int system = planets[i].pl_system;
+	if (system < 0 || system > numsystems)
+	  break;
+	if (planets[i].pl_flags & PLMETAL)
+	  metalcount[system]++;
+      }
+      break;
+     default:
+      ;
+      /* don't care about other stuff */
+    }
+  }
+
+  for (i = 1; i <= numsystems; i++)
+  {
+    if (metalcount[i] < 1)
+    {
+      int to, from;
+      int randbase;
+
+      randbase = lrand48() % (numsystems + 1);
+
+      for (j = 0; j <= numsystems; j++)
+	if (metalcount[(j + randbase) % (numsystems + 1)] > 1)
+	  break;
+      if (j > numsystems)
+      {
+	fprintf(stderr, "error stealing metal planet.  Too few!\n");
+	return;
+      }
+      j = (j + randbase) % (numsystems + 1);
+      to = pick_planet_from_system(i, -1);
+      from = pick_metal_planet_from_system(j, metalcount[j]);
+      planets[to].pl_flags |= PLMETAL;
+      planets[from].pl_flags &= ~PLMETAL;
+      if (!configvals->resource_bombing)
+      {
+	planets[to].pl_flags |= PLREPAIR;
+	planets[from].pl_flags &= PLREPAIR;
+      }
+      metalcount[i]++;
+      metalcount[j]--;
+    }
+  }
+  free(metalcount);
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/pl_gen1.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,509 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+#include "config.h"
+#include <math.h>
+
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "shmem.h"
+#include "planets.h"
+
+#define SYSWIDTH	(GWIDTH/5)	/* width of a system */
+
+#define SYSTEMS		9	/* number of planetary systems */
+
+/* atmosphere chances form a cascade win rand()%100 */
+#define PATMOS1		40	/* chance for normal atmosphere */
+#define PATMOS2		70	/* chance for thin atmosphere */
+#define PATMOS3		90	/* chance for slightly toxic stmos */
+#define PPOISON		100	/* chance for poison atmos */
+
+/* defines that deal with planets resources and types */
+#define NMETAL		2	/* number of metal deposits */
+#define NDILYTH		10	/* number of dilythium deposits */
+#define NARABLE		30	/* number of arable land planets */
+/* defines that deal with star placement */
+
+#define GW	((float)GWIDTH)	/* size of galaxy in floating point */
+#define	STARBORD	(SYSWIDTH/2)
+#define STARMIN		(GW/3.7)/* min dist between stars */
+#define STARMIN2	STARMIN*STARMIN	/* min star dist squared */
+
+/* defines that deal with systems and their planets */
+#define SYSMINP		4	/* min number of planets per system */
+#define SYSADD		2	/* number possible above min number */
+#define SYSBORD		(5000.0	+ (float)(GWIDTH/200))	/* min distance from
+							 * border wall */
+#define SYSMIN		(7000.0	+ (float)(GWIDTH/100))	/* min distance between
+							 * objects */
+#define SYSMIN2		(SYSMIN*SYSMIN)	/* square of sysmin distance */
+#define SYSPLMIN	5	/* min number of planets for system */
+#define SYSPLADD	0	/* number of possible extra planets */
+#define MINARMY 8		/* min numer of armies on a planet */
+#define MAXARMY 15		/* max number of armies on a planet */
+
+/* other defines */
+#define HOMEARMIES 30		/* number of armies on home planets */
+#define COLONYARMIES 10		/* number of armies for colony planet */
+
+
+/* defines dealing with growth timers */
+#define PLGFUEL		configvals->plgrow.fuel	/* time for growth of fuel
+						 * depot */
+#define PLGAGRI		configvals->plgrow.agri	/* time for growth of agri */
+#define PLGREPAIR	configvals->plgrow.repair	/* time for growth of
+							 * repair */
+#define PLGSHIP		configvals->plgrow.shipyard	/* time for growth of
+							 * shipyard */
+
+
+#if 0
+/*-------------------------------GENRESOURCES----------------------------*/
+/*
+ * This function goes through the planets structure and determines what kind
+ * of atmosphere and what kind of surface the planets have.  It generates the
+ * stars that will be used as system centers ans then places atmospheres on
+ * the other planets.  It then distributes the resources on the planet
+ * surfaces.
+ */
+
+static void
+genresources()
+{
+  int i;			/* looping vars */
+  int t;			/* temp var */
+
+  for (i = 0; i < SYSTEMS; i++)	/* first planets are stars */
+    planets[i].pl_flags |= PLSTAR;	/* or in star flag */
+  for (i = SYSTEMS; i < NUMPLANETS; i++)
+  {				/* generate atmospheres */
+    t = lrand48() % 100;	/* random # 0-99 */
+    if (t < PATMOS1)		/* is it atmosphere type 1 */
+      planets[i].pl_flags |= PLATYPE1;
+    else if (t < PATMOS2)	/* is it atmosphere type 2 */
+      planets[i].pl_flags |= PLATYPE2;
+    else if (t < PATMOS3)	/* is it atmosphere type 3 */
+      planets[i].pl_flags |= PLATYPE3;
+    else if (t < PPOISON)	/* is it poison atmosphere */
+      planets[i].pl_flags |= PLPOISON;
+  }
+  for (i = 0; i < NMETAL; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % (NUMPLANETS - SYSTEMS) + SYSTEMS;	/* random planet */
+    planets[t].pl_flags |= PLMETAL;	/* OR in the metal flag */
+    if (!configvals->resource_bombing)
+      planets[t].pl_flags |= PLREPAIR;
+  }
+  for (i = 0; i < NDILYTH; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % (NUMPLANETS - SYSTEMS) + SYSTEMS;	/* random planet */
+    planets[t].pl_flags |= PLDILYTH;	/* OR in the dilyth flag */
+    planets[t].pl_flags &= ~(PLATMASK | PLARABLE);	/* zero off previous
+							 * atmos */
+    planets[t].pl_flags |= PLPOISON;	/* dilyth poisons atmosphere */
+    if (!configvals->resource_bombing)
+      planets[t].pl_flags |= PLFUEL;
+  }
+  for (i = 0; i < NARABLE; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % (NUMPLANETS - SYSTEMS) + SYSTEMS;	/* random planet */
+    planets[t].pl_flags |= PLARABLE | PLATYPE1;	/* OR in the arable flag */
+    if (!configvals->resource_bombing)
+      planets[t].pl_flags |= PLAGRI;
+  }
+}
+#endif
+
+
+#if 0
+/*--------------------------------PLACESTARS------------------------------*/
+/*
+ * This function places each system's star.  The stars are expected to be in
+ * the first SYSTEMS number of planets.  The coordinates of the stars are
+ * placed in the space grid.
+ */
+
+static int
+placestars()
+{
+  int i, j;			/* looping vars */
+  double x, y;			/* to hold star coordinates */
+  int done;			/* flag to indicate done */
+  double dx, dy;		/* delta x and y's */
+  int attempts;
+
+  for (i = 0; i < SYSTEMS; i++)
+  {				/* star for each system */
+    x = drand48() * GW;		/* pick intial coords */
+    y = drand48() * GW;
+    attempts = 0;
+    do
+    {				/* do until location found */
+      attempts++;
+      done = 0;			/* not done yet */
+      x = fmod(x + 3574.0, GW);	/* offset coords a little */
+      y = fmod(y + 134.0, GW);	/* every loop */
+      if ((x > GW - STARBORD) || (x < STARBORD)
+	  || (y < STARBORD) || (y > GW - STARBORD))
+	continue;		/* too close to border? */
+      done = 1;			/* assume valid cord found */
+      for (j = 0; j < i; j++)
+      {				/* go through previous stars */
+	dx = fabs(x - (double) planets[j].pl_x);
+	dy = fabs(y - (double) planets[j].pl_y);
+	if (dx * dx + dy * dy < STARMIN2)	/* if too close then */
+	  done = 0;		/* we must get another coord */
+      }
+    } while (!done && attempts < 1000);	/* do until location found */
+
+    if (!done)
+      return 0;
+
+    planets[i].pl_owner = NOBODY;	/* no team owns a star */
+    planets[i].pl_flags |= PLSTAR;	/* mark planet as a star */
+    move_planet(i, (int) x, (int) y, 0);
+    planets[i].pl_system = i + 1;	/* mark the sytem number */
+    planets[i].pl_hinfo = ALLTEAM;	/* all teams know its a star */
+    for (j = 0; j < MAXTEAM + 1; j++)
+    {				/* go put in info for teams */
+      planets[i].pl_tinfo[j].owner = NOBODY;	/* nobody owns it */
+      planets[i].pl_tinfo[j].armies = 0;
+      planets[i].pl_tinfo[j].flags = planets[i].pl_flags;
+    }
+  }
+  return 1;
+}
+#endif
+
+
+
+/*-----------------------------PLACESYSTEMS------------------------------*/
+/*
+ * This function places the planets in each star's system.  The function will
+ * return the index of the first planet that was not placed in a system. The
+ * coordinates of the planets are placed in the space grid.
+ */
+
+static int
+placesystems()
+{
+  int i, j, k;			/* looping vars */
+  double x, y;			/* to hold star coordinates */
+  int done;			/* flag to indicate done */
+  double dx, dy;		/* delta x and y's */
+  int n;			/* number of planet to place */
+  int np;			/* number of planets in system */
+  int attempts;
+
+  n = SYSTEMS;			/* first planet to place */
+  for (i = 0; i < SYSTEMS; i++)
+  {				/* planets for each system */
+    np = SYSPLMIN + lrand48() % (SYSPLADD + 1);	/* how many planets */
+    for (k = 0; k < np; k++)
+    {				/* go place the planets */
+      attempts = 0;
+      do
+      {				/* do until location found */
+	attempts++;
+	done = 0;		/* not done yet */
+	dx = (drand48() * SYSWIDTH - SYSWIDTH / 2.0);
+	dy = (drand48() * SYSWIDTH - SYSWIDTH / 2.0);
+	if (dx * dx + dy * dy > (SYSWIDTH / 2.0) * (SYSWIDTH / 2.0))
+	  continue;		/* might orbit its way out of the galaxy */
+	x = planets[i].pl_x + dx;
+	y = planets[i].pl_y + dy;
+	if ((x > GW - SYSBORD) || (x < SYSBORD)
+	    || (y < SYSBORD) || (y > GW - SYSBORD))
+	  continue;		/* too close to border? */
+
+	done = 1;		/* assume valid coord found */
+	for (j = 0; j < n; j++)
+	{			/* go through previous planets */
+	  dx = fabs(x - (double) planets[j].pl_x);
+	  dy = fabs(y - (double) planets[j].pl_y);
+	  if (dx * dx + dy * dy < SYSMIN2)
+	  {			/* if too close to another star */
+	    done = 0;		/* we must get another coord */
+	  }
+	}
+      } while (!done && attempts < 100);	/* do until location found */
+
+      if (!done)
+	return 0;		/* universe too crowded, try again */
+
+      move_planet(n, (int) x, (int) y, 0);
+      planets[n].pl_system = i + 1;	/* mark the sytem number */
+      planets[n].pl_armies = MINARMY + lrand48() % (MAXARMY - MINARMY);
+      n++;			/* go to next planet */
+    }
+  }
+  return (n);			/* return index of next planet */
+}
+
+
+
+
+/*-----------------------------PLACEINDEP------------------------------*/
+/*
+ * This function places idependent planets that are not in a system. They can
+ * appear anywhere in the galaxy as long as they are not too close to another
+ * planet.  The coords are put in the space grid.
+ */
+
+static int
+placeindep(n)
+  int n;
+/* number of planet to start with */
+{
+  int i, j;			/* looping vars */
+  double x, y;			/* to hold star coordinates */
+  int done;			/* flag to indicate done */
+  double dx, dy;		/* delta x and y's */
+  int attempts;
+
+  for (i = n; i < (NUMPLANETS - (WORMPAIRS * 2)); i++)
+  {
+    /* go through rest of planets */
+    x = drand48() * GW;		/* pick intial coords */
+    y = drand48() * GW;
+    attempts = 0;
+    do
+    {				/* do until location found */
+      attempts++;
+      done = 0;			/* not done yet */
+      x = fmod(x + 3574.0, GW);	/* offset coords a little */
+      y = fmod(y + 134.0, GW);	/* every loop */
+      if ((x > GW - SYSBORD) || (x < SYSBORD)
+	  || (y < SYSBORD) || (y > GW - SYSBORD))
+	continue;		/* too close to border? */
+      done = 1;			/* assume valid coord */
+      for (j = 0; j < n; j++)
+      {				/* go through previous planets */
+	dx = fabs(x - (double) planets[j].pl_x);
+	dy = fabs(y - (double) planets[j].pl_y);
+	if (dx * dx + dy * dy < SYSMIN2)
+	{			/* if planet to close */
+	  done = 0;		/* we must get another coord */
+	}
+      }
+    } while (!done && attempts < 100);	/* do until location found */
+
+    if (!done)
+      return 0;
+
+    move_planet(n, (int) x, (int) y, 0);
+    planets[n].pl_system = 0;	/* mark the no sytem */
+    planets[n].pl_armies = MINARMY + lrand48() % (MAXARMY - MINARMY);
+    n++;			/* go to next planet */
+  }
+  for (i = n; i < NUMPLANETS; i++)	/* now place wormholes */
+  {
+    x = drand48() * GW;		/* pick intial coords */
+    y = drand48() * GW;
+    attempts = 0;
+    do
+    {				/* do until location found */
+      attempts++;
+      done = 0;			/* not done yet */
+      x = fmod(x + 3574.0, GW);	/* offset coords a little */
+      y = fmod(y + 1034.0, GW);	/* every loop */
+#if 0
+      if ((x > GW) || (y > GW))
+	continue;		/* too close to border? */
+#endif
+      done = 1;			/* assume valid coord */
+      for (j = 0; j < n; j++)
+      {				/* go through previous planets */
+	dx = fabs(x - (double) planets[j].pl_x);
+	dy = fabs(y - (double) planets[j].pl_y);
+	if (dx * dx + dy * dy < SYSMIN2)
+	{			/* if planet to close */
+	  done = 0;		/* we must get another coord */
+	}
+      }
+    } while (!done && attempts < 200);	/* do until location found */
+
+    if (!done)
+      return 0;
+
+    move_planet(n, (int) x, (int) y, 0);
+    planets[n].pl_system = 0;	/* mark the no system */
+    planets[n].pl_flags |= PLWHOLE;	/* mark the planet as a wormhole */
+    /* the armies in a wormhole is the other wormhole's x coord */
+    /* the radius is the other wormhole's y coord */
+    if (NUMPLANETS % 2)
+    {
+      if (!(n % 2))
+      {
+	planets[n].pl_armies = planets[n - 1].pl_x;
+	planets[n].pl_radius = planets[n - 1].pl_y;
+	planets[n - 1].pl_armies = planets[n].pl_x;
+	planets[n - 1].pl_radius = planets[n].pl_y;
+      }
+    }
+    else
+    {
+      if (n % 2)
+      {
+	planets[n].pl_armies = planets[n - 1].pl_x;
+	planets[n].pl_radius = planets[n - 1].pl_y;
+	planets[n - 1].pl_armies = planets[n].pl_x;
+	planets[n - 1].pl_radius = planets[n].pl_y;
+      }
+    }
+    planets[i].pl_owner = NOBODY;	/* no team owns a star */
+    planets[i].pl_hinfo = ALLTEAM;	/* all teams know its a star */
+    for (j = 0; j < MAXTEAM + 1; j++)
+    {				/* go put in info for teams */
+      planets[i].pl_tinfo[j].owner = NOBODY;	/* nobody owns it */
+      planets[i].pl_tinfo[j].armies = 0;
+      planets[i].pl_tinfo[j].flags = planets[i].pl_flags;
+    }
+    n++;			/* go to next planet */
+  }
+  return 1;
+}
+
+
+
+
+/*---------------------------------PLACERACES------------------------------*/
+/*
+ * This function places the races in the galaxy.  Each race is placed in a
+ * different system.  The race is given a home world with an Agri and Ship-
+ * yard on it and HOMEARMIES.  They are also given a conoly planet with
+ * dilythium deposits and COLONYARMIES on it.
+ */
+
+static void
+placeraces()
+{
+  int i, j, k;			/* looping vars */
+  int p;			/* to hold planet for race */
+
+  for (i = 0; i < 4; i++)
+  {				/* go through races */
+    /* find home planet */
+    p = lrand48() % NUMPLANETS;	/* pick random planet */
+    while ((planets[p].pl_system != i + 1)
+	   || (PL_TYPE(planets[p]) == PLSTAR)
+	   || (planets[p].pl_owner != NOBODY))
+      p = (p + 1) % NUMPLANETS;	/* go on to next planet */
+    planets[p].pl_flags &= ~PLSURMASK;	/* make sure no dilithium */
+    planets[p].pl_flags |= (PLMETAL | PLARABLE);	/* metal and arable */
+    planets[p].pl_flags |= PLATYPE1;	/* good atmosphere */
+    planets[p].pl_flags |= (PLAGRI | PLSHIPYARD | PLREPAIR);
+    planets[p].pl_tagri = PLGAGRI;	/* set timers for resources */
+    planets[p].pl_tshiprepair = PLGSHIP;
+    planets[p].pl_owner = 1 << i;	/* make race the owner */
+#if 0				/* home planets do not have traditional names */
+    strcpy(planets[p].pl_name, homenames[1 << i]);	/* set name and length */
+    planets[p].pl_namelen = strlen(homenames[1 << i]);
+#endif
+    planets[p].pl_armies = HOMEARMIES;	/* set the armies */
+    planets[p].pl_hinfo = 1 << i;	/* race has info on planet */
+    planets[p].pl_tinfo[1 << i].owner = 1 << i;	/* know about owner */
+    planets[p].pl_tinfo[1 << i].armies = planets[p].pl_armies;
+    planets[p].pl_tinfo[1 << i].flags = planets[p].pl_flags;
+    /* find colony planet */
+    p = lrand48() % NUMPLANETS;	/* pick random planet */
+    while ((planets[p].pl_system != i + 1)
+	   || (PL_TYPE(planets[p]) == PLSTAR)
+	   || (planets[p].pl_owner != NOBODY))
+      p = (p + 1) % NUMPLANETS;	/* go on to next planet */
+    planets[p].pl_flags |= PLFUEL;	/* make fuel depot */
+    planets[p].pl_tfuel = PLGFUEL;	/* set timer for fuel depot */
+    planets[p].pl_flags &= ~PLATMASK;	/* take off previous atmos */
+    planets[p].pl_flags |= PLPOISON;	/* poison atmosphere */
+    planets[p].pl_flags |= PLDILYTH;	/* dilythium deposits */
+    planets[p].pl_owner = 1 << i;	/* make race the owner */
+    planets[p].pl_armies = COLONYARMIES;	/* set the armies */
+    planets[p].pl_hinfo = 1 << i;	/* race knows about */
+    planets[p].pl_tinfo[1 << i].owner = 1 << i;	/* know about owner */
+    planets[p].pl_tinfo[1 << i].armies = planets[p].pl_armies;
+    planets[p].pl_tinfo[1 << i].flags = planets[p].pl_flags;
+    for (j = 0; j < NUMPLANETS; j++)
+    {
+      if ((planets[j].pl_system == i + 1) && (PL_TYPE(planets[j]) != PLSTAR))
+      {
+#ifdef LEAGUE_SUPPORT
+	for (k = (status2->league ? 0 : i);
+	     k < (status2->league ? 4 : i + 1);
+	     k++)
+#else
+	k = i;
+#endif
+	{
+	  planets[j].pl_owner = 1 << i;
+	  planets[j].pl_hinfo =
+#ifdef LEAGUE_SUPPORT
+	    status2->league ? (1 << 4) - 1 :
+#endif
+	    (1 << i);
+	  planets[j].pl_tinfo[1 << k].owner = 1 << i;
+	  planets[j].pl_tinfo[1 << k].armies = planets[j].pl_armies;
+	  planets[j].pl_tinfo[1 << k].flags = planets[j].pl_flags;
+	}
+      }
+    }
+  }
+}
+
+/*
+ * Generate a complete galaxy.  This is the algorithm used by the paradise
+ * 2.01 server in its first release. I hope.
+ */
+
+void
+gen_galaxy_1()
+{
+  int t;
+  while (1)
+  {
+		NUMPLANETS = 60;
+    GWIDTH = 200000;
+    initplanets();		/* initialize planet structures */
+
+    /* place the resources */
+    zero_plflags(planets, NUMPLANETS);
+    randomize_atmospheres(planets + SYSTEMS, NUMPLANETS - SYSTEMS,
+			  PATMOS1, PATMOS2, PATMOS3, PPOISON);
+    randomize_resources(planets + SYSTEMS, NUMPLANETS - SYSTEMS,
+			NMETAL, NDILYTH, NARABLE);
+
+    /* place system centers */
+    if (!place_stars(planets, SYSTEMS,
+		     (int) STARBORD, (int) STARMIN, (int) GW,
+		     (struct planet *) 0, 0))
+      continue;
+
+    t = placesystems();		/* place planets in systems */
+    if (!t)
+      continue;
+
+    if (!placeindep(t))		/* place independent planets */
+      continue;
+
+    break;			/* success */
+  }
+  if (configvals->justify_galaxy)
+    justify_galaxy(SYSTEMS);
+
+  placeraces();			/* place home planets for races */
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/pl_gen2.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,559 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <math.h>
+
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "shmem.h"
+#include "planets.h"
+
+#define SYSWIDTH	(GWIDTH/8)	/* width of a system */
+
+#define SYSTEMS		8	/* number of planetary systems */
+
+/* atmosphere chances form a cascade win rand()%100 */
+#define PATMOS1		40	/* chance for normal atmosphere */
+#define PATMOS2		70	/* chance for thin atmosphere */
+#define PATMOS3		90	/* chance for slightly toxic stmos */
+#define PPOISON		100	/* chance for poison atmos */
+
+/* defines that deal with planets resources and types */
+#define NMETAL		3	/* number of metal deposits */
+#define NDILYTH		10	/* number of dilythium deposits */
+#define NARABLE		30	/* number of arable land planets */
+/* defines that deal with star placement */
+
+#define	GW	((float)GWIDTH)	/* size of galaxy in floating point */
+#define	STARBORD ((GWIDTH-SYSWIDTH*4)/2)	/* cannot be this close to
+						 * wall */
+#define STARMIN		(6*SYSWIDTH/5)	/* min dist between stars */
+#define STARMIN2	STARMIN*STARMIN	/* min star dist squared */
+
+/* defines that deal with systems and their planets */
+#define SYSMINP		4	/* min number of planets per system */
+#define SYSADD		2	/* number possible above min number */
+#define SYSBORD		(SYSWIDTH/4)	/* min distance from border wall */
+#define SYSMIN		(SYSWIDTH/3)	/* min distance between objects */
+#define SYSMIN2		(SYSMIN*SYSMIN)	/* square of sysmin distance */
+#define SYSPLMIN	5	/* min number of planets for system */
+#define SYSPLADD	0	/* number of possible extra planets */
+#define MINARMY 8		/* min numer of armies on a planet */
+#define MAXARMY 15		/* max number of armies on a planet */
+
+/* other defines */
+#define HOMEARMIES 30		/* number of armies on home planets */
+#define COLONYARMIES 10		/* number of armies for colony planet */
+
+
+/* defines dealing with growth timers */
+#define PLGFUEL		configvals->plgrow.fuel	/* time for growth of fuel
+						 * depot */
+#define PLGAGRI		configvals->plgrow.agri	/* time for growth of agri */
+#define PLGREPAIR	configvals->plgrow.repair	/* time for growth of
+							 * repair */
+#define PLGSHIP		configvals->plgrow.shipyard	/* time for growth of
+							 * shipyard */
+
+
+#if 0
+/*-------------------------------GENRESOURCES----------------------------*/
+/*
+ * This function goes through the planets structure and determines what kind
+ * of atmosphere and what kind of surface the planets have.  It generates the
+ * stars that will be used as system centers ans then places atmospheres on
+ * the other planets.  It then distributes the resources on the planet
+ * surfaces.
+ */
+
+static void
+genresources()
+{
+  int i;			/* looping vars */
+  int t;			/* temp var */
+
+  for (i = 0; i < NUMPLANETS; i++)
+    planets[i].pl_flags = 0;
+
+  for (i = 0; i < SYSTEMS; i++)	/* first planets are stars */
+    planets[i].pl_flags |= PLSTAR;	/* or in star flag */
+  for (i = SYSTEMS; i < NUMPLANETS; i++)
+  {				/* generate atmospheres */
+    t = lrand48() % 100;	/* random # 0-99 */
+    if (t < PATMOS1)		/* is it atmosphere type 1 */
+      planets[i].pl_flags |= PLATYPE1;
+    else if (t < PATMOS2)	/* is it atmosphere type 2 */
+      planets[i].pl_flags |= PLATYPE2;
+    else if (t < PATMOS3)	/* is it atmosphere type 3 */
+      planets[i].pl_flags |= PLATYPE3;
+    else if (t < PPOISON)	/* is it poison atmosphere */
+      planets[i].pl_flags |= PLPOISON;
+  }
+  for (i = 0; i < NMETAL; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % (NUMPLANETS - SYSTEMS) + SYSTEMS;	/* random planet */
+    planets[t].pl_flags |= PLMETAL;	/* OR in the metal flag */
+    if (!configvals->resource_bombing)
+      planets[t].pl_flags |= PLREPAIR;
+  }
+  for (i = 0; i < NDILYTH; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % (NUMPLANETS - SYSTEMS) + SYSTEMS;	/* random planet */
+    planets[t].pl_flags |= PLDILYTH;	/* OR in the dilyth flag */
+    planets[t].pl_flags &= ~(PLATMASK | PLARABLE);	/* zero off previous
+							 * atmos */
+    planets[t].pl_flags |= PLPOISON;	/* dilyth poisons atmosphere */
+    if (!configvals->resource_bombing)
+      planets[t].pl_flags |= PLFUEL;
+  }
+  for (i = 0; i < NARABLE; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % (NUMPLANETS - SYSTEMS) + SYSTEMS;	/* random planet */
+    planets[t].pl_flags |= PLARABLE | PLATYPE1;	/* OR in the arable flag */
+    if (!configvals->resource_bombing)
+      planets[t].pl_flags |= PLAGRI;
+  }
+}
+#endif
+
+
+#if 0
+/*--------------------------------PLACESTARS------------------------------*/
+/*
+ * This function places each system's star.  The stars are expected to be in
+ * the first SYSTEMS number of planets.  The coordinates of the stars are
+ * placed in the space grid.
+ */
+
+static int
+placestars()
+{
+  int i, j;			/* looping vars */
+  double x, y;			/* to hold star coordinates */
+  int done;			/* flag to indicate done */
+  double dx, dy;		/* delta x and y's */
+  int attempts;
+
+  for (i = 0; i < SYSTEMS; i++)
+  {				/* star for each system */
+    x = drand48() * GW;		/* pick intial coords */
+    y = drand48() * GW;
+    attempts = 0;
+    do
+    {				/* do until location found */
+      attempts++;
+      done = 0;			/* not done yet */
+      x = fmod(x + 3574.0, GW);	/* offset coords a little */
+      y = fmod(y + 134.0, GW);	/* every loop */
+      if ((x > GW - STARBORD) || (x < STARBORD)
+	  || (y < STARBORD) || (y > GW - STARBORD))
+	continue;		/* too close to border? */
+      done = 1;			/* assume valid cord found */
+      for (j = 0; j < i; j++)
+      {				/* go through previous stars */
+	dx = fabs(x - (double) planets[j].pl_x);
+	dy = fabs(y - (double) planets[j].pl_y);
+	if (dx * dx + dy * dy < STARMIN2)	/* if too close then */
+	  done = 0;		/* we must get another coord */
+      }
+    } while (!done && attempts < 1000);	/* do until location found */
+
+    if (!done)
+      return 0;
+
+    planets[i].pl_flags |= PLSTAR;	/* mark planet as a star */
+    move_planet(i, (int) x, (int) y, 0);
+    planets[i].pl_system = i + 1;	/* mark the sytem number */
+    planets[i].pl_hinfo = ALLTEAM;	/* all teams know its a star */
+    for (j = 0; j < MAXTEAM + 1; j++)
+    {				/* go put in info for teams */
+      planets[i].pl_tinfo[j].owner = NOBODY;	/* nobody owns it */
+      planets[i].pl_tinfo[j].armies = 0;
+      planets[i].pl_tinfo[j].flags = planets[i].pl_flags;
+    }
+  }
+  return 1;
+}
+#endif
+
+
+
+/*-----------------------------PLACESYSTEMS------------------------------*/
+/*
+ * This function places the planets in each star's system.  The function will
+ * return the index of the first planet that was not placed in a system. The
+ * coordinates of the planets are placed in the space grid.
+ */
+
+static int
+placesystems()
+{
+  int i, j, k;			/* looping vars */
+  double x, y;			/* to hold star coordinates */
+  int done;			/* flag to indicate done */
+  double dx, dy;		/* delta x and y's */
+  int n;			/* number of planet to place */
+  int np;			/* number of planets in system */
+  int attempts;
+
+  n = SYSTEMS;			/* first planet to place */
+  for (i = 0; i < SYSTEMS; i++)
+  {				/* planets for each system */
+    np = SYSPLMIN + lrand48() % (SYSPLADD + 1);	/* how many planets */
+    for (k = 0; k < np; k++)
+    {				/* go place the planets */
+      attempts = 0;
+      do
+      {				/* do until location found */
+	attempts++;
+	done = 0;		/* not done yet */
+	dx = (drand48() * SYSWIDTH - SYSWIDTH / 2.0);
+	dy = (drand48() * SYSWIDTH - SYSWIDTH / 2.0);
+	if (dx * dx + dy * dy > (SYSWIDTH / 2.0) * (SYSWIDTH / 2.0))
+	  continue;		/* might orbit its way out of the galaxy */
+	x = planets[i].pl_x + dx;
+	y = planets[i].pl_y + dy;
+	if ((x > GW - SYSBORD) || (x < SYSBORD)
+	    || (y < SYSBORD) || (y > GW - SYSBORD))
+	  continue;		/* too close to border? */
+
+	done = 1;		/* assume valid coord found */
+	for (j = 0; j < n; j++)
+	{			/* go through previous planets */
+	  dx = fabs(x - (double) planets[j].pl_x);
+	  dy = fabs(y - (double) planets[j].pl_y);
+	  if (dx * dx + dy * dy < SYSMIN2)
+	  {			/* if too close to another star */
+	    done = 0;		/* we must get another coord */
+	  }
+	}
+      } while (!done && attempts < 100);	/* do until location found */
+
+      if (!done)
+	return 0;		/* universe too crowded, try again */
+
+      move_planet(n, (int) x, (int) y, 0);
+      planets[n].pl_system = i + 1;	/* mark the sytem number */
+      planets[n].pl_armies = MINARMY + lrand48() % (MAXARMY - MINARMY);
+      n++;			/* go to next planet */
+    }
+  }
+  return (n);			/* return index of next planet */
+}
+
+
+
+
+/*-----------------------------PLACEINDEP------------------------------*/
+/*
+ * This function places idependent planets that are not in a system. They can
+ * appear anywhere in the galaxy as long as they are not too close to another
+ * planet.  The coords are put in the space grid.
+ */
+
+static int
+placeindep(n)
+  int n;
+/* number of planet to start with */
+{
+  int i, j;			/* looping vars */
+  double x, y;			/* to hold star coordinates */
+  int done;			/* flag to indicate done */
+  double dx, dy;		/* delta x and y's */
+  int attempts;
+
+  for (i = n; i < (NUMPLANETS - (WORMPAIRS * 2)); i++)
+  {
+    /* go through rest of planets */
+    x = drand48() * GW;		/* pick intial coords */
+    y = drand48() * GW;
+    attempts = 0;
+    do
+    {				/* do until location found */
+      attempts++;
+      done = 0;			/* not done yet */
+      x = fmod(x + 3574.0, GW);	/* offset coords a little */
+      y = fmod(y + 134.0, GW);	/* every loop */
+      if ((x > GW - SYSBORD) || (x < SYSBORD)
+	  || (y < SYSBORD) || (y > GW - SYSBORD))
+	continue;		/* too close to border? */
+      done = 1;			/* assume valid coord */
+      for (j = 0; j < i; j++)
+      {				/* go through previous planets */
+	dx = fabs(x - (double) planets[j].pl_x);
+	dy = fabs(y - (double) planets[j].pl_y);
+	if (dx * dx + dy * dy < SYSMIN2 * (2 * 2))
+	{			/* if planet to close */
+	  done = 0;		/* we must get another coord */
+	}
+      }
+    } while (!done && attempts < 100);	/* do until location found */
+
+    if (!done)
+      return 0;
+
+    move_planet(n, (int) x, (int) y, 0);
+    planets[n].pl_system = 0;	/* mark the no sytem */
+    planets[n].pl_armies = MINARMY + lrand48() % (MAXARMY - MINARMY);
+    n++;			/* go to next planet */
+  }
+  for (i = n; i < NUMPLANETS; i++)	/* now place wormholes */
+  {
+    x = drand48() * GW;		/* pick intial coords */
+    y = drand48() * GW;
+    attempts = 0;
+    do
+    {				/* do until location found */
+      attempts++;
+      done = 0;			/* not done yet */
+      x = fmod(x + 3574.0, GW);	/* offset coords a little */
+      y = fmod(y + 134.0, GW);	/* every loop */
+
+      done = 1;			/* assume valid coord */
+      for (j = 0; j < n; j++)
+      {				/* go through previous planets */
+	dx = fabs(x - (double) planets[j].pl_x);
+	dy = fabs(y - (double) planets[j].pl_y);
+	if (dx * dx + dy * dy < SYSMIN2)
+	{			/* if planet to close */
+	  done = 0;		/* we must get another coord */
+	}
+      }
+    } while (!done && attempts < 200);	/* do until location found */
+
+    if (!done)
+      return 0;
+
+    move_planet(n, (int) x, (int) y, 0);
+    planets[n].pl_system = 0;	/* mark the no system */
+    planets[n].pl_flags |= PLWHOLE;	/* mark the planet as a wormhole */
+    /* the armies in a wormhole is the other wormhole's x coord */
+    /* the radius is the other wormhole's y coord */
+    if (NUMPLANETS % 2)
+    {
+      if (!(n % 2))
+      {
+	planets[n].pl_armies = planets[n - 1].pl_x;
+	planets[n].pl_radius = planets[n - 1].pl_y;
+	planets[n - 1].pl_armies = planets[n].pl_x;
+	planets[n - 1].pl_radius = planets[n].pl_y;
+      }
+    }
+    else
+    {
+      if (n % 2)
+      {
+	planets[n].pl_armies = planets[n - 1].pl_x;
+	planets[n].pl_radius = planets[n - 1].pl_y;
+	planets[n - 1].pl_armies = planets[n].pl_x;
+	planets[n - 1].pl_radius = planets[n].pl_y;
+      }
+    }
+    planets[i].pl_owner = NOBODY;	/* no team owns a wormhole */
+    planets[i].pl_hinfo = ALLTEAM;	/* all teams know its a wormhole */
+    for (j = 0; j < MAXTEAM + 1; j++)
+    {				/* go put in info for teams */
+      planets[i].pl_tinfo[j].owner = NOBODY;	/* nobody owns it */
+      planets[i].pl_tinfo[j].armies = 0;
+      planets[i].pl_tinfo[j].flags = planets[i].pl_flags;
+    }
+    n++;			/* go to next planet */
+  }
+  return 1;
+}
+
+
+
+
+/*---------------------------------PLACERACES------------------------------*/
+/*
+ * This function places the races in the galaxy.  Each race is placed in a
+ * different system.  The race is given a home world with an Agri and Ship-
+ * yard on it and HOMEARMIES.  They are also given a colony planet with
+ * dilythium deposits and COLONYARMIES on it.
+ */
+
+static int
+placeraces()
+{
+  int i, j, k;			/* looping vars */
+  int p;			/* to hold planet for race */
+  int racestars[4 /* NRACES */ ];	/* hold the home stars of each race */
+  int attempts = 0;
+  int n;
+
+  for (i = 0; i < NUMPLANETS; i++)
+  {
+    /* zero ownership */
+    planets[i].pl_owner = NOBODY;	/* no team owns a star */
+    if ((PL_TYPE(planets[i]) != PLSTAR) && (PL_TYPE(planets[i]) != PLWHOLE))
+      planets[i].pl_hinfo = NOBODY;	/* no race has planet info */
+  }
+
+  for (i = 0; i < 4 && attempts < 1000; i++)
+  {				/* go through races */
+    /* find home planet */
+    attempts++;
+    p = lrand48() % NUMPLANETS;	/* pick random planet */
+    while ((planets[p].pl_system == 0)
+	   || (PL_TYPE(planets[p]) == PLSTAR)
+	   || (planets[p].pl_owner != NOBODY))
+      p = (p + 1) % NUMPLANETS;	/* go on to next planet */
+
+    racestars[i] = planets[p].pl_system - 1;
+    for (j = 0; j < i; j++)
+    {
+      struct planet *mystar = &planets[racestars[i]];
+      struct planet *otherstar = &planets[racestars[j]];
+      float dx, dy;
+      dx = otherstar->pl_x - mystar->pl_x;
+      dy = otherstar->pl_y - mystar->pl_y;
+      if (dx * dx + dy * dy < 3.0 * SYSWIDTH * SYSWIDTH)
+      {
+	break;
+      }
+    }
+    if (j < i)
+    {
+      i--;
+      continue;
+    }
+    planets[p].pl_flags &= ~PLSURMASK;	/* make sure no dilithium */
+    planets[p].pl_flags |= (PLMETAL | PLARABLE);	/* metal and arable */
+    planets[p].pl_flags |= PLATYPE1;	/* good atmosphere */
+    planets[p].pl_flags |= (PLAGRI | PLSHIPYARD | PLREPAIR);
+    planets[p].pl_tagri = PLGAGRI;	/* set timers for resources */
+    planets[p].pl_tshiprepair = PLGSHIP;
+    planets[p].pl_owner = 1 << i;	/* make race the owner */
+#if 0				/* home planets do not have traditional names */
+    strcpy(planets[p].pl_name, homenames[1 << i]);	/* set name and length */
+    planets[p].pl_namelen = strlen(homenames[1 << i]);
+#endif
+    planets[p].pl_armies = HOMEARMIES;	/* set the armies */
+    planets[p].pl_hinfo = 1 << i;	/* race has info on planet */
+    planets[p].pl_tinfo[1 << i].owner = 1 << i;	/* know about owner */
+    planets[p].pl_tinfo[1 << i].armies = planets[p].pl_armies;
+    planets[p].pl_tinfo[1 << i].flags = planets[p].pl_flags;
+
+    /* find colony planet */
+    p = lrand48() % NUMPLANETS;	/* pick random planet */
+    while ((planets[p].pl_system != racestars[i] + 1)
+	   || (PL_TYPE(planets[p]) == PLSTAR)
+	   || (planets[p].pl_owner != NOBODY))
+      p = (p + 1) % NUMPLANETS;	/* go on to next planet */
+    planets[p].pl_flags |= PLFUEL;	/* make fuel depot */
+    planets[p].pl_tfuel = PLGFUEL;	/* set timer for fuel depot */
+    planets[p].pl_flags &= ~PLATMASK;	/* take off previous atmos */
+    planets[p].pl_flags |= PLPOISON;	/* poison atmosphere */
+    planets[p].pl_flags |= PLDILYTH;	/* dilythium deposits */
+    planets[p].pl_owner = 1 << i;	/* make race the owner */
+    planets[p].pl_armies = COLONYARMIES;	/* set the armies */
+    planets[p].pl_hinfo = 1 << i;	/* race knows about */
+    planets[p].pl_tinfo[1 << i].owner = 1 << i;	/* know about owner */
+    planets[p].pl_tinfo[1 << i].armies = planets[p].pl_armies;
+    planets[p].pl_tinfo[1 << i].flags = planets[p].pl_flags;
+    n = 0;
+    for (j = 0; j < NUMPLANETS; j++)
+    {
+      if ((planets[j].pl_system == racestars[i] + 1)
+	  && !(PL_TYPE(planets[j]) == PLSTAR))
+      {
+
+	if (planets[j].pl_owner == NOBODY)
+	{
+	  planets[j].pl_flags &= ~(PLATMASK | PLSURMASK);
+	  planets[j].pl_flags |= (n << PLATSHIFT) | (1 << (n + PLSURSHIFT));
+	  n++;
+	}
+	planets[j].pl_owner = 1 << i;
+	planets[j].pl_hinfo =
+#ifdef LEAGUE_SUPPORT
+	  status2->league ? (1 << 4) - 1 :
+#endif
+	  (1 << i);
+
+#ifdef LEAGUE_SUPPORT
+	for (k = (status2->league ? 0 : i);
+	     k < (status2->league ? 4 : i + 1);
+	     k++)
+#else
+	k = i;
+#endif
+	{
+	  struct teaminfo *info = &planets[j].pl_tinfo[1 << k];
+	  info->owner = 1 << i;
+	  info->armies = planets[j].pl_armies;
+	  info->flags = planets[j].pl_flags;
+	}
+      }
+    }
+  }
+  return i >= 4;
+}
+
+/*
+ * generate a complete galaxy.
+ * 
+ * This algorithm was invented by Robert Forsman.
+ * 
+ * It tries to make a denser universe with fairer race resource distribution and
+ * races that are further apart..
+ */
+
+void
+gen_galaxy_2()
+{
+  int t;
+
+  while (1)
+  {
+    NUMPLANETS = 60;
+    GWIDTH = 200000;
+
+    initplanets();		/* initialize planet structures */
+
+    /* place the resources */
+    zero_plflags(planets, NUMPLANETS);
+    randomize_atmospheres(planets + SYSTEMS, NUMPLANETS - SYSTEMS,
+			  PATMOS1, PATMOS2, PATMOS3, PPOISON);
+    randomize_resources(planets + SYSTEMS, NUMPLANETS - SYSTEMS,
+			NMETAL, NDILYTH, NARABLE);
+
+    if (!place_stars(planets, SYSTEMS,
+		     (int) STARBORD, (int) STARMIN, (int) GW,
+		     (struct planet *) 0, 0))	/* place system centers */
+      continue;
+
+    t = placesystems();		/* place planets in systems */
+    if (!t)
+      continue;
+
+    if (!placeindep(t))		/* place independent planets */
+      continue;
+
+    if (configvals->justify_galaxy)
+      justify_galaxy(SYSTEMS);
+
+    if (!placeraces())		/* place home planets for races */
+      continue;
+    break;			/* success! */
+  }
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/pl_gen3.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,561 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+
+#include "config.h"
+#include <math.h>
+
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "shmem.h"
+#include "planets.h"
+
+#define SYSWIDTH	(GWIDTH/5.9)	/* width of a system */
+
+#define SYSTEMS		9	/* number of planetary systems */
+
+/* atmosphere chances form a cascade win rand()%100 */
+#define PATMOS1		40	/* chance for normal atmosphere */
+#define PATMOS2		70	/* chance for thin atmosphere */
+#define PATMOS3		90	/* chance for slightly toxic stmos */
+#define PPOISON		100	/* chance for poison atmos */
+
+/* defines that deal with planets resources and types */
+#define NMETAL		13	/* number of metal deposits */
+#define NDILYTH		10	/* number of dilythium deposits */
+#define NARABLE		15	/* number of arable land planets */
+/* defines that deal with star placement */
+
+#define GW	((float)GWIDTH)	/* size of galaxy in floating point */
+#define	STARBORD	(GW*0.27)
+#define TEAMBORD	(GW*0.32)
+#define STARMIN		(GW/5.6)/* min dist between stars */
+#define STARMAX		GW
+#define TEAMMIN		(GW/2.8)/* min dist between team stars */
+#define TEAMMAX		(GW/1.8)/* max dist between team stars */
+
+/* defines that deal with systems and their planets */
+#define SYSADD		2	/* number possible above min number */
+#define SYSBORD		(7000.0 + (float)GWIDTH/200)	/* min distance from
+							 * border wall */
+#define INDBORD		(GW*0.23)
+#define SYSMIN		(5500.0 + (float)GWIDTH/200)	/* min distance between
+							 * objects */
+#define SYSMIN2		(SYSMIN*SYSMIN)	/* square of sysmin distance */
+#define SYSPLMIN	5	/* min number of planets for system */
+#define SYSPLADD	0	/* number of possible extra planets */
+#define MINARMY 8		/* min numer of armies on a planet */
+#define MAXARMY 15		/* max number of armies on a planet */
+
+/* other defines */
+#define HOMEARMIES 30		/* number of armies on home planets */
+#define COLONYARMIES 10		/* number of armies for colony planet */
+
+
+/* defines dealing with growth timers */
+#define PLGFUEL		configvals->plgrow.fuel	/* time for growth of fuel
+						 * depot */
+#define PLGAGRI		configvals->plgrow.agri	/* time for growth of agri */
+#define PLGREPAIR	configvals->plgrow.repair	/* time for growth of
+							 * repair */
+#define PLGSHIP		configvals->plgrow.shipyard	/* time for growth of
+							 * shipyard */
+
+
+#if 0
+/*-------------------------------GENRESOURCES----------------------------*/
+/*
+ * This function goes through the planets structure and determines what kind
+ * of atmosphere and what kind of surface the planets have.  It generates the
+ * stars that will be used as system centers ans then places atmospheres on
+ * the other planets.  It then distributes the resources on the planet
+ * surfaces.
+ */
+
+static void
+genresources()
+{
+  int i;			/* looping vars */
+  int t;			/* temp var */
+
+  for (i = 0; i < SYSTEMS; i++)	/* first planets are stars */
+    planets[i].pl_flags |= PLSTAR;	/* or in star flag */
+  for (i = SYSTEMS; i < NUMPLANETS; i++)
+  {				/* generate atmospheres */
+    t = lrand48() % 100;	/* random # 0-99 */
+    if (t < PATMOS1)		/* is it atmosphere type 1 */
+      planets[i].pl_flags |= PLATYPE1;
+    else if (t < PATMOS2)	/* is it atmosphere type 2 */
+      planets[i].pl_flags |= PLATYPE2;
+    else if (t < PATMOS3)	/* is it atmosphere type 3 */
+      planets[i].pl_flags |= PLATYPE3;
+    else if (t < PPOISON)	/* is it poison atmosphere */
+      planets[i].pl_flags |= PLPOISON;
+  }
+  for (i = 0; i < NMETAL; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % (NUMPLANETS - SYSTEMS) + SYSTEMS;	/* random planet */
+    planets[t].pl_flags |= PLMETAL;	/* OR in the metal flag */
+    if (!configvals->resource_bombing)
+      planets[t].pl_flags |= PLREPAIR;
+  }
+  for (i = 0; i < NDILYTH; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % (NUMPLANETS - SYSTEMS) + SYSTEMS;	/* random planet */
+    planets[t].pl_flags |= PLDILYTH;	/* OR in the dilyth flag */
+    planets[t].pl_flags &= ~(PLATMASK | PLARABLE);	/* zero off previous
+							 * atmos */
+    planets[t].pl_flags |= PLPOISON;	/* dilyth poisons atmosphere */
+    if (!configvals->resource_bombing)
+      planets[t].pl_flags |= PLFUEL;
+  }
+  for (i = 0; i < NARABLE; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % (NUMPLANETS - SYSTEMS) + SYSTEMS;	/* random planet */
+    planets[t].pl_flags |= PLARABLE | PLATYPE1;	/* OR in the arable flag */
+    if (!configvals->resource_bombing)
+      planets[t].pl_flags |= PLAGRI;
+  }
+}
+#endif
+
+
+
+#if 0
+/*--------------------------------PLACESTARS------------------------------*/
+/*
+ * This function places each system's star.  The stars are expected to be in
+ * the first SYSTEMS number of planets.  The coordinates of the stars are
+ * placed in the space grid.
+ */
+
+static int
+placestars()
+{
+  int i, j;			/* looping vars */
+  double x, y;			/* to hold star coordinates */
+  int done;			/* flag to indicate done */
+  double dx, dy;		/* delta x and y's */
+  int attempts;
+  double min, max, dist, bord, nbwidth;
+  double xoff, yoff;
+
+  for (i = 0; i < SYSTEMS; i++)
+  {				/* star for each system */
+    if (i < 2)
+    {				/* choose optimal position for first two */
+      min = TEAMMIN2;
+      max = TEAMMAX2;
+      bord = TEAMBORD;
+    }
+    else if (i < 4)
+    {
+      min = TEAMMIN2;
+      max = STARMAX2;
+      bord = STARBORD * 0.8;
+    }
+    else
+    {
+      min = STARMIN2;
+      max = STARMAX2;
+      bord = STARBORD;
+    }
+    nbwidth = GW - 2 * bord;
+    x = drand48() * nbwidth + bord;	/* pick intial coords */
+    y = drand48() * nbwidth + bord;
+    xoff = 3574.0 - bord;
+    yoff = 1034.0 - bord;
+    attempts = 0;
+    do
+    {				/* do until location found */
+      attempts++;
+      done = 0;			/* not done yet */
+      x = bord + fmod(x + xoff, nbwidth);	/* offset coords a little */
+      y = bord + fmod(y + yoff, nbwidth);	/* every loop */
+#if 0
+      if ((x > GW - bord) || (x < bord)
+	  || (y < bord) || (y > GW - bord))
+	continue;		/* too close to border? */
+#endif
+      done = 1;			/* assume valid cord found */
+      for (j = 0; j < i; j++)
+      {				/* go through previous stars */
+	dx = fabs(x - (double) planets[j].pl_x);
+	dy = fabs(y - (double) planets[j].pl_y);
+	dist = dx * dx + dy * dy;
+	if (dist < min || dist > max)	/* if too close or too far then */
+	  done = 0;		/* we must get another coord */
+      }
+    } while (!done && attempts < 1000);	/* do until location found */
+
+    if (!done)
+      return 0;
+
+    planets[i].pl_owner = NOBODY;	/* no team owns a star */
+    planets[i].pl_flags |= PLSTAR;	/* mark planet as a star */
+    move_planet(i, (int) x, (int) y, 0);
+    planets[i].pl_system = i + 1;	/* mark the sytem number */
+    planets[i].pl_hinfo = ALLTEAM;	/* all teams know its a star */
+    for (j = 0; j < MAXTEAM + 1; j++)
+    {				/* go put in info for teams */
+      planets[i].pl_tinfo[j].owner = NOBODY;	/* nobody owns it */
+      planets[i].pl_tinfo[j].armies = 0;
+      planets[i].pl_tinfo[j].flags = planets[i].pl_flags;
+    }
+  }
+  return 1;
+}
+#endif
+
+
+
+/*-----------------------------PLACESYSTEMS------------------------------*/
+/*
+ * This function places the planets in each star's system.  The function will
+ * return the index of the first planet that was not placed in a system. The
+ * coordinates of the planets are placed in the space grid.
+ */
+
+static int
+placesystems()
+{
+  int i, j, k;			/* looping vars */
+  double x, y;			/* to hold star coordinates */
+  int done;			/* flag to indicate done */
+  double dx, dy;		/* delta x and y's */
+  int n;			/* number of planet to place */
+  int np;			/* number of planets in system */
+  int attempts;
+
+  n = SYSTEMS;			/* first planet to place */
+  for (i = 0; i < SYSTEMS; i++)
+  {				/* planets for each system */
+    np = SYSPLMIN + lrand48() % (SYSPLADD + 1);	/* how many planets */
+    for (k = 0; k < np; k++)
+    {				/* go place the planets */
+      attempts = 0;
+      do
+      {				/* do until location found */
+	attempts++;
+	done = 0;		/* not done yet */
+	dx = (drand48() * SYSWIDTH - SYSWIDTH / 2.0);
+	dy = (drand48() * SYSWIDTH - SYSWIDTH / 2.0);
+	if (dx * dx + dy * dy > (SYSWIDTH / 2.0) * (SYSWIDTH / 2.0))
+	  continue;		/* might orbit its way out of the galaxy */
+	x = planets[i].pl_x + dx;
+	y = planets[i].pl_y + dy;
+	if ((x > GW - SYSBORD) || (x < SYSBORD)
+	    || (y < SYSBORD) || (y > GW - SYSBORD))
+	  continue;		/* too close to border? */
+
+	done = 1;		/* assume valid coord found */
+	for (j = 0; j < n; j++)
+	{			/* go through previous planets */
+	  dx = fabs(x - (double) planets[j].pl_x);
+	  dy = fabs(y - (double) planets[j].pl_y);
+	  if (dx * dx + dy * dy < SYSMIN2)
+	  {			/* if too close to another star */
+	    done = 0;		/* we must get another coord */
+	  }
+	}
+      } while (!done && attempts < 200);	/* do until location found */
+
+      if (!done)
+	return 0;		/* universe too crowded, try again */
+
+      move_planet(n, (int) x, (int) y, 0);
+      planets[n].pl_system = i + 1;	/* mark the sytem number */
+      planets[n].pl_armies = MINARMY + lrand48() % (MAXARMY - MINARMY);
+      n++;			/* go to next planet */
+    }
+  }
+  return (n);			/* return index of next planet */
+}
+
+
+
+
+/*-----------------------------PLACEINDEP------------------------------*/
+/*
+ * This function places idependent planets that are not in a system. They can
+ * appear anywhere in the galaxy as long as they are not too close to another
+ * planet.  The coords are put in the space grid.
+ */
+
+static int
+placeindep(n)
+  int n;
+/* number of planet to start with */
+{
+  int i, j;			/* looping vars */
+  double x, y;			/* to hold star coordinates */
+  int done;			/* flag to indicate done */
+  double dx, dy;		/* delta x and y's */
+  int attempts;
+
+  for (i = n; i < (NUMPLANETS - (WORMPAIRS * 2)); i++)
+  {				/* go through rest of planets */
+    x = drand48() * (GW - 2 * INDBORD) + INDBORD;	/* pick initial coords */
+    y = drand48() * (GW - 2 * INDBORD) + INDBORD;
+    attempts = 0;
+    do
+    {				/* do until location found */
+      attempts++;
+      done = 0;			/* not done yet */
+      x = INDBORD + fmod(x + (3574.0 - INDBORD), GW - 2 * INDBORD);	/* offset coords a
+									 * little */
+      y = INDBORD + fmod(y + (1034.0 - INDBORD), GW - 2 * INDBORD);	/* every loop */
+#if 0
+      if ((x > GW - INDBORD) || (x < INDBORD)
+	  || (y < INDBORD) || (y > GW - INDBORD))
+	continue;		/* too close to border? */
+#endif
+      done = 1;			/* assume valid coord */
+      for (j = 0; j < n; j++)
+      {				/* go through previous planets */
+	dx = fabs(x - (double) planets[j].pl_x);
+	dy = fabs(y - (double) planets[j].pl_y);
+	if (dx * dx + dy * dy < SYSMIN2)
+	{			/* if planet to close */
+	  done = 0;		/* we must get another coord */
+	}
+      }
+    } while (!done && attempts < 200);	/* do until location found */
+
+    if (!done)
+      return 0;
+
+    move_planet(n, (int) x, (int) y, 0);
+    planets[n].pl_system = 0;	/* mark the no sytem */
+    planets[n].pl_armies = MINARMY + lrand48() % (MAXARMY - MINARMY);
+    n++;			/* go to next planet */
+  }
+  for (i = n; i < NUMPLANETS; i++)	/* now place wormholes */
+  {
+    x = drand48() * GW;		/* pick intial coords */
+    y = drand48() * GW;
+    attempts = 0;
+    do
+    {				/* do until location found */
+      attempts++;
+      done = 0;			/* not done yet */
+      x = fmod(x + 3574.0, GW);	/* offset coords a little */
+      y = fmod(y + 1034.0, GW);	/* every loop */
+#if 0
+      if ((x > GW) || (y > GW))
+	continue;		/* too close to border? */
+#endif
+      done = 1;			/* assume valid coord */
+      for (j = 0; j < n; j++)
+      {				/* go through previous planets */
+	dx = fabs(x - (double) planets[j].pl_x);
+	dy = fabs(y - (double) planets[j].pl_y);
+	if (dx * dx + dy * dy < SYSMIN2)
+	{			/* if planet to close */
+	  done = 0;		/* we must get another coord */
+	}
+      }
+    } while (!done && attempts < 200);	/* do until location found */
+
+    if (!done)
+      return 0;
+
+    move_planet(n, (int) x, (int) y, 0);
+    planets[n].pl_system = 0;	/* mark the no system */
+    planets[n].pl_flags |= PLWHOLE;	/* mark the planet as a wormhole */
+    /* the armies in a wormhole is the other wormhole's x coord */
+    /* the radius is the other wormhole's y coord */
+    if (NUMPLANETS % 2)
+    {
+      if (!(n % 2))
+      {
+	planets[n].pl_armies = planets[n - 1].pl_x;
+	planets[n].pl_radius = planets[n - 1].pl_y;
+	planets[n - 1].pl_armies = planets[n].pl_x;
+	planets[n - 1].pl_radius = planets[n].pl_y;
+      }
+    }
+    else
+    {
+      if (n % 2)
+      {
+	planets[n].pl_armies = planets[n - 1].pl_x;
+	planets[n].pl_radius = planets[n - 1].pl_y;
+	planets[n - 1].pl_armies = planets[n].pl_x;
+	planets[n - 1].pl_radius = planets[n].pl_y;
+      }
+    }
+    planets[i].pl_owner = NOBODY;	/* no team owns a star */
+    planets[i].pl_hinfo = ALLTEAM;	/* all teams know its a star */
+    for (j = 0; j < MAXTEAM + 1; j++)
+    {				/* go put in info for teams */
+      planets[i].pl_tinfo[j].owner = NOBODY;	/* nobody owns it */
+      planets[i].pl_tinfo[j].armies = 0;
+      planets[i].pl_tinfo[j].flags = planets[i].pl_flags;
+    }
+    n++;			/* go to next planet */
+  }
+  return 1;
+}
+
+
+
+
+/*---------------------------------PLACERACES------------------------------*/
+/*
+ * This function places the races in the galaxy.  Each race is placed in a
+ * different system.  The race is given a home world with an Agri and Ship-
+ * yard on it and HOMEARMIES.  They are also given a conoly planet with
+ * dilythium deposits and COLONYARMIES on it.
+ */
+
+static void
+placeraces()
+{
+  int i, j, k;			/* looping vars */
+  int p;			/* to hold planet for race */
+  int r[4], t;
+
+  r[0] = r[1] = lrand48() % 4;	/* pick two races at random.  They will be */
+  while (r[0] == r[1])		/* the races whose systems are 'optimally' */
+    r[1] = lrand48() % 4;	/* placed. */
+  i = 0;
+  while (i == r[0] || i == r[1])
+    i++;
+  r[2] = i++;
+  while (i == r[0] || i == r[1])
+    i++;
+  r[3] = i;
+  status2->nontteamlock = (1 << r[0]) | (1 << r[1]);	/* only allow these
+							 * teams */
+
+  for (i = 0; i < 4; i++)
+  {				/* go through races */
+    t = r[i];			/* which team */
+    p = lrand48() % NUMPLANETS;	/* pick random planet */
+    while ((planets[p].pl_system != i + 1)
+	   || (PL_TYPE(planets[p]) == PLSTAR)
+	   || (planets[p].pl_owner != NOBODY))
+      p = (p + 1) % NUMPLANETS;	/* go on to next planet */
+
+    planets[p].pl_flags &= ~PLSURMASK;	/* make sure no dilithium */
+    planets[p].pl_flags |= (PLMETAL | PLARABLE);	/* metal and arable */
+    planets[p].pl_flags |= PLATYPE1;	/* good atmosphere */
+    planets[p].pl_flags |= (PLAGRI | PLSHIPYARD | PLREPAIR);
+    planets[p].pl_tagri = PLGAGRI;	/* set timers for resources */
+    planets[p].pl_tshiprepair = PLGSHIP;
+    planets[p].pl_owner = 1 << t;	/* make race the owner */
+    planets[p].pl_armies = HOMEARMIES;	/* set the armies */
+    planets[p].pl_hinfo = 1 << t;	/* race has info on planet */
+    planets[p].pl_tinfo[1 << t].owner = 1 << t;	/* know about owner */
+    planets[p].pl_tinfo[1 << t].armies = planets[p].pl_armies;
+    planets[p].pl_tinfo[1 << t].flags = planets[p].pl_flags;
+    /* find colony planet */
+    p = lrand48() % NUMPLANETS;	/* pick random planet */
+    while ((planets[p].pl_system != i + 1)
+	   || (PL_TYPE(planets[p]) == PLSTAR)
+	   || (planets[p].pl_owner != NOBODY))
+      p = (p + 1) % NUMPLANETS;	/* go on to next planet */
+    planets[p].pl_flags |= PLFUEL;	/* make fuel depot */
+    planets[p].pl_tfuel = PLGFUEL;	/* set timer for fuel depot */
+    planets[p].pl_flags &= ~PLATMASK;	/* take off previous atmos */
+    planets[p].pl_flags |= PLPOISON;	/* poison atmosphere */
+    planets[p].pl_flags |= PLDILYTH;	/* dilythium deposits */
+    planets[p].pl_owner = 1 << t;	/* make race the owner */
+    planets[p].pl_armies = COLONYARMIES;	/* set the armies */
+    planets[p].pl_hinfo = 1 << t;	/* race knows about */
+    planets[p].pl_tinfo[1 << t].owner = 1 << t;	/* know about owner */
+    planets[p].pl_tinfo[1 << t].armies = planets[p].pl_armies;
+    planets[p].pl_tinfo[1 << t].flags = planets[p].pl_flags;
+    for (j = 0; j < NUMPLANETS; j++)
+    {
+      if ((planets[j].pl_system == i + 1) && (PL_TYPE(planets[j]) != PLSTAR))
+      {
+#ifdef LEAGUE_SUPPORT
+	for (k = (status2->league ? 0 : t);
+	     k < (status2->league ? 4 : t + 1);
+	     k++)
+#else
+	k = t;
+#endif
+	{
+	  planets[j].pl_owner = 1 << t;
+	  planets[j].pl_hinfo =
+#ifdef LEAGUE_SUPPORT
+	    status2->league ? (1 << 4) - 1 :
+#endif
+	    (1 << t);
+	  planets[j].pl_tinfo[1 << k].owner = 1 << t;
+	  planets[j].pl_tinfo[1 << k].armies = planets[j].pl_armies;
+	  planets[j].pl_tinfo[1 << k].flags = planets[j].pl_flags;
+	}
+      }
+    }
+  }
+}
+
+/*
+ * Generate a complete galaxy. This variation is similar to gen_galaxy_1;
+ * except that it tries to place the races at consistent distances from one
+ * another.
+ */
+
+void
+gen_galaxy_3()
+{
+  int t;
+
+  NUMPLANETS = 60;		/* planets + wormholes */
+  GWIDTH = 200000;
+
+  while (1)
+  {
+    initplanets();		/* initialize planet structures */
+
+    /* place the resources */
+    zero_plflags(planets, NUMPLANETS);
+    randomize_atmospheres(planets + SYSTEMS, NUMPLANETS - SYSTEMS,
+			  PATMOS1, PATMOS2, PATMOS3, PPOISON);
+    randomize_resources(planets + SYSTEMS, NUMPLANETS - SYSTEMS,
+			NMETAL, NDILYTH, NARABLE);
+
+    /* place system centers */
+    t = place_stars(planets, 2,
+		    (int) TEAMBORD, (int) TEAMMIN, (int) TEAMMAX,
+		    (struct planet *) 0, 0)
+      && place_stars(planets + 2, 2,
+		     (int) (STARBORD * 0.8), (int) TEAMMIN, (int) STARMAX,
+		     planets, 2)
+      && place_stars(planets + 4, SYSTEMS - 4,
+		     (int) STARBORD, (int) STARMIN, (int) STARMAX,
+		     planets, 4);
+
+    if (!t)
+      continue;
+    t = placesystems();		/* place planets in systems */
+    if (!t)
+      continue;
+    t = placeindep(t);		/* place independent planets */
+    if (t)
+      break;			/* success */
+  }
+  if (configvals->justify_galaxy)
+    justify_galaxy(SYSTEMS);
+  placeraces();			/* place home planets for races */
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/pl_gen4.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,260 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+#include "config.h"
+#include <math.h>
+#include <memory.h>
+
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "shmem.h"
+#include "planets.h"
+
+#if 0
+#define SYSWIDTH	(GWIDTH/5.75)	/* width of a system */
+#endif
+
+#define SYSTEMS		0	/* number of planetary systems */
+
+/* atmosphere chances form a cascade win rand()%100 */
+#define PATMOS1		100	/* chance for normal atmosphere */
+#define PATMOS2		100	/* chance for thin atmosphere */
+#define PATMOS3		100	/* chance for slightly toxic stmos */
+#define PPOISON		100	/* chance for poison atmos */
+
+/* defines that deal with planets resources and types */
+#define NMETAL		8	/* number of metal deposits */
+#define NDILYTH		12	/* number of dilythium deposits */
+#define NARABLE		8	/* number of arable land planets */
+
+#define MINARMY 8		/* min numer of armies on a planet */
+#define MAXARMY 15		/* max number of armies on a planet */
+
+/* other defines */
+#define HOMEARMIES 30		/* number of armies on home planets */
+#define COLONYARMIES 10		/* number of armies for colony planet */
+
+
+/* defines dealing with growth timers */
+#define PLGFUEL		configvals->plgrow.fuel	/* time for growth of fuel
+						 * depot */
+#define PLGAGRI		configvals->plgrow.agri	/* time for growth of agri */
+#define PLGREPAIR	configvals->plgrow.repair	/* time for growth of
+							 * repair */
+#define PLGSHIP		configvals->plgrow.shipyard	/* time for growth of
+							 * shipyard */
+
+
+/*-------------------------------INITBRONCO------------------------------*/
+/*
+ * Initializes the planet array the way normaltrek did it -- not much
+ * variety, but some people dig playing chess from the same setup over and
+ * over again too. :)
+ */
+static void
+initbronco()
+{
+  int i, j;
+
+  static struct planet pdata[MAXPLANETS] = {
+    {0, (FED | PLHOME | PLCORE | PLFUEL | PLREPAIR | PLSHIPYARD | PLMETAL | PLARABLE | PLDILYTH),
+      FED, 20000, 80000, 0, 0, 0, "Earth", 5,
+    (ROM | KLI | ORI), 0, 0, 0, 30, 0, FED},
+    {1, FED | PLCORE, FED, 30000, 90000, 0, 0, 0, "Deneb", 5,
+    (ROM | KLI | ORI), 0, 0, 0, 30, 0, FED},
+    {2, FED | PLCORE, FED, 11000, 75000, 0, 0, 0, "Altair", 6,
+    (ROM | KLI | ORI), 0, 0, 0, 30, 0, FED},
+    {3, FED | PLCORE, FED, 8000, 93000, 0, 0, 0, "Vega", 4,
+    (ROM | KLI | ORI), 0, 0, 0, 30, 0, FED},
+    {4, FED, FED, 10000, 60000, 0, 0, 0, "Rigel", 5,
+    (ROM | KLI | ORI), 0, 0, 0, 30, 0, FED},
+    {5, FED, FED, 25000, 60000, 0, 0, 0, "Canopus", 7,
+    (ROM | KLI | ORI), 0, 0, 0, 30, 0, FED},
+    {6, FED, FED, 44000, 81000, 0, 0, 0, "Beta Crucis", 11,
+    (ROM | KLI | ORI), 0, 0, 0, 30, 0, FED},
+    {7, FED, FED, 39000, 55000, 0, 0, 0, "Organia", 7,
+    (ROM | KLI | ORI), 0, 0, 0, 30, 0, FED},
+    {8, FED, FED, 45000, 66000, 0, 0, 0, "Ceti Alpha V", 12,
+    (ROM | KLI | ORI), 0, 0, 0, 30, 0, FED},
+    {9, FED, FED, 32000, 74000, 0, 0, 0, "Alpha Centauri", 14,
+    (ROM | KLI | ORI), 0, 0, 0, 30, 0, FED},
+    {10, (ROM | PLHOME | PLCORE | PLFUEL | PLREPAIR | PLSHIPYARD | PLMETAL | PLARABLE | PLDILYTH),
+      ROM, 20000, 20000, 0, 0, 0, "Romulus", 7,
+    (FED | KLI | ORI), 0, 0, 0, 30, 0, ROM},
+    {11, ROM | PLCORE, ROM, 28000, 8000, 0, 0, 0, "Tauri", 5,
+    (FED | KLI | ORI), 0, 0, 0, 30, 0, ROM},
+    {12, ROM | PLCORE, ROM, 28000, 23000, 0, 0, 0, "Draconis", 8,
+    (FED | KLI | ORI), 0, 0, 0, 30, 0, ROM},
+    {13, ROM | PLCORE, ROM, 4000, 12000, 0, 0, 0, "Aldeberan", 9,
+    (FED | KLI | ORI), 0, 0, 0, 30, 0, ROM},
+    {14, ROM, ROM, 45000, 7000, 0, 0, 0, "Eridani", 7,
+    (FED | KLI | ORI), 0, 0, 0, 30, 0, ROM},
+    {15, ROM, ROM, 42000, 44000, 0, 0, 0, "Regulus", 7,
+    (FED | KLI | ORI), 0, 0, 0, 30, 0, ROM},
+    {16, ROM, ROM, 13000, 45000, 0, 0, 0, "Capella", 7,
+    (FED | KLI | ORI), 0, 0, 0, 30, 0, ROM},
+    {17, ROM, ROM, 40000, 25000, 0, 0, 0, "Sirius", 6,
+    (FED | KLI | ORI), 0, 0, 0, 30, 0, ROM},
+    {18, ROM, ROM, 25000, 44000, 0, 0, 0, "Indi", 4,
+    (FED | KLI | ORI), 0, 0, 0, 30, 0, ROM},
+    {19, ROM, ROM, 8000, 29000, 0, 0, 0, "Hydrae", 6,
+    (FED | KLI | ORI), 0, 0, 0, 30, 0, ROM},
+    {20, (KLI | PLHOME | PLCORE | PLFUEL | PLREPAIR | PLSHIPYARD | PLMETAL | PLARABLE | PLDILYTH),
+      KLI, 80000, 20000, 0, 0, 0, "Klingus", 7,
+    (FED | ROM | ORI), 0, 0, 0, 30, 0, KLI},
+    {21, KLI | PLCORE, KLI, 88000, 12000, 0, 0, 0, "Pollux", 6,
+    (FED | ROM | ORI), 0, 0, 0, 30, 0, KLI},
+    {22, KLI | PLCORE, KLI, 69000, 31000, 0, 0, 0, "Scorpii", 7,
+    (FED | ROM | ORI), 0, 0, 0, 30, 0, KLI},
+    {23, KLI | PLCORE, KLI, 73000, 5000, 0, 0, 0, "Castor", 6,
+    (FED | ROM | ORI), 0, 0, 0, 30, 0, KLI},
+    {24, KLI, KLI, 70000, 40000, 0, 0, 0, "Pleiades V", 10,
+    (FED | ROM | ORI), 0, 0, 0, 30, 0, KLI},
+    {25, KLI, KLI, 60000, 10000, 0, 0, 0, "Andromeda", 9,
+    (FED | ROM | ORI), 0, 0, 0, 30, 0, KLI},
+    {26, KLI, KLI, 54000, 40000, 0, 0, 0, "Lalande", 7,
+    (FED | ROM | ORI), 0, 0, 0, 30, 0, KLI},
+    {27, KLI, KLI, 90000, 37000, 0, 0, 0, "Lyrae", 5,
+    (FED | ROM | ORI), 0, 0, 0, 30, 0, KLI},
+    {28, KLI, KLI, 83000, 48000, 0, 0, 0, "Mira", 4,
+    (FED | ROM | ORI), 0, 0, 0, 30, 0, KLI},
+    {29, KLI, KLI, 54000, 21000, 0, 0, 0, "Cygni", 5,
+    (FED | ROM | ORI), 0, 0, 0, 30, 0, KLI},
+    {30, (ORI | PLHOME | PLCORE | PLFUEL | PLREPAIR | PLSHIPYARD | PLMETAL | PLARABLE | PLDILYTH),
+      ORI, 80000, 80000, 0, 0, 0, "Orion", 5,
+    (FED | ROM | KLI), 0, 0, 0, 30, 0, ORI},
+    {31, ORI | PLCORE, ORI, 72000, 69000, 0, 0, 0, "Procyon", 7,
+    (FED | ROM | KLI), 0, 0, 0, 30, 0, ORI},
+    {32, ORI | PLCORE, ORI, 91000, 94000, 0, 0, 0, "Ursae Majoris", 13,
+    (FED | ROM | KLI), 0, 0, 0, 30, 0, ORI},
+    {33, ORI | PLCORE, ORI, 85000, 70000, 0, 0, 0, "Antares", 7,
+    (FED | ROM | KLI), 0, 0, 0, 30, 0, ORI},
+    {34, ORI, ORI, 92000, 59000, 0, 0, 0, "Cassiopia", 9,
+    (FED | ROM | KLI), 0, 0, 0, 30, 0, ORI},
+    {35, ORI, ORI, 65000, 55000, 0, 0, 0, "El Nath", 7,
+    (FED | ROM | KLI), 0, 0, 0, 30, 0, ORI},
+    {36, ORI, ORI, 52000, 60000, 0, 0, 0, "Spica", 5,
+    (FED | ROM | KLI), 0, 0, 0, 30, 0, ORI},
+    {37, ORI, ORI, 64000, 80000, 0, 0, 0, "Polaris", 7,
+    (FED | ROM | KLI), 0, 0, 0, 30, 0, ORI},
+    {38, ORI, ORI, 56000, 89000, 0, 0, 0, "Arcturus", 8,
+    (FED | ROM | KLI), 0, 0, 0, 30, 0, ORI},
+    {39, ORI, ORI, 70000, 93000, 0, 0, 0, "Herculis", 8,
+    (FED | ROM | KLI), 0, 0, 0, 30, 0, ORI}
+  };
+  for (i = 0; i < NUMPLANETS; i++)
+  {
+    for (j = 0; j < MAXTEAM + 1; j++)
+    {
+      pdata[i].pl_tinfo[j].owner = pdata[i].pl_owner;
+      pdata[i].pl_tinfo[j].armies = pdata[i].pl_armies;
+      pdata[i].pl_tinfo[j].flags = pdata[i].pl_flags;
+      pdata[i].pl_tinfo[j].timestamp = 0;
+    }
+    pdata[i].pl_trevolt = 0;
+    pdata[i].pl_next = 0;
+    pdata[i].pl_prev = 0;
+    pdata[i].pl_gridnum = 0;
+  }
+  memcpy((char *) planets, (char *) pdata, MAXPLANETS * sizeof(struct planet));
+}
+/*-------------------------------GENRESOURCES----------------------------*/
+/*
+ * This function goes through the planets structure and determines what kind
+ * of atmosphere and what kind of surface the planets have.  It generates the
+ * stars that will be used as system centers ans then places atmospheres on
+ * the other planets.  It then distributes the resources on the planet
+ * surfaces.  This version's been bronco-ified. :)
+ */
+
+static void
+genresources_bronco()
+{
+  int i, j;			/* looping vars */
+  int t;			/* temp var */
+
+  for (i = SYSTEMS; i < NUMPLANETS; i++)
+  {				/* generate atmospheres */
+    t = lrand48() % 100;	/* random # 0-99 */
+    if (t < PATMOS1)		/* is it atmosphere type 1 */
+      planets[i].pl_flags |= PLATYPE1;
+    else if (t < PATMOS2)	/* is it atmosphere type 2 */
+      planets[i].pl_flags |= PLATYPE2;
+    else if (t < PATMOS3)	/* is it atmosphere type 3 */
+      planets[i].pl_flags |= PLATYPE3;
+    else if (t < PPOISON)	/* is it poison atmosphere */
+      planets[i].pl_flags |= PLPOISON;
+  }
+  for (i = 0; i < NMETAL; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % ((NUMPLANETS - SYSTEMS) / 4) + SYSTEMS +
+      ((i / (NMETAL / 4)) * ((NUMPLANETS - SYSTEMS) / 4));
+    if (!(planets[t].pl_flags & PLMETAL & PLSURMASK))
+    {
+      planets[t].pl_flags |= PLMETAL;	/* OR in the metal flag */
+      planets[t].pl_flags |= PLREPAIR;
+    }
+    else
+      i--;
+  }
+  for (i = 0; i < NDILYTH; i++)
+  {				/* place the crystals */
+    t = lrand48() % ((NUMPLANETS - SYSTEMS) / 4) + SYSTEMS +
+      ((i / (NDILYTH / 4)) * ((NUMPLANETS - SYSTEMS) / 4));
+    if (!(planets[t].pl_flags & PLDILYTH & PLSURMASK))
+    {
+      planets[t].pl_flags |= PLDILYTH;	/* OR in the dilyth flag */
+      planets[t].pl_flags |= PLFUEL;
+    }
+    else
+      i--;
+  }
+  for (i = 0; i < NARABLE; i++)
+  {				/* place the farms */
+    t = lrand48() % ((NUMPLANETS - SYSTEMS) / 4) + SYSTEMS +
+      ((i / (NARABLE / 4)) * ((NUMPLANETS - SYSTEMS) / 4));
+    if (!(planets[t].pl_flags & PLARABLE & PLSURMASK) &&
+	!(planets[t].pl_flags & PLHOME))
+    {
+      planets[t].pl_flags |= PLARABLE | PLATYPE1;	/* OR in the arable flag */
+      planets[t].pl_flags |= PLAGRI;
+    }
+    else
+      i--;
+  }
+  for (i = 0; i < NUMPLANETS; i++)
+    for (j = 0; j < MAXTEAM + 1; j++)
+      if (j == planets[i].pl_owner)
+	planets[i].pl_tinfo[j].flags = planets[i].pl_flags;
+}
+
+
+
+/*
+ * Generate a complete galaxy. Uses ye old bronco planet setup.
+ */
+
+void
+gen_galaxy_4()
+{
+  GWIDTH = 100000;
+  NUMPLANETS = 40;
+  initbronco();			/* initialize planet structures */
+  genresources_bronco();	/* place the resources */
+  return;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/pl_gen5.c	Sat Dec 06 04:37:03 1997 +0000
@@ -0,0 +1,544 @@
+/*--------------------------------------------------------------------------
+NETREK II -- Paradise
+
+Permission to use, copy, modify, and distribute this software and its
+documentation, or any derivative works thereof, for any NON-COMMERCIAL
+purpose and without fee is hereby granted, provided that this copyright
+notice appear in all copies.  No representations are made about the
+suitability of this software for any purpose.  This software is provided
+"as is" without express or implied warranty.
+
+    Xtrek Copyright 1986                            Chris Guthrie
+    Netrek (Xtrek II) Copyright 1989                Kevin P. Smith
+                                                    Scott Silvey
+    Paradise II (Netrek II) Copyright 1993          Larry Denys
+                                                    Kurt Olsen
+                                                    Brandon Gillespie
+--------------------------------------------------------------------------*/
+#include "config.h"
+#include <math.h>
+
+#include "defs.h"
+#include "struct.h"
+#include "data.h"
+#include "shmem.h"
+#include "planets.h"
+
+#define SYSWIDTH	(GWIDTH/5.75)	/* width of a system */
+
+#define SYSTEMS		7	/* number of planetary systems */
+
+/* atmosphere chances form a cascade win rand()%100 */
+#define PATMOS1		40	/* chance for normal atmosphere */
+#define PATMOS2		70	/* chance for thin atmosphere */
+#define PATMOS3		90	/* chance for slightly toxic stmos */
+#define PPOISON		100	/* chance for poison atmos */
+
+/* defines that deal with planets resources and types */
+#define NMETAL		10	/* number of metal deposits */
+#define NDILYTH		8	/* number of dilythium deposits */
+#define NARABLE		12	/* number of arable land planets */
+/* defines that deal with star placement */
+
+#define GW	((float)GWIDTH)	/* size of galaxy in floating point */
+#define	STARBORD	((GW/5.2)*1.3)
+#define TEAMBORD	((GW/5.2)/1.1)
+#define STARMIN		(GW/5.0)/* min dist between stars */
+#define STARMIN2	(STARMIN*STARMIN)	/* min star dist squared */
+#define STARMAX		GW
+#define STARMAX2	(GW*GW)
+#define TEAMMIN		(GW/2.8)/* min dist between team stars */
+#define TEAMMIN2	(TEAMMIN*TEAMMIN)
+#define TEAMMAX		(GW/1.4)/* max dist between team stars */
+#define TEAMMAX2	(TEAMMAX*TEAMMAX)
+
+/* defines that deal with systems and their planets */
+#define SYSADD		2	/* number possible above min number */
+#define SYSBORD		(4000.0 + (float)GWIDTH/200)	/* min distance from
+							 * border wall */
+#define INDBORD		(GW * 0.1)
+#define SYSMIN		(6000.0 + (float)GWIDTH/200)	/* min distance between
+							 * objects */
+#define SYSMIN2		(SYSMIN*SYSMIN)	/* square of sysmin distance */
+#define SYSPLMIN	5	/* min number of planets for system */
+#define SYSPLADD	0	/* number of possible extra planets */
+#define MINARMY 8		/* min numer of armies on a planet */
+#define MAXARMY 15		/* max number of armies on a planet */
+
+/* other defines */
+#define HOMEARMIES 30		/* number of armies on home planets */
+#define COLONYARMIES 10		/* number of armies for colony planet */
+
+
+/* defines dealing with growth timers */
+#define PLGFUEL		configvals->plgrow.fuel	/* time for growth of fuel
+						 * depot */
+#define PLGAGRI		configvals->plgrow.agri	/* time for growth of agri */
+#define PLGREPAIR	configvals->plgrow.repair	/* time for growth of
+							 * repair */
+#define PLGSHIP		configvals->plgrow.shipyard	/* time for growth of
+							 * shipyard */
+
+
+#if 0
+/*-------------------------------GENRESOURCES----------------------------*/
+/*
+ * This function goes through the planets structure and determines what kind
+ * of atmosphere and what kind of surface the planets have.  It generates the
+ * stars that will be used as system centers ans then places atmospheres on
+ * the other planets.  It then distributes the resources on the planet
+ * surfaces.
+ */
+
+static void
+genresources()
+{
+  int i;			/* looping vars */
+  int t;			/* temp var */
+
+  for (i = 0; i < SYSTEMS; i++)	/* first planets are stars */
+    planets[i].pl_flags |= PLSTAR;	/* or in star flag */
+  for (i = SYSTEMS; i < NUMPLANETS; i++)
+  {				/* generate atmospheres */
+    t = lrand48() % 100;	/* random # 0-99 */
+    if (t < PATMOS1)		/* is it atmosphere type 1 */
+      planets[i].pl_flags |= PLATYPE1;
+    else if (t < PATMOS2)	/* is it atmosphere type 2 */
+      planets[i].pl_flags |= PLATYPE2;
+    else if (t < PATMOS3)	/* is it atmosphere type 3 */
+      planets[i].pl_flags |= PLATYPE3;
+    else if (t < PPOISON)	/* is it poison atmosphere */
+      planets[i].pl_flags |= PLPOISON;
+  }
+  for (i = 0; i < NMETAL; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % (NUMPLANETS - SYSTEMS) + SYSTEMS;	/* random planet */
+    planets[t].pl_flags |= PLMETAL;	/* OR in the metal flag */
+    if (!configvals->resource_bombing)
+      planets[t].pl_flags |= PLREPAIR;
+  }
+  for (i = 0; i < NDILYTH; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % (NUMPLANETS - SYSTEMS) + SYSTEMS;	/* random planet */
+    planets[t].pl_flags |= PLDILYTH;	/* OR in the dilyth flag */
+    planets[t].pl_flags &= ~(PLATMASK | PLARABLE);	/* zero off previous
+							 * atmos */
+    planets[t].pl_flags |= PLPOISON;	/* dilyth poisons atmosphere */
+    if (!configvals->resource_bombing)
+      planets[t].pl_flags |= PLFUEL;
+  }
+  for (i = 0; i < NARABLE; i++)
+  {				/* place the metal deposits */
+    t = lrand48() % (NUMPLANETS - SYSTEMS) + SYSTEMS;	/* random planet */
+    planets[t].pl_flags |= PLARABLE | PLATYPE1;	/* OR in the arable flag */
+    if (!configvals->resource_bombing)
+      planets[t].pl_flags |= PLAGRI;
+  }
+}
+#endif
+
+
+#if 0
+
+/*--------------------------------PLACESTARS------------------------------*/
+/*
+ * This function places each system's star.  The stars are expected to be in
+ * the first SYSTEMS number of planets.  The coordinates of the stars are
+ * placed in the space grid.
+ */
+
+static int
+placestars()
+{
+  int i, j;			/* looping vars */
+  double x, y;			/* to hold star coordinates */
+  int done;			/* flag to indicate done */
+  double dx, dy;		/* delta x and y's */
+  int attempts;
+  double min, max, dist, bord, nbwidth;
+  double xoff, yoff;
+
+  for (i = 0; i < SYSTEMS; i++)
+  {				/* star for each system */
+    if (i < 4)
+    {
+      min = TEAMMIN2;
+      max = TEAMMAX2;
+      bord = TEAMBORD;
+    }
+    else
+    {
+      min = STARMIN2;
+      max = STARMAX2;
+      bord = STARBORD;
+    }
+    nbwidth = GW - 2 * bord;
+    x = drand48() * nbwidth + bord;	/* pick intial coords */
+    y = drand48() * nbwidth + bord;
+    xoff = 3574.0 - bord;
+    yoff = 1034.0 - bord;
+    attempts = 0;
+    do
+    {				/* do until location found */
+      attempts++;
+      done = 0;			/* not done yet */
+      x = bord + fmod(x + xoff, nbwidth);	/* offset coords a little */
+      y = bord + fmod(y + yoff, nbwidth);	/* every loop */
+#if 0
+      if ((x > GW - bord) || (x < bord)
+	  || (y < bord) || (y > GW - bord))
+	continue;		/* too close to border? */
+#endif
+      done = 1;			/* assume valid cord found */
+      for (j = 0; j < i; j++)
+      {				/* go through previous stars */
+	dx = fabs(x - (double) planets[j].pl_x);
+	dy = fabs(y - (double) planets[j].pl_y);
+	dist = dx * dx + dy * dy;
+	if (dist < min || dist > max)	/* if too close or too far then */
+	  done = 0;		/* we must get another coord */
+      }
+    } while (!done && attempts < 1000);	/* do until location found */
+
+    if (!done)
+      return 0;
+
+    planets[i].pl_owner = NOBODY;	/* no team owns a star */
+    planets[i].pl_flags |= PLSTAR;	/* mark planet as a star */
+    move_planet(i, (int) x, (int) y, 0);
+    planets[i].pl_system = i + 1;	/* mark the sytem number */
+    planets[i].pl_hinfo = ALLTEAM;	/* all teams know its a star */
+    for (j = 0; j < MAXTEAM + 1; j++)
+    {				/* go put in info for teams */
+      planets[i].pl_tinfo[j].owner = NOBODY;	/* nobody owns it */
+      planets[i].pl_tinfo[j].armies = 0;
+      planets[i].pl_tinfo[j].flags = planets[i].pl_flags;
+    }
+  }
+  return 1;
+}
+
+#endif
+
+
+/*-----------------------------PLACESYSTEMS------------------------------*/
+/*
+ * This function places the planets in each star's system.  The function will
+ * return the index of the first planet that was not placed in a system. The
+ * coordinates of the planets are placed in the space grid.
+ */
+
+static int
+placesystems()
+{
+  int i, j, k;			/* looping vars */
+  double x, y;			/* to hold star coordinates */
+  int done;			/* flag to indicate done */
+  double dx, dy;		/* delta x and y's */
+  int n;			/* number of planet to place */
+  int np;			/* number of planets in system */
+  int attempts;
+
+  n = SYSTEMS;			/* first planet to place */
+  for (i = 0; i < SYSTEMS; i++)
+  {				/* planets for each system */
+    np = SYSPLMIN + lrand48() % (SYSPLADD + 1);	/* how many planets */
+    for (k = 0; k < np; k++)
+    {				/* go place the planets */
+      attempts = 0;
+      do
+      {				/* do until location found */
+	attempts++;
+	done = 0;		/* not done yet */
+	dx = (drand48() * SYSWIDTH - SYSWIDTH / 2.0);
+	dy = (drand48() * SYSWIDTH - SYSWIDTH / 2.0);
+	if (dx * dx + dy * dy > (SYSWIDTH / 2.0) * (SYSWIDTH / 2.0))
+	  continue;		/* might orbit its way out of the galaxy */
+	x = planets[i].pl_x + dx;
+	y = planets[i].pl_y + dy;
+	if ((x > GW - SYSBORD) || (x < SYSBORD)
+	    || (y < SYSBORD) || (y > GW - SYSBORD))
+	  continue;		/* too close to border? */
+
+	done = 1;		/* assume valid coord found */
+	for (j = 0; j < n; j++)
+	{			/* go through previous planets */
+	  dx = fabs(x - (double) planets[j].pl_x);
+	  dy = fabs(y - (double) planets[j].pl_y);
+	  if (dx * dx + dy * dy < SYSMIN2)
+	  {			/* if too close to another star */
+	    done = 0;		/* we must get another coord */
+	  }
+	}
+      } while (!done && attempts < 200);	/* do until location found */
+
+      if (!done)
+	return 0;		/* universe too crowded, try again */
+
+      move_planet(n, (int) x, (int) y, 0);
+      planets[n].pl_system = i + 1;	/* mark the sytem number */
+      planets[n].pl_armies = MINARMY + lrand48() % (MAXARMY - MINARMY);
+      n++;			/* go to next planet */
+    }
+  }
+  return (n);			/* return index of next planet */
+}
+
+
+
+
+/*-----------------------------PLACEINDEP------------------------------*/
+/*
+ * This function places idependent planets that are not in a system. They can
+ * appear anywhere in the galaxy as long as they are not too close to another
+ * planet.  The coords are put in the space grid.
+ */
+
+static int
+placeindep(n)
+  int n;
+/* number of planet to start with */
+{
+  int i, j;			/* looping vars */
+  double x, y;			/* to hold star coordinates */
+  int done;			/* flag to indicate done */
+  double dx, dy;		/* delta x and y's */
+  int attempts;
+
+  for (i = n; i < (NUMPLANETS - (WORMPAIRS * 2)); i++)
+  {
+    /* go through rest of planets */
+    x = drand48() * (GW - 2 * INDBORD) + INDBORD;	/* pick intial coords */
+    y = drand48() * (GW - 2 * INDBORD) + INDBORD;
+    attempts = 0;
+    do
+    {				/* do until location found */
+      attempts++;
+      done = 0;			/* not done yet */
+      x = INDBORD + fmod(x + (3574.0 - INDBORD), GW - 2 * INDBORD);	/* offset coords a
+									 * little */
+      y = INDBORD + fmod(y + (1034.0 - INDBORD), GW - 2 * INDBORD);	/* every loop */
+#if 0
+      if ((x > GW - INDBORD) || (x < INDBORD)
+	  || (y < INDBORD) || (y > GW - INDBORD))
+	continue;		/* too close to border? */
+#endif
+      done = 1;			/* assume valid coord */
+      for (j = 0; j < n; j++)
+      {				/* go through previous planets */
+	dx = fabs(x - (double) planets[j].pl_x);
+	dy = fabs(y - (double) planets[j].pl_y);
+	if (dx * dx + dy * dy < SYSMIN2)
+	{			/* if planet to close */
+	  done = 0;		/* we must get another coord */
+	}
+      }
+    } while (!done && attempts < 200);	/* do until location found */
+
+    if (!done)
+      return 0;
+
+    move_planet(n, (int) x, (int) y, 0);
+    planets[n].pl_system = 0;	/* mark the no sytem */
+    planets[n].pl_armies = MINARMY + lrand48() % (MAXARMY - MINARMY);
+    n++;			/* go to next planet */
+  }
+  for (i = n; i < NUMPLANETS; i++)	/* now place wormholes */
+  {
+    x = drand48() * GW;		/* pick intial coords */
+    y = drand48() * GW;
+    attempts = 0;
+    do
+    {				/* do until location found */
+      attempts++;
+      done = 0;			/* not done yet */
+      x = fmod(x + 3574.0, GW);	/* offset coords a little */
+      y = fmod(y + 1034.0, GW);	/* every loop */
+#if 0
+      if ((x > GW) || (y > GW))
+	continue;		/* too close to border? */
+#endif
+      done = 1;			/* assume valid coord */
+      for (j = 0; j < n; j++)
+      {				/* go through previous planets */
+	dx = fabs(x - (double) planets[j].pl_x);
+	dy = fabs(y - (double) planets[j].pl_y);
+	if (dx * dx + dy * dy < SYSMIN2)
+	{			/* if planet to close */
+	  done = 0;		/* we must get another coord */
+	}
+      }
+    } while (!done && attempts < 200);	/* do until location found */
+
+    if (!done)
+      return 0;
+
+    move_planet(n, (int) x, (int) y, 0);
+    planets[n].pl_system = 0;	/* mark the no system */
+    planets[n].pl_flags |= PLWHOLE;	/* mark the planet as a wormhole */
+    /* the armies in a wormhole is the other wormhole's x coord */
+    /* the radius is the other wormhole's y coord */
+    if (NUMPLANETS % 2)
+    {
+      if (!(n % 2))
+      {
+	planets[n].pl_armies = planets[n - 1].pl_x;
+	planets[n].pl_radius = planets[n - 1].pl_y;
+	planets[n - 1].pl_armies = planets[n].pl_x;
+	planets[n - 1].pl_radius = planets[n].pl_y;
+      }
+    }
+    else
+    {
+      if (n % 2)
+      {
+	planets[n].pl_armies = planets[n - 1].pl_x;
+	planets[n].pl_radius = planets[n - 1].pl_y;
+	planets[n - 1].pl_armies = planets[n].pl_x;
+	planets[n - 1].pl_radius = planets[n].pl_y;
+      }
+    }
+    planets[i].pl_owner = NOBODY;	/* no team owns a star */
+    planets[i].pl_hinfo = ALLTEAM;	/* all teams know its a star */
+    for (j = 0; j < MAXTEAM + 1; j++)
+    {				/* go put in info for teams */
+      planets[i].pl_tinfo[j].owner = NOBODY;	/* nobody owns it */
+      planets[i].pl_tinfo[j].armies = 0;
+      planets[i].pl_tinfo[j].flags = planets[i].pl_flags;
+    }
+    n++;			/* go to next planet */
+  }
+  return 1;
+}
+
+
+
+
+/*---------------------------------PLACERACES------------------------------*/
+/*
+ * This function places the races in the galaxy.  Each race is placed in a
+ * different system.  The race is given a home world with an Agri and Ship-
+ * yard on it and HOMEARMIES.  They are also given a conoly planet with
+ * dilythium deposits and COLONYARMIES on it.
+ */
+
+static void
+placeraces()
+{
+  int i, j, k;			/* looping vars */
+  int p;			/* to hold planet for race */
+
+  for (i = 0; i < 4; i++)
+  {				/* go through races */
+    /* find home planet */
+    p = lrand48() % NUMPLANETS;	/* pick random planet */
+    while ((planets[p].pl_system != i + 1)
+	   || (PL_TYPE(planets[p]) == PLSTAR)
+	   || (planets[p].pl_owner != NOBODY))
+      p = (p + 1) % NUMPLANETS;	/* go on to next planet */
+    planets[p].pl_flags &= ~PLSURMASK;	/* make sure no dilithium */
+    planets[p].pl_flags |= (PLMETAL | PLARABLE);	/* metal and arable */
+    planets[p].pl_flags |= PLATYPE1;	/* good atmosphere */
+    planets[p].pl_flags |= (PLAGRI | PLSHIPYARD | PLREPAIR);
+    planets[p].pl_tagri = PLGAGRI;	/* set timers for resources */
+    planets[p].pl_tshiprepair = PLGSHIP;
+    planets[p].pl_owner = 1 << i;	/* make race the owner */
+#if 0				/* home planets do not have traditional names */
+    strcpy(planets[p].pl_name, homenames[1 << i]);	/* set name and length */
+    planets[p].pl_namelen = strlen(homenames[1 << i]);
+#endif
+    planets[p].pl_armies = HOMEARMIES;	/* set the armies */
+    planets[p].pl_hinfo = 1 << i;	/* race has info on planet */
+    planets[p].pl_tinfo[1 << i].owner = 1 << i;	/* know about owner */
+    planets[p].pl_tinfo[1 << i].armies = planets[p].pl_armies;
+    planets[p].pl_tinfo[1 << i].flags = planets[p].pl_flags;
+    /* find colony planet */
+    p = lrand48() % NUMPLANETS;	/* pick random planet */
+    while ((planets[p].pl_system != i + 1)
+	   || (PL_TYPE(planets[p]) == PLSTAR)
+	   || (planets[p].pl_owner != NOBODY))
+      p = (p + 1) % NUMPLANETS;	/* go on to next planet */
+    planets[p].pl_flags |= PLFUEL;	/* make fuel depot */
+    planets[p].pl_tfuel = PLGFUEL;	/* set timer for fuel depot */
+    planets[p].pl_flags &= ~PLATMASK;	/* take off previous atmos */
+    planets[p].pl_flags |= PLPOISON;	/* poison atmosphere */
+    planets[p].pl_flags |= PLDILYTH;	/* dilythium deposits */
+    planets[p].pl_owner = 1 << i;	/* make race the owner */
+    planets[p].pl_armies = COLONYARMIES;	/* set the armies */
+    planets[p].pl_hinfo = 1 << i;	/* race knows about */
+    planets[p].pl_tinfo[1 << i].owner = 1 << i;	/* know about owner */
+    planets[p].pl_tinfo[1 << i].armies = planets[p].pl_armies;
+    planets[p].pl_tinfo[1 << i].flags = planets[p].pl_flags;
+    for (j = 0; j < NUMPLANETS; j++)
+    {
+      if ((planets[j].pl_system == i + 1) && (PL_TYPE(planets[j]) != PLSTAR))
+      {
+#ifdef LEAGUE_SUPPORT
+	for (k = (status2->league ? 0 : i);
+	     k < (status2->league ? 4 : i + 1);
+	     k++)
+#else
+	k = i;
+#endif
+	{
+	  planets[j].pl_owner = 1 << i;
+	  planets[j].pl_hinfo =
+#ifdef LEAGUE_SUPPORT
+	    status2->league ? (1 << 4) - 1 :
+#endif
+	    (1 << i);
+	  planets[j].pl_tinfo[1 << k].owner = 1 << i;
+	  planets[j].pl_tinfo[1 << k].armies = planets[j].pl_armies;
+	  planets[j].pl_tinfo[1 << k].flags = planets[j].pl_flags;
+	}
+      }
+    }
+  }
+}
+
+/*
+ * Generate a complete galaxy.  Generates like generator 3, only in a smaller
+ * environment
+ */
+
+void
+gen_galaxy_5()
+{
+  int t;
+
+  GWIDTH = 100000;
+  NUMPLANETS = 48;
+
+  while (1)
+  {
+    initplanets();		/* initialize planet structures */
+
+    /* place the resources */
+    zero_plflags(planets, NUMPLANETS);
+    randomize_atmospheres(planets + SYSTEMS, NUMPLANETS - SYSTEMS,
+			  PATMOS1, PATMOS2, PATMOS3, PPOISON);
+    randomize_resources(planets + SYSTEMS, NUMPLANETS - SYSTEMS,
+			NMETAL, NDILYTH, NARABLE);
+
+    /* place system centers */
+    t = place_stars(planets, 4,
+		    (int) TEAMBORD, (int) TEAMMIN, (int) TEAMMAX,
+		    (struct planet *) 0, 0)
+      && place_stars(planets + 4, SYSTEMS - 4,
+		     (int) STARBORD, (int) STARMIN, (int) STARMAX,
+		     planets, 4);
+
+    if (!t)
+      continue;
+    t = placesystems();		/* place planets in systems */
+    if (!t)
+      continue;
+    t = placeindep(t);		/* place independent planets */
+    if (t)
+      break;			/* success */
+  }
+  if (configvals->justify_galaxy)
+    justify_galaxy(SYSTEMS);
+  placeraces();			/* place home planets for races */
+
+}