#include <stdio.h>
#ifdef VMS
#include <stat.h>
#else
#include <sys/stat.h>
#endif

/*
 * Note: For SHM version check if image_data or the pixmap is needed
 *       Check if rect, nrects, points and npoints are needed.
 *       scanline_ptr.
 */

static char *rcsid = "$Id: atari_x11.c,v 1.23 1996/07/19 19:52:22 david Exp $";

#ifdef XVIEW
#include <xview/xview.h>
#include <xview/frame.h>
#include <xview/panel.h>
#include <xview/canvas.h>
#include <xview/notice.h>
#include <xview/file_chsr.h>
#endif

#ifdef MOTIF
#include <Xm/MainW.h>
#include <Xm/DrawingA.h>
#include <Xm/MessageB.h>
#include <Xm/FileSB.h>

static XtAppContext app;
static Widget toplevel;
static Widget main_w;
static Widget drawing_area;
static Widget fsel;
#endif

#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/keysym.h>

#ifndef VMS
#include "config.h"
#endif
#include "atari.h"
#include "colours.h"

#ifdef FPS_MONITOR
#include <sys/time.h>

static struct timeval tp;
static struct timezone tzp;

static double basetime;
static int totframes = 0;
static int nframes = 0;
#endif

#ifdef SHM
#include <sys/ipc.h>
#include <sys/shm.h>
#include <X11/extensions/XShm.h>

static XShmSegmentInfo shminfo;
static XImage *image;
extern char *atari_screen;
extern int colour_translation_table[256];
#endif

#ifdef LINUX_JOYSTICK
#include <linux/joystick.h>
#include <fcntl.h>

static int js0;
static int js1;

static int js0_centre_x;
static int js0_centre_y;
static int js1_centre_x;
static int js1_centre_y;

static struct JS_DATA_TYPE js_data;
#endif

#define	FALSE	0
#define	TRUE	1

typedef enum
{
  Small,
  Large,
  Huge
} WindowSize;

static WindowSize windowsize = Large;

static int x11bug = FALSE;

static int window_width;
static int window_height;

static Display	*display;
static Screen	*screen;
static Window	window;
#ifndef SHM
static Pixmap	pixmap;
#endif
static Visual	*visual;

static GC gc;
static GC gc_colour[256];
static int colours[256];

static XComposeStatus	keyboard_status;

#ifdef XVIEW
static Frame frame;
static Panel panel;
static Canvas canvas;
static Menu system_menu;
static Menu coldstart_menu;
static Menu consol_menu;
static Menu options_menu;
static Frame chooser;

static Frame controllers_frame;
static Panel controllers_panel;
static Panel_item keypad_item;
static Panel_item mouse_item;

#ifdef LINUX_JOYSTICK
static Panel_item js0_item;
static Panel_item js1_item;
#endif

static Frame performance_frame;
static Panel performance_panel;
static Panel_item refresh_slider;
static Panel_item countdown_slider;
#endif

static int	SHIFT = 0x00;
static int	CONTROL = 0x00;
static UBYTE	*image_data;
static int	modified;

static int keypad_mode = -1; /* Joystick */
static int keypad_trig = 1; /* Keypad Trigger Position */
static int keypad_stick = 0x0f; /* Keypad Joystick Position */

static int mouse_mode = -1; /* Joystick, Paddle and Light Pen */
static int mouse_stick; /* Mouse Joystick Position */

static int js0_mode = -1;
static int js1_mode = -1;

static int	last_colour = -1;

#define	NPOINTS	(4096/4)
#define	NRECTS	(4096/4)

static int	nrects = 0;
static int	npoints = 0;

static XPoint		points[NPOINTS];
static XRectangle	rectangles[NRECTS];

static int consol;

/*
   ==========================================
   Import a few variables from atari_custom.c
   ==========================================
*/

extern int refresh_rate;
extern int countdown_rate;
extern double deltatime;

int GetKeyCode (XEvent *event)
{
  KeySym keysym;
  char buffer[128];
  int keycode = AKEY_NONE;

  XLookupString ((XKeyEvent*)event, buffer, 128,
		 &keysym, &keyboard_status);

  switch (event->type)
    {
    case Expose :
#ifndef SHM
      XCopyArea (display, pixmap, window, gc,
		 0, 0,
		 window_width, window_height,
		 0, 0);
#endif
      break;
    case KeyPress :
      switch (keysym)
	{
	case XK_Shift_L :
	case XK_Shift_R :
	  SHIFT = AKEY_SHFT;
	  break;
	case XK_Control_L :
	case XK_Control_R :
	  CONTROL = AKEY_CTRL;
	  break;
	case XK_Caps_Lock :
	  if (SHIFT)
	    keycode = AKEY_CAPSLOCK;
	  else
	    keycode = AKEY_CAPSTOGGLE;
	  break;
	case XK_Shift_Lock :
	  if (x11bug)
	    printf ("XK_Shift_Lock\n");
	  break;
	case XK_Alt_L :
	case XK_Alt_R :
	  keycode = AKEY_ATARI;
	  break;
	case XK_F1 :
	  keycode = AKEY_WARMSTART;
	  break;
	case XK_F2 :
	  consol &= 0x03;
	  keycode = AKEY_NONE;
	  break;
	case XK_F3 :
	  consol &= 0x05;
	  keycode = AKEY_NONE;
	  break;
	case XK_F4 :
	  consol &= 0x6;
	  keycode = AKEY_NONE;
	  break;
	case XK_F5 :
	  keycode = AKEY_COLDSTART;
	  break;
	case XK_F6 :
	  keycode = AKEY_PIL;
	  break;
	case XK_F7 :
	  keycode = AKEY_BREAK;
	  break;
	case XK_F8 :
	  keycode = AKEY_DISKCHANGE;
	  break;
	case XK_F9 :
	  keycode = AKEY_EXIT;
	  break;
	case XK_F10 :
	  keycode = AKEY_NONE;
	  if (deltatime == 0.0)
	    deltatime = (1.0 / 50.0);
	  else
	    deltatime = 0.0;
	  break;
	case XK_Home :
	  keycode = 0x76;
	  break;
	case XK_Insert :
	  if (SHIFT)
	    keycode = AKEY_INSERT_LINE;
	  else
	    keycode = AKEY_INSERT_CHAR;
	  break;
	case XK_BackSpace :
	  if (CONTROL)
	    keycode = AKEY_DELETE_CHAR;
	  else if (SHIFT)
	    keycode = AKEY_DELETE_LINE;
	  else
	    keycode = AKEY_BACKSPACE;
	  break;
	case XK_Delete :
	  if (CONTROL)
	    keycode = AKEY_DELETE_CHAR;
	  else if (SHIFT)
	    keycode = AKEY_DELETE_LINE;
	  else
	    keycode = AKEY_BACKSPACE;
	  break;
	case XK_Left :
	  keycode = AKEY_LEFT;
	  keypad_stick = STICK_LEFT;
	  break;
	case XK_Up :
	  keycode = AKEY_UP;
	  keypad_stick = STICK_FORWARD;
	  break;
	case XK_Right :
	  keycode = AKEY_RIGHT;
	  keypad_stick = STICK_RIGHT;
	  break;
	case XK_Down :
	  keycode = AKEY_DOWN;
	  keypad_stick = STICK_BACK;
	  break;
	case XK_Escape :
	  keycode = AKEY_ESCAPE;
	  break;
	case XK_Tab :
	  if (CONTROL)
	    keycode = AKEY_CLRTAB;
	  else if (SHIFT)
	    keycode = AKEY_SETTAB;
	  else
	    keycode = AKEY_TAB;
	  break;
	case XK_exclam :
	  keycode = AKEY_EXCLAMATION;
	  break;
	case XK_quotedbl :
	  keycode = AKEY_DBLQUOTE;
	  break;
	case XK_numbersign :
	  keycode = AKEY_HASH;
	  break;
	case XK_dollar :
	  keycode = AKEY_DOLLAR;
	  break;
	case XK_percent :
	  keycode = AKEY_PERCENT;
	  break;
	case XK_ampersand :
	  keycode = AKEY_AMPERSAND;
	  break;
	case XK_quoteright :
	  keycode = AKEY_QUOTE;
	  break;
	case XK_at :
	  keycode = AKEY_AT;
	  break;
	case XK_parenleft :
	  keycode = AKEY_PARENLEFT;
	  break;
	case XK_parenright :
	  keycode = AKEY_PARENRIGHT;
	  break;
	case XK_less :
	  keycode = AKEY_LESS;
	  break;
	case XK_greater :
	  keycode = AKEY_GREATER;
	  break;
	case XK_equal :
	  keycode = AKEY_EQUAL;
	  break;
	case XK_question :
	  keycode = AKEY_QUESTION;
	  break;
	case XK_minus :
	  keycode = AKEY_MINUS;
	  break;
	case XK_plus :
	  keycode = AKEY_PLUS;
	  break;
	case XK_asterisk :
	  keycode = AKEY_ASTERISK;
	  break;
	case XK_slash :
	  keycode = AKEY_SLASH;
	  break;
	case XK_colon :
	  keycode = AKEY_COLON;
	  break;
	case XK_semicolon :
	  keycode = AKEY_SEMICOLON;
	  break;
	case XK_comma :
	  keycode = AKEY_COMMA;
	  break;
	case XK_period :
	  keycode = AKEY_FULLSTOP;
	  break;
	case XK_underscore :
	  keycode = AKEY_UNDERSCORE;
	  break;
	case XK_bracketleft :
	  keycode = AKEY_BRACKETLEFT;
	  break;
	case XK_bracketright :
	  keycode = AKEY_BRACKETRIGHT;
	  break;
	case XK_asciicircum :
	  keycode = AKEY_CIRCUMFLEX;
	  break;
	case XK_backslash :
	  keycode = AKEY_BACKSLASH;
	  break;
	case XK_bar :
	  keycode = AKEY_BAR;
	  break;
	case XK_space :
	  keycode = AKEY_SPACE;
	  keypad_trig = 0;
	  break;
	case XK_Return :
	  keycode = AKEY_RETURN;
	  keypad_stick = STICK_CENTRE;
	  break;
	case XK_0 :
	  keycode = CONTROL | AKEY_0;
	  break;
	case XK_1 :
	  keycode = CONTROL | AKEY_1;
	  break;
	case XK_2 :
	  keycode = CONTROL | AKEY_2;
	  break;
	case XK_3 :
	  keycode = CONTROL | AKEY_3;
	  break;
	case XK_4 :
	  keycode = CONTROL | AKEY_4;
	  break;
	case XK_5 :
	  keycode = CONTROL | AKEY_5;
	  break;
	case XK_6 :
	  keycode = CONTROL | AKEY_6;
	  break;
	case XK_7 :
	  keycode = CONTROL | AKEY_7;
	  break;
	case XK_8 :
	  keycode = CONTROL | AKEY_8;
	  break;
	case XK_9 :
	  keycode = CONTROL | AKEY_9;
	  break;
	case XK_A : case XK_a :
	  keycode = SHIFT | CONTROL | AKEY_a;
	  break;
	case XK_B : case XK_b :
	  keycode = SHIFT | CONTROL | AKEY_b;
	  break;
	case XK_C : case XK_c :
	  keycode = SHIFT | CONTROL | AKEY_c;
	  break;
	case XK_D : case XK_d :
	  keycode = SHIFT | CONTROL | AKEY_d;
	  break;
	case XK_E : case XK_e :
	  keycode = SHIFT | CONTROL | AKEY_e;
	  break;
	case XK_F : case XK_f :
	  keycode = SHIFT | CONTROL | AKEY_f;
	  break;
	case XK_G : case XK_g :
	  keycode = SHIFT | CONTROL | AKEY_g;
	  break;
	case XK_H : case XK_h :
	  keycode = SHIFT | CONTROL | AKEY_h;
	  break;
	case XK_I : case XK_i :
	  keycode = SHIFT | CONTROL | AKEY_i;
	  break;
	case XK_J : case XK_j :
	  keycode = SHIFT | CONTROL | AKEY_j;
	  break;
	case XK_K : case XK_k :
	  keycode = SHIFT | CONTROL | AKEY_k;
	  break;
	case XK_L : case XK_l :
	  keycode = SHIFT | CONTROL | AKEY_l;
	  break;
	case XK_M : case XK_m :
	  keycode = SHIFT | CONTROL | AKEY_m;
	  break;
	case XK_N : case XK_n :
	  keycode = SHIFT | CONTROL | AKEY_n;
	  break;
	case XK_O : case XK_o :
	  keycode = SHIFT | CONTROL | AKEY_o;
	  break;
	case XK_P : case XK_p :
	  keycode = SHIFT | CONTROL | AKEY_p;
	  break;
	case XK_Q : case XK_q :
	  keycode = SHIFT | CONTROL | AKEY_q;
	  break;
	case XK_R : case XK_r :
	  keycode = SHIFT | CONTROL | AKEY_r;
	  break;
	case XK_S : case XK_s :
	  keycode = SHIFT | CONTROL | AKEY_s;
	  break;
	case XK_T : case XK_t :
	  keycode = SHIFT | CONTROL | AKEY_t;
	  break;
	case XK_U : case XK_u :
	  keycode = SHIFT | CONTROL | AKEY_u;
	  break;
	case XK_V : case XK_v :
	  keycode = SHIFT | CONTROL | AKEY_v;
	  break;
	case XK_W : case XK_w :
	  keycode = SHIFT | CONTROL | AKEY_w;
	  break;
	case XK_X : case XK_x :
	  keycode = SHIFT | CONTROL | AKEY_x;
	  break;
	case XK_Y : case XK_y :
	  keycode = SHIFT | CONTROL | AKEY_y;
	  break;
	case XK_Z : case XK_z :
	  keycode = SHIFT | CONTROL | AKEY_z;
	  break;
	case XK_KP_0 :
	  keypad_trig = 0;
	  break;
	case XK_KP_1 :
	  keypad_stick = STICK_LL;
	  break;
	case XK_KP_2 :
	  keypad_stick = STICK_BACK;
	  break;
	case XK_KP_3 :
	  keypad_stick = STICK_LR;
	  break;
	case XK_KP_4 :
	  keypad_stick = STICK_LEFT;
	  break;
	case XK_KP_5 :
	  keypad_stick = STICK_CENTRE;
	  break;
	case XK_KP_6 :
	  keypad_stick = STICK_RIGHT;
	  break;
	case XK_KP_7 :
	  keypad_stick = STICK_UL;
	  break;
	case XK_KP_8 :
	  keypad_stick = STICK_FORWARD;
	  break;
	case XK_KP_9 :
	  keypad_stick = STICK_UR;
	  break;
	default :
	  if (x11bug)
	    printf ("Pressed Keysym = %x\n", (int)keysym);
	  break;
	}
      break;
    case KeyRelease :
      switch (keysym)
	{
	case XK_Shift_L :
	case XK_Shift_R :
	  SHIFT = 0x00;
	  break;
	case XK_Control_L :
	case XK_Control_R :
	  CONTROL = 0x00;
	  break;
	case XK_Caps_Lock :
	  if (SHIFT)
	    keycode = AKEY_CAPSLOCK;
	  else
	    keycode = AKEY_CAPSTOGGLE;
	  break;
	case XK_Shift_Lock :
	  if (x11bug)
	    printf ("XK_Shift_Lock\n");
	  break;
	default :
	  break;
	}
      break;
    }

  return keycode;
}

static int xview_keycode = AKEY_NONE;

#ifdef XVIEW

void event_proc (Xv_Window window, Event *event, Notify_arg arg)
{
  int keycode;

  keycode = GetKeyCode (event->ie_xevent);
  if (keycode != AKEY_NONE)
    xview_keycode = keycode;
}

static int auto_reboot;

int disk_change (char *a, char *full_filename, char *filename)
{
  int diskno;
  int status;

  diskno = 1;

  if (!auto_reboot)
    diskno = notice_prompt (panel, NULL,
			    NOTICE_MESSAGE_STRINGS,
			      "Insert Disk into which drive?",
			      NULL,
			    NOTICE_BUTTON, "1", 1,
			    NOTICE_BUTTON, "2", 2,
			    NOTICE_BUTTON, "3", 3,
			    NOTICE_BUTTON, "4", 4,
			    NOTICE_BUTTON, "5", 5,
			    NOTICE_BUTTON, "6", 6,
			    NOTICE_BUTTON, "7", 7,
			    NOTICE_BUTTON, "8", 8,
			    NULL);

  if ((diskno < 1) || (diskno > 8))
    {
      printf ("Invalid diskno: %d\n", diskno);
      exit (1);
    }

  SIO_Dismount(diskno);
  if (!SIO_Mount (diskno,full_filename))
    status = XV_ERROR;
  else
    {
      if (auto_reboot)
	Coldstart ();
      status = XV_OK;
    }

  return status;
}

boot_callback ()
{
  auto_reboot = TRUE;

  xv_set (chooser,
	  FRAME_LABEL, "Disk Selector",
	  FILE_CHOOSER_DIRECTORY, ATARI_DISK_DIR,
	  FILE_CHOOSER_NOTIFY_FUNC, disk_change,
	  XV_SHOW, TRUE,
	  NULL);
}

insert_callback ()
{
  auto_reboot = FALSE;

  xv_set (chooser,
	  FRAME_LABEL, "Disk Selector",
	  FILE_CHOOSER_DIRECTORY, ATARI_DISK_DIR,
	  FILE_CHOOSER_NOTIFY_FUNC, disk_change,
	  XV_SHOW, TRUE,
	  NULL);
}

eject_callback ()
{
  int diskno;

  diskno = notice_prompt (panel, NULL,
			  NOTICE_MESSAGE_STRINGS,
			    "Eject Disk from drive?",
			    NULL,
			  NOTICE_BUTTON, "1", 1,
			  NOTICE_BUTTON, "2", 2,
			  NOTICE_BUTTON, "3", 3,
			  NOTICE_BUTTON, "4", 4,
			  NOTICE_BUTTON, "5", 5,
			  NOTICE_BUTTON, "6", 6,
			  NOTICE_BUTTON, "7", 7,
			  NOTICE_BUTTON, "8", 8,
			  NULL);

  if ((diskno < 1) || (diskno > 8))
    {
      printf ("Invalid diskno: %d\n", diskno);
      exit (1);
    }

  SIO_Dismount(diskno);
}

int rom_change (char *a, char *full_filename, char *filename)
{
  struct stat buf;
  int status = XV_ERROR;
  int yesno;

  stat (full_filename, &buf);

  switch (buf.st_size)
    {
    case 0x2000 :
      Remove_ROM ();
      if (Insert_8K_ROM(full_filename))
	{
	  Coldstart ();
	  status = XV_OK;
	}
      break;
    case 0x4000 :
      yesno = notice_prompt (panel, NULL,
			     NOTICE_MESSAGE_STRINGS,
			       filename,
			       "Is this an OSS Supercartridge?",
			       NULL,
			     NOTICE_BUTTON_YES, "No",
			     NOTICE_BUTTON_NO, "Yes",
			     NULL);
      if (yesno == NOTICE_YES)
	{
	  Remove_ROM ();
	  if (Insert_16K_ROM(full_filename))
	    {
	      Coldstart ();
	      status = XV_OK;
	    }
	}
      else
	{
	  Remove_ROM ();
	  if (Insert_OSS_ROM(full_filename))
	    {
	      Coldstart ();
	      status = XV_OK;
	    }
	}
      break;
    case 0x8000 :
      Remove_ROM ();
      if (machine == Atari5200)
	{
	  if (Insert_32K_5200ROM(full_filename))
	    {
	      Coldstart ();
	      status = XV_OK;
	    }
	}
      else
	{
	  if (Insert_DB_ROM(full_filename))
	    {
	      Coldstart ();
	      status = XV_OK;
	    }
	}
      break;
    default :
      break;
    }

  return status;
}

insert_rom_callback ()
{
  xv_set (chooser,
	  FRAME_LABEL, "ROM Selector",
	  FILE_CHOOSER_DIRECTORY, ATARI_ROM_DIR,
	  FILE_CHOOSER_NOTIFY_FUNC, rom_change,
	  XV_SHOW, TRUE,
	  NULL);
}

remove_rom_callback ()
{
  Remove_ROM ();
  Coldstart ();
}

enable_pill_callback ()
{
  EnablePILL ();
  Coldstart ();
}

exit_callback ()
{
  exit (1);
}

option_callback ()
{
  consol &= 0x03;
}

select_callback ()
{
  consol &= 0x05;
}

start_callback ()
{
  consol &= 0x6;
}

help_callback ()
{
  xview_keycode = AKEY_HELP;
}

break_callback ()
{
  xview_keycode = AKEY_BREAK;
}

reset_callback ()
{
  Warmstart ();
}

coldstart_callback ()
{
  Coldstart ();
}

coldstart_osa_callback ()
{
  int status;

  status = Initialise_AtariOSA ();
  if (status)
    {
      Menu_item menuitem;

      menuitem = xv_get (consol_menu,
			 MENU_NTH_ITEM, 4);

      xv_set (menuitem,
	      MENU_INACTIVE, TRUE,
	      NULL);
    }
  else
    {
      notice_prompt (panel, NULL,
		     NOTICE_MESSAGE_STRINGS,
		       "Sorry, OS/A ROM Unavailable",
		       NULL,
		     NOTICE_BUTTON, "Cancel", 1,
		     NULL);
    }
}

coldstart_osb_callback ()
{
  int status;

  status = Initialise_AtariOSB ();
  if (status)
    {
      Menu_item menuitem;
      
      menuitem = xv_get (consol_menu,
			 MENU_NTH_ITEM, 4);

      xv_set (menuitem,
	      MENU_INACTIVE, TRUE,
	      NULL);
    }
  else
    {
      notice_prompt (panel, NULL,
		     NOTICE_MESSAGE_STRINGS,
		       "Sorry, OS/B ROM Unavailable",
		       NULL,
		     NOTICE_BUTTON, "Cancel", 1,
		     NULL);
    }
}

coldstart_xl_callback ()
{
  int status;

  status = Initialise_AtariXL ();
  if (status)
    {
      Menu_item menuitem;

      menuitem = xv_get (consol_menu,
			 MENU_NTH_ITEM, 4);
      
      xv_set (menuitem,
	      MENU_INACTIVE, FALSE,
	      NULL);
    }
  else
    {
      notice_prompt (panel, NULL,
		     NOTICE_MESSAGE_STRINGS,
		       "Sorry, XL/XE ROM Unavailable",
		       NULL,
		     NOTICE_BUTTON, "Cancel", 1,
		     NULL);
    }
}

coldstart_xe_callback ()
{
  int status;

  status = Initialise_AtariXE ();
  if (status)
    {
      Menu_item menuitem;

      menuitem = xv_get (consol_menu,
			 MENU_NTH_ITEM, 4);

      xv_set (menuitem,
	      MENU_INACTIVE, FALSE,
	      NULL);
    }
  else
    {
      notice_prompt (panel, NULL,
		     NOTICE_MESSAGE_STRINGS,
		       "Sorry, XL/XE ROM Unavailable",
		       NULL,
		     NOTICE_BUTTON, "Cancel", 1,
		     NULL);
    }
}

coldstart_5200_callback ()
{
  int status;

  status = Initialise_Atari5200 ();
  if (status)
    {
      Menu_item menuitem;

      menuitem = xv_get (consol_menu,
			 MENU_NTH_ITEM, 4);

      xv_set (menuitem,
	      MENU_INACTIVE, FALSE,
	      NULL);
    }
  else
    {
      notice_prompt (panel, NULL,
		     NOTICE_MESSAGE_STRINGS,
		       "Sorry, 5200 ROM Unavailable",
		       NULL,
		     NOTICE_BUTTON, "Cancel", 1,
		     NULL);
    }
}

controllers_ok_callback ()
{
  xv_set (controllers_frame,
	  XV_SHOW, FALSE,
	  NULL);
}

controllers_callback ()
{
  xv_set (controllers_frame,
	  XV_SHOW, TRUE,
	  NULL);
}

void sorry_message ()
{
  notice_prompt (panel, NULL,
		 NOTICE_MESSAGE_STRINGS,
		   "Sorry, controller already assign",
		   "to another device",
		   NULL,
		 NOTICE_BUTTON, "Cancel", 1,
		 NULL);
}

keypad_callback ()
{
  int new_mode;

  new_mode = xv_get (keypad_item, PANEL_VALUE);

  if ((new_mode != mouse_mode) &&
      (new_mode != js0_mode) &&
      (new_mode != js1_mode))
    {
      keypad_mode = new_mode;
    }
  else
    {
      sorry_message ();
      xv_set (keypad_item,
	      PANEL_VALUE, keypad_mode,
	      NULL);
    }
}

mouse_callback ()
{
  int new_mode;

  new_mode = xv_get (mouse_item, PANEL_VALUE);

  if ((new_mode != keypad_mode) &&
      (new_mode != js0_mode) &&
      (new_mode != js1_mode))
    {
      mouse_mode = new_mode;
    }
  else
    {
      sorry_message ();
      xv_set (mouse_item,
	      PANEL_VALUE, mouse_mode,
	      NULL);
    }
}

#ifdef LINUX_JOYSTICK
js0_callback ()
{
  int new_mode;

  new_mode = xv_get (js0_item, PANEL_VALUE);

  if ((new_mode != keypad_mode) &&
      (new_mode != mouse_mode) &&
      (new_mode != js1_mode))
    {
      js0_mode = new_mode;
    }
  else
    {
      sorry_message ();
      xv_set (js0_item,
	      PANEL_VALUE, js0_mode,
	      NULL);
    }
}

js1_callback ()
{
  int new_mode;

  new_mode = xv_get (js1_item, PANEL_VALUE);

  if ((new_mode != keypad_mode) &&
      (new_mode != mouse_mode) &&
      (new_mode != js0_mode))
    {
      js1_mode = new_mode;
    }
  else
    {
      sorry_message ();
      xv_set (js1_item,
	      PANEL_VALUE, js1_mode,
	      NULL);
    }
}
#endif

performance_ok_callback ()
{
  xv_set (performance_frame,
	  XV_SHOW, FALSE,
	  NULL);
}

performance_callback ()
{
  xv_set (performance_frame,
	  XV_SHOW, TRUE,
	  NULL);
}

refresh_callback (Panel_item item, int value, Event *event)
{
  refresh_rate = value;
}

countdown_callback (Panel_item item, int value, Event *event)
{
  countdown_rate = value;
}
#endif

void Atari_WhatIs (int mode)
{
  switch (mode)
    {
    case 0 :
      printf ("Joystick 0");
      break;
    case 1 :
      printf ("Joystick 1");
      break;
    case 2 :
      printf ("Joystick 2");
      break;
    case 3 :
      printf ("Joystick 3");
      break;
    default :
      printf ("not available");
      break;
    }
}

#ifdef MOTIF
void motif_boot_disk (Widget fs, XtPointer client_data,
		      XmFileSelectionBoxCallbackStruct *cbs)
{
  char *filename;

  if (XmStringGetLtoR(cbs->value, XmSTRING_DEFAULT_CHARSET, &filename))
    {
      if (*filename)
	{
	  SIO_Dismount(1);
	  if (SIO_Mount (1, filename))
	    Coldstart ();
	}

      XtFree (filename);
    }

  XtUnmanageChild (fs);
  XtPopdown (XtParent(fs));
}

void motif_fs_cancel (Widget fs)
{
  XtUnmanageChild (fs);
  XtPopdown (XtParent(fs));
}

void motif_system_cback (Widget w, int item_no)
{
  XmString t;
  int status;
  char *errmsg = NULL;

  switch (item_no)
    {
    case 0 :
      t = XmStringCreateSimple ("/usr/local/lib/atari/DISKS/*");
      XmFileSelectionDoSearch (fsel, t);
      XmStringFree (t);
      XtAddCallback (fsel, XmNokCallback, motif_boot_disk, NULL);
      XtAddCallback (fsel, XmNcancelCallback, motif_fs_cancel, NULL);
      XtManageChild (fsel);
      XtPopup (XtParent(fsel), XtGrabNone);
      break;
    case 1 :
    case 2 :
    case 3 :
      break;
    case 4 :
      Remove_ROM ();
      Coldstart ();
      break;
    case 5 :
      EnablePILL ();
      Coldstart ();
      break;
    case 6 :
      status = Initialise_AtariOSA ();
      if (status == 0)
	errmsg = "Sorry, OS/A ROM Unavailable";
      break;
    case 7 :
      status = Initialise_AtariOSB ();
      if (status == 0)
	errmsg = "Sorry, OS/B ROM Unavailable";
      break;
    case 8 :
      status = Initialise_AtariXL ();
      if (status == 0)
	errmsg = "Sorry, XL/XE ROM Unavailable";
      break;
    case 9 :
      status = Initialise_AtariXE ();
      if (status == 0)
	errmsg = "Sorry, XL/XE ROM Unavailable";
      break;
    case 10 :
      status = Initialise_Atari5200 ();
      if (status == 0)
	errmsg = "Sorry, 5200 ROM Unavailable";
      break;
    case 11 :
      exit (0);
    }

  if (errmsg)
    {
      static Widget dialog;

      if (!dialog)
	{
	  Arg arg[1];

	  dialog = XmCreateErrorDialog (main_w, "message", arg, 0);

	  XtVaSetValues (dialog,
			 XmNdialogStyle, XmDIALOG_FULL_APPLICATION_MODAL,
			 NULL);

	  XtUnmanageChild (XmMessageBoxGetChild(dialog, XmDIALOG_OK_BUTTON));
	  XtUnmanageChild (XmMessageBoxGetChild(dialog, XmDIALOG_HELP_BUTTON));
	}

      t = XmStringCreateSimple (errmsg);
      XtVaSetValues (dialog,
		     XmNmessageString, t,
		     NULL);
      XmStringFree (t);
      XtManageChild (dialog);
    }
}

void motif_consol_cback (Widget w, int item_no)
{
  switch (item_no)
    {
    case 0 :
      consol &= 0x03; /* Option Pressed */
      break;
    case 1 :
      consol &= 0x05; /* Select Pressed */
      break;
    case 2 :
      consol &= 0x06; /* Start Pressed */
      break;
    case 3 :
      xview_keycode = AKEY_HELP;
      break;
    case 4 :
      xview_keycode = AKEY_BREAK;
      break;
    case 5 :
      Warmstart ();
      break;
    case 6 :
      Coldstart ();
      break;
    }
}

void motif_keypress (Widget w, XtPointer client_data, XEvent *event)
{
  int keycode;

  keycode = GetKeyCode (event);
  if (keycode != AKEY_NONE)
    xview_keycode = keycode;
}
#endif

void Atari_Initialise (int *argc, char *argv[])
{
  XSetWindowAttributes	xswda;

  XGCValues	xgcvl;

  int depth;
  int i, j;
  int mode = 0;

#ifdef XVIEW
  int ypos;

  xv_init (XV_INIT_ARGC_PTR_ARGV, argc, argv, NULL);
#endif

#ifdef MOTIF
  toplevel = XtVaAppInitialize (&app, "Atari800",
				NULL, 0,
				argc, argv, NULL,
				XtNtitle, ATARI_TITLE,
				NULL);
#endif

#ifdef FPS_MONITOR
  gettimeofday (&tp, &tzp);
  basetime = tp.tv_sec + (tp.tv_usec / 1000000.0);
#endif

  for (i=j=1;i<*argc;i++)
    {
      if (strcmp(argv[i],"-small") == 0)
	windowsize = Small;
      else if (strcmp(argv[i],"-large") == 0)
	windowsize = Large;
      else if (strcmp(argv[i],"-huge") == 0)
	windowsize = Huge;
      else if (strcmp(argv[i],"-x11bug") == 0)
	x11bug = TRUE;
      else
	{
	  if (strcmp(argv[i],"-help") == 0)
	    {
	      printf ("\t-small        Small window (%dx%d)\n",
		      ATARI_WIDTH, ATARI_HEIGHT);
	      printf ("\t-large        Large window (%dx%d)\n",
		      ATARI_WIDTH*2, ATARI_HEIGHT*2);
	      printf ("\t-huge         Huge window (%dx%d)\n",
		      ATARI_WIDTH*3, ATARI_HEIGHT*3);
	      printf ("\t-x11bug       Enable debug code in atari_x11.c\n");
	    }

	  argv[j++] = argv[i];
	}
    }

  *argc = j;

#ifdef NAS
  NAS_Initialise (argc, argv);
#endif

#ifdef SHM
  if (windowsize != Small)
    {
      printf ("X Shared memory version only supports small window\n");
      windowsize = Small;
    }
#endif

  switch (windowsize)
    {
    case Small :
      window_width = ATARI_WIDTH;
      window_height = ATARI_HEIGHT;
      break;
    case Large :
      window_width = ATARI_WIDTH * 2;
      window_height = ATARI_HEIGHT * 2;
      break;
    case Huge :
      window_width = ATARI_WIDTH * 3;
      window_height = ATARI_HEIGHT * 3;
      break;
    }

#ifdef LINUX_JOYSTICK
  js0 = open ("/dev/js0", O_RDONLY, 0777);
  if (js0 != -1)
    {
      int status;

      status = read (js0, &js_data, JS_RETURN);
      if (status != JS_RETURN)
	{
	  perror ("/dev/js0");
	  exit (1);
	}

      js0_centre_x = js_data.x;
      js0_centre_y = js_data.y;

      if (x11bug)
	printf ("Joystick 0: centre_x = %d, centry_y = %d\n",
		js0_centre_x, js0_centre_y);

      js0_mode = mode++;
    }

  js1 = open ("/dev/js1", O_RDONLY, 0777);
  if (js1 != -1)
    {
      int status;

      status = read (js1, &js_data, JS_RETURN);
      if (status != JS_RETURN)
	{
	  perror ("/dev/js1");
	  exit (1);
	}

      js1_centre_x = js_data.x;
      js1_centre_y = js_data.y;

      if (x11bug)
	printf ("Joystick 1: centre_x = %d, centry_y = %d\n",
		js1_centre_x, js1_centre_y);

      js1_mode = mode++;
    }
#endif

  mouse_mode = mode++;
  keypad_mode = mode++;

#ifdef XVIEW
  frame = (Frame)xv_create ((Xv_opaque)NULL, FRAME,
			    FRAME_LABEL, ATARI_TITLE,
			    FRAME_SHOW_RESIZE_CORNER, FALSE,
			    XV_WIDTH, window_width,
			    XV_HEIGHT, window_height + 27,
#ifdef FPS_MONITOR
			    FRAME_SHOW_FOOTER, TRUE,
#endif
			    XV_SHOW, TRUE,
			    NULL);

  panel = (Panel)xv_create (frame, PANEL,
			    XV_HEIGHT, 25,
			    XV_SHOW, TRUE,
			    NULL);

  system_menu = xv_create ((Xv_opaque)NULL, MENU,
			   MENU_ITEM,
			     MENU_STRING, "Boot Disk",
			     MENU_NOTIFY_PROC, boot_callback,
			     NULL,
			   MENU_ITEM,
			     MENU_STRING, "Insert Disk",
			     MENU_NOTIFY_PROC, insert_callback,
			     NULL,
			   MENU_ITEM,
			     MENU_STRING, "Eject Disk",
			     MENU_NOTIFY_PROC, eject_callback,
			     NULL,
			   MENU_ITEM,
			     MENU_STRING, "Insert Cartridge",
			     MENU_NOTIFY_PROC, insert_rom_callback,
			     NULL,
			   MENU_ITEM,
			     MENU_STRING, "Remove Cartridge",
			     MENU_NOTIFY_PROC, remove_rom_callback,
			     NULL,
			   MENU_ITEM,
			     MENU_STRING, "Enable PILL",
			     MENU_NOTIFY_PROC, enable_pill_callback,
			     NULL,
			   MENU_ITEM,
			     MENU_STRING, "Atari 800 OS/A",
			     MENU_NOTIFY_PROC, coldstart_osa_callback,
			     NULL,
			   MENU_ITEM,
			     MENU_STRING, "Atari 800 OS/B",
			     MENU_NOTIFY_PROC, coldstart_osb_callback,
			     NULL,
			   MENU_ITEM,
			     MENU_STRING, "Atari 800XL",
			     MENU_NOTIFY_PROC, coldstart_xl_callback,
			     NULL,
			   MENU_ITEM,
			     MENU_STRING, "Atari 130XE",
			     MENU_NOTIFY_PROC, coldstart_xe_callback,
			     NULL,
			   MENU_ITEM,
			     MENU_STRING, "Atari 5200",
			     MENU_NOTIFY_PROC, coldstart_5200_callback,
			     NULL,
			   MENU_ITEM,
			     MENU_STRING, "Exit",
			     MENU_NOTIFY_PROC, exit_callback,
			     NULL,
			   NULL);

  xv_create (panel, PANEL_BUTTON,
	     PANEL_LABEL_STRING, "System",
	     PANEL_ITEM_MENU, system_menu,
	     NULL);

  consol_menu = (Menu)xv_create ((Xv_opaque)NULL, MENU,
				 MENU_ITEM,
				   MENU_STRING, "Option",
				   MENU_NOTIFY_PROC, option_callback,
				   NULL,
				 MENU_ITEM,
				   MENU_STRING, "Select",
				   MENU_NOTIFY_PROC, select_callback,
				   NULL,
				 MENU_ITEM,
				   MENU_STRING, "Start",
				   MENU_NOTIFY_PROC, start_callback,
				   NULL,
				 MENU_ITEM,
				   MENU_STRING, "Help",
				   MENU_NOTIFY_PROC, help_callback,
				   MENU_INACTIVE, (machine == Atari),
				   NULL,
				 MENU_ITEM,
				   MENU_STRING, "Break",
				   MENU_NOTIFY_PROC, break_callback,
				   NULL,
				 MENU_ITEM,
				   MENU_STRING, "Reset",
				   MENU_NOTIFY_PROC, reset_callback,
				   NULL,
				 MENU_ITEM,
				   MENU_STRING, "Coldstart",
				   MENU_NOTIFY_PROC, coldstart_callback,
				   NULL,
				 NULL);

  xv_create (panel, PANEL_BUTTON,
	     PANEL_LABEL_STRING, "Console",
	     PANEL_ITEM_MENU, consol_menu,
	     NULL);

  options_menu = (Menu)xv_create ((Xv_opaque)NULL, MENU,
				  MENU_ITEM,
				    MENU_STRING, "Controllers",
				    MENU_NOTIFY_PROC, controllers_callback,
				    NULL,
				  MENU_ITEM,
				    MENU_STRING, "Performance",
				    MENU_NOTIFY_PROC, performance_callback,
				    NULL,
				  NULL);

  xv_create (panel, PANEL_BUTTON,
	     PANEL_LABEL_STRING, "Options",
	     PANEL_ITEM_MENU, options_menu,
	     NULL);

  canvas = (Canvas)xv_create (frame, CANVAS,
			      CANVAS_WIDTH, window_width,
			      CANVAS_HEIGHT, window_height,
			      NULL);
/*
   =====================================
   Create Controller Configuration Frame
   =====================================
*/
  controllers_frame = (Frame)xv_create (frame, FRAME_CMD,
				       FRAME_LABEL, "Controller Configuration",
				       XV_WIDTH, 300,
				       XV_HEIGHT, 150,
				       NULL);

  controllers_panel = (Panel)xv_get (controllers_frame, FRAME_CMD_PANEL,
				     NULL);

  ypos = 10;
  keypad_item = (Panel_item)xv_create (controllers_panel, PANEL_CHOICE_STACK,
				       PANEL_VALUE_X, 150,
				       PANEL_VALUE_Y, ypos,
				       PANEL_LAYOUT, PANEL_HORIZONTAL,
				       PANEL_LABEL_STRING, "Numeric Keypad",
				       PANEL_CHOICE_STRINGS,
				         "Joystick 1",
				         "Joystick 2",
				         "Joystick 3",
				         "Joystick 4",
				         NULL,
				       PANEL_VALUE, keypad_mode,
				       PANEL_NOTIFY_PROC, keypad_callback,
				       NULL);
  ypos += 25;

  mouse_item = (Panel_item)xv_create (controllers_panel, PANEL_CHOICE_STACK,
				      PANEL_VALUE_X, 150,
				      PANEL_VALUE_Y, ypos,
				      PANEL_LAYOUT, PANEL_HORIZONTAL,
				      PANEL_LABEL_STRING, "Mouse",
				      PANEL_CHOICE_STRINGS,
				        "Joystick 1",
				        "Joystick 2",
				        "Joystick 3",
				        "Joystick 4",
				        "Paddle 1",
				        "Paddle 2",
				        "Paddle 3",
				        "Paddle 4",
				        "Paddle 5",
				        "Paddle 6",
				        "Paddle 7",
				        "Paddle 8",
				        "Light Pen",
				        NULL,
				      PANEL_VALUE, mouse_mode,
				      PANEL_NOTIFY_PROC, mouse_callback,
				      NULL);
  ypos += 25;

#ifdef LINUX_JOYSTICK
  if (js0 != -1)
    {
      js0_item = (Panel_item)xv_create (controllers_panel, PANEL_CHOICE_STACK,
					PANEL_VALUE_X, 150,
					PANEL_VALUE_Y, ypos,
					PANEL_LAYOUT, PANEL_HORIZONTAL,
					PANEL_LABEL_STRING, "/dev/js0",
					PANEL_CHOICE_STRINGS,
					  "Joystick 1",
					  "Joystick 2",
					  "Joystick 3",
					  "Joystick 4",
					NULL,
					PANEL_VALUE, js0_mode,
					PANEL_NOTIFY_PROC, js0_callback,
					NULL);
      ypos += 25;
    }

  if (js1 != -1)
    {
      js1_item = (Panel_item)xv_create (controllers_panel, PANEL_CHOICE_STACK,
					PANEL_VALUE_X, 150,
					PANEL_VALUE_Y, ypos,
					PANEL_LAYOUT, PANEL_HORIZONTAL,
					PANEL_LABEL_STRING, "/dev/js1",
					PANEL_CHOICE_STRINGS,
					  "Joystick 1",
					  "Joystick 2",
					  "Joystick 3",
					  "Joystick 4",
					NULL,
					PANEL_VALUE, js1_mode,
					PANEL_NOTIFY_PROC, js1_callback,
					NULL);
      ypos += 25;
    }
#endif

  xv_create (controllers_panel, PANEL_BUTTON,
	     XV_X, 130,
	     XV_Y, 125,
	     PANEL_LABEL_STRING, "OK",
	     PANEL_NOTIFY_PROC, controllers_ok_callback,
	     NULL);
/*
   ======================================
   Create Performance Configuration Frame
   ======================================
*/
  performance_frame = (Frame)xv_create (frame, FRAME_CMD,
					FRAME_LABEL, "Performance Configuration",
					XV_WIDTH, 400,
					XV_HEIGHT, 100,
					NULL);

  performance_panel = (Panel)xv_get (performance_frame, FRAME_CMD_PANEL,
				     NULL);

  ypos = 10;
  refresh_slider = (Panel_item)xv_create (performance_panel, PANEL_SLIDER,
					  PANEL_VALUE_X, 155,
					  PANEL_VALUE_Y, ypos,
					  PANEL_LAYOUT, PANEL_HORIZONTAL,
					  PANEL_LABEL_STRING, "Screen Refresh Rate",
					  PANEL_VALUE, refresh_rate,
					  PANEL_MIN_VALUE, 1,
					  PANEL_MAX_VALUE, 32,
					  PANEL_SLIDER_WIDTH, 100,
					  PANEL_TICKS, 32,
					  PANEL_NOTIFY_PROC, refresh_callback,
					  NULL);
  ypos += 25;

  countdown_slider = (Panel_item)xv_create (performance_panel, PANEL_SLIDER,
					    PANEL_VALUE_X, 155,
					    PANEL_VALUE_Y, ypos,
					    PANEL_LAYOUT, PANEL_HORIZONTAL,
					    PANEL_LABEL_STRING, "Instructions per VBI",
					    PANEL_VALUE, countdown_rate,
					    PANEL_MIN_VALUE, 1000,
					    PANEL_MAX_VALUE, 10000,
					    PANEL_SLIDER_WIDTH, 100,
					    PANEL_TICKS, 10,
					    PANEL_NOTIFY_PROC, countdown_callback,
					    NULL);

  xv_create (performance_panel, PANEL_BUTTON,
	     XV_X, 180,
	     XV_Y, 75,
	     PANEL_LABEL_STRING, "OK",
	     PANEL_NOTIFY_PROC, performance_ok_callback,
	     NULL);
/*
   ====================
   Get X Window Objects
   ====================
*/
  display = (Display*)xv_get(frame, XV_DISPLAY);
  if (!display)
    {
      printf ("Failed to open display\n");
      exit (1);
    }

  screen = XDefaultScreenOfDisplay (display);
  if (!screen)
    {
      printf ("Unable to get screen\n");
      exit (1);
    }

  visual = XDefaultVisualOfScreen (screen);
  if (!visual)
    {
      printf ("Unable to get visual\n");
      exit (1);
    }

  window = (Window)xv_get(canvas_paint_window(canvas), XV_XID);
  depth = XDefaultDepthOfScreen (screen);

  chooser = (Frame)xv_create (frame, FILE_CHOOSER,
			      FILE_CHOOSER_TYPE, FILE_CHOOSER_OPEN,
			      NULL);

  xv_set (canvas_paint_window(canvas),
	  WIN_EVENT_PROC, event_proc,
	  WIN_CONSUME_EVENTS, WIN_ASCII_EVENTS, WIN_MOUSE_BUTTONS, NULL,
	  NULL);
#endif

#ifdef MOTIF
  {
    Widget menubar;

    XmString s_system;
    XmString s_boot_disk;
    XmString s_insert_disk;
    XmString s_eject_disk;
    XmString s_insert_cart;
    XmString s_remove_cart;
    XmString s_enable_pill;
    XmString s_osa;
    XmString s_osb;
    XmString s_osxl;
    XmString s_osxe;
    XmString s_os5200;
    XmString s_exit;

    XmString s_console;
    XmString s_option;
    XmString s_select;
    XmString s_start;
    XmString s_help;
    XmString s_break;
    XmString s_warmstart;
    XmString s_coldstart;

    main_w = XtVaCreateManagedWidget ("main_window",
				      xmMainWindowWidgetClass, toplevel,
				      NULL);

    s_system = XmStringCreateSimple ("System");
    s_boot_disk = XmStringCreateSimple ("Boot Disk");
    s_insert_disk = XmStringCreateSimple ("Insert Disk");
    s_eject_disk = XmStringCreateSimple ("Eject Disk");
    s_insert_cart = XmStringCreateSimple ("Insert Cartridge");
    s_remove_cart = XmStringCreateSimple ("Remove Cartridge");
    s_enable_pill = XmStringCreateSimple ("Enable PILL");
    s_osa = XmStringCreateSimple ("Atari 800 OS/A");
    s_osb = XmStringCreateSimple ("Atari 800 OS/B");
    s_osxl = XmStringCreateSimple ("Atari 800XL");
    s_osxe = XmStringCreateSimple ("Atari 130XE");
    s_os5200 = XmStringCreateSimple ("Atari 5200");
    s_exit = XmStringCreateSimple ("Exit");

    s_console = XmStringCreateSimple ("Console");
    s_option = XmStringCreateSimple ("Option");
    s_select = XmStringCreateSimple ("Select");
    s_start = XmStringCreateSimple ("Start");
    s_help = XmStringCreateSimple ("Help");
    s_break = XmStringCreateSimple ("Break");
    s_warmstart = XmStringCreateSimple ("Warmstart");
    s_coldstart = XmStringCreateSimple ("Coldstart");
;
    menubar = XmVaCreateSimpleMenuBar (main_w, "menubar",
				       XmVaCASCADEBUTTON, s_system, 'S',
				       XmVaCASCADEBUTTON, s_console, 'C',
				       NULL);

    XmVaCreateSimplePulldownMenu (menubar, "system_menu", 0, motif_system_cback,
				  XmVaPUSHBUTTON, s_boot_disk, 'O', NULL, NULL,
				  XmVaPUSHBUTTON, s_insert_disk, 'I', NULL, NULL,
				  XmVaPUSHBUTTON, s_eject_disk, 'J', NULL, NULL,
				  XmVaSEPARATOR,
				  XmVaPUSHBUTTON, s_insert_cart, 'N', NULL, NULL,
				  XmVaPUSHBUTTON, s_remove_cart, 'R', NULL, NULL,
				  XmVaPUSHBUTTON, s_enable_pill, 'P', NULL, NULL,
				  XmVaSEPARATOR,
				  XmVaPUSHBUTTON, s_osa, 'A', NULL, NULL,
				  XmVaPUSHBUTTON, s_osb, 'B', NULL, NULL,
				  XmVaPUSHBUTTON, s_osxl, 'L', NULL, NULL,
				  XmVaPUSHBUTTON, s_osxe, 'E', NULL, NULL,
				  XmVaPUSHBUTTON, s_os5200, '5', NULL, NULL,
				  XmVaSEPARATOR,
				  XmVaPUSHBUTTON, s_exit, 'x', NULL, NULL,
				  NULL);

    XmVaCreateSimplePulldownMenu (menubar, "console_menu", 1, motif_consol_cback,
				  XmVaPUSHBUTTON, s_option, 'O', NULL, NULL,
				  XmVaPUSHBUTTON, s_select, 't', NULL, NULL,
				  XmVaPUSHBUTTON, s_start, 'S', NULL, NULL,
				  XmVaSEPARATOR,
				  XmVaPUSHBUTTON, s_help, 'H', NULL, NULL,
				  XmVaPUSHBUTTON, s_break, 'B', NULL, NULL,
				  XmVaSEPARATOR,
				  XmVaPUSHBUTTON, s_warmstart, 'W', NULL, NULL,
				  XmVaPUSHBUTTON, s_coldstart, 'C', NULL, NULL,
				  NULL);

    XmStringFree (s_system);
    XmStringFree (s_osa);
    XmStringFree (s_osb);
    XmStringFree (s_osxl);
    XmStringFree (s_osxe);
    XmStringFree (s_os5200);
    XmStringFree (s_exit);

    XmStringFree (s_console);
    XmStringFree (s_option);
    XmStringFree (s_select);
    XmStringFree (s_start);
    XmStringFree (s_help);
    XmStringFree (s_break);
    XmStringFree (s_warmstart);
    XmStringFree (s_coldstart);

    XtManageChild (menubar);

    fsel = XmCreateFileSelectionDialog (toplevel, "filesb", NULL, 0);

    drawing_area = XtVaCreateManagedWidget ("Canvas",
					    xmDrawingAreaWidgetClass, main_w,
					    XmNunitType, XmPIXELS,
					    XmNheight, window_height,
					    XmNwidth, window_width,
					    XmNresizePolicy, XmNONE,
					    NULL);

    XtAddEventHandler (drawing_area,
		       KeyPressMask | KeyReleaseMask,
		       False,
		       motif_keypress, NULL);

    XtRealizeWidget (toplevel);
  }

  display = XtDisplay (drawing_area);

  window = XtWindow (drawing_area);

  screen = XDefaultScreenOfDisplay (display);
  if (!screen)
    {
      printf ("Unable to get screen\n");
      exit (1);
    }

  visual = XDefaultVisualOfScreen (screen);
  if (!visual)
    {
      printf ("Unable to get visual\n");
      exit (1);
    }

  depth = XDefaultDepthOfScreen (screen);
#endif

#ifndef MOTIF
#ifndef XVIEW
  display = XOpenDisplay (NULL);
  if (!display)
    {
      printf ("Failed to open display\n");
      exit (1);
    }

  screen = XDefaultScreenOfDisplay (display);
  if (!screen)
    {
      printf ("Unable to get screen\n");
      exit (1);
    }

  visual = XDefaultVisualOfScreen (screen);
  if (!visual)
    {
      printf ("Unable to get visual\n");
      exit (1);
    }

  depth = XDefaultDepthOfScreen (screen);

  xswda.event_mask = KeyPressMask | KeyReleaseMask | ExposureMask;

  window = XCreateWindow (display,
			  XRootWindowOfScreen(screen),
			  50, 50,
			  window_width, window_height, 3, depth,
			  InputOutput, visual,
			  CWEventMask | CWBackPixel,
			  &xswda);

  XStoreName (display, window, ATARI_TITLE);
#endif
#endif

#ifdef SHM
  {
    int major;
    int minor;
    Bool pixmaps;
    Status status;

    status = XShmQueryVersion (display, &major, &minor, &pixmaps);
    if (!status)
      {
	printf ("X Shared Memory extensions not available\n");
	exit (1);
      }

    printf ("Using X11 Shared Memory Extensions\n");

    image = XShmCreateImage (display, visual, depth, ZPixmap,
			     NULL, &shminfo, window_width, window_height);

    shminfo.shmid = shmget (IPC_PRIVATE,
			    (ATARI_HEIGHT+16) * ATARI_WIDTH,
			    IPC_CREAT | 0777);
    shminfo.shmaddr = image->data = atari_screen = shmat (shminfo.shmid, 0, 0);
    shminfo.readOnly = False;

    XShmAttach (display, &shminfo);

    XSync (display, False);

    shmctl (shminfo.shmid, IPC_RMID, 0);
  }
#else
  pixmap = XCreatePixmap (display, window,
			  window_width, window_height, depth);
#endif

  for (i=0;i<256;i+=2)
    {
      XColor	colour;

      int	rgb = colortable[i];
      int	status;

      colour.red = (rgb & 0x00ff0000) >> 8;
      colour.green = (rgb & 0x0000ff00);
      colour.blue = (rgb & 0x000000ff) << 8;

      status = XAllocColor (display,
			    XDefaultColormapOfScreen(screen),
			    &colour);

      colours[i] = colour.pixel;
      colours[i+1] = colour.pixel;

#ifdef SHM
      colour_translation_table[i] = colours[i];
      colour_translation_table[i+1] = colours[i+1];
#endif
    }

  for (i=0;i<256;i++)
    {
      xgcvl.background = colours[0];
      xgcvl.foreground = colours[i];

      gc_colour[i] = XCreateGC (display, window,
				GCForeground | GCBackground,
				&xgcvl);
    }

  xgcvl.background = colours[0];
  xgcvl.foreground = colours[0];

  gc = XCreateGC (display, window,
		  GCForeground | GCBackground,
		  &xgcvl);

#ifndef SHM
  XFillRectangle (display, pixmap, gc, 0, 0,
		  window_width, window_height);
#endif

  XMapWindow (display, window);

  XSync (display, False);
/*
   ============================
   Storage for Atari 800 Screen
   ============================
*/
  image_data = (UBYTE*) malloc (ATARI_WIDTH * ATARI_HEIGHT);
  if (!image_data)
    {
      printf ("Failed to allocate space for image\n");
      exit (1);
    }

  consol = 7;

  if (x11bug)
    {
      printf ("Initial X11 controller configuration\n");
      printf ("------------------------------------\n\n");
      printf ("Keypad is "); Atari_WhatIs (keypad_mode); printf ("\n");
      printf ("Mouse is "); Atari_WhatIs (mouse_mode); printf ("\n");
      printf ("/dev/js0 is "); Atari_WhatIs (js0_mode); printf ("\n");
      printf ("/dev/js1 is "); Atari_WhatIs (js1_mode); printf ("\n");
    }
}

int Atari_Exit (int run_monitor)
{
  int restart;

  if (run_monitor)
    restart = monitor();
  else
    restart = FALSE;

  if (!restart)
    {
      free (image_data);

      XSync (display, True);

#ifdef SHM
      XDestroyImage (image);
#else
      XFreePixmap (display, pixmap);
#endif
      XUnmapWindow (display, window);
      XDestroyWindow (display, window);
      XCloseDisplay (display);

#ifdef LINUX_JOYSTICK
      if (js0 != -1)
	close (js0);

      if (js1 != -1)
	close (js1);
#endif

#ifdef NAS
      NAS_Exit ();
#endif
    }

  return restart;
}

#ifndef SHM
void Atari_ScanLine_Flush ()
{
  if (windowsize == Small)
    {
      if (npoints != 0)
	{
	  XDrawPoints (display, pixmap, gc_colour[last_colour],
		       points, npoints, CoordModeOrigin);
	  npoints = 0;
	  modified = TRUE;
	}
    }
  else
    {
      if (nrects != 0)
	{
	  XFillRectangles (display, pixmap, gc_colour[last_colour],
			   rectangles, nrects);
	  nrects = 0;
	  modified = TRUE;
	}
    }

  last_colour = -1;
}
#endif

void Atari_DisplayScreen (UBYTE *screen)
{
#ifdef SHM
  XShmPutImage (display, window, gc, image, 0, 0, 0, 0,
		window_width, window_height, 0);
#else
  UBYTE *scanline_ptr = image_data;
  int xpos;
  int ypos;
  int i;

  modified = FALSE;

  for (ypos=0;ypos<ATARI_HEIGHT;ypos++)
    {
      for (xpos=0;xpos<ATARI_WIDTH;xpos++)
	{
	  UBYTE colour;

	  colour = *screen++;

	  if (colour != *scanline_ptr)
	    {
	      int flush = FALSE;

	      if (windowsize == Small)
		{
		  if (npoints == NPOINTS)
		    flush = TRUE;
		}
	      else
		{
		  if (nrects == NRECTS)
		    flush = TRUE;
		}

	      if (colour != last_colour)
		flush = TRUE;

	      if (flush)
		{
		  Atari_ScanLine_Flush ();
		  last_colour = colour;
		}

	      if (windowsize == Small)
		{
		  points[npoints].x = xpos;
		  points[npoints].y = ypos;
		  npoints++;
		}
	      else if (windowsize == Large)
		{
		  rectangles[nrects].x = xpos << 1;
		  rectangles[nrects].y = ypos << 1;
		  rectangles[nrects].width = 2;
		  rectangles[nrects].height = 2;
		  nrects++;
		}
	      else
		{
		  rectangles[nrects].x = xpos + xpos + xpos;
		  rectangles[nrects].y = ypos + ypos + ypos;
		  rectangles[nrects].width = 3;
		  rectangles[nrects].height = 3;
		  nrects++;
		}

	      *scanline_ptr++ = colour;
	    }
	  else
	    {
	      scanline_ptr++;
	    }
	}
    }

  Atari_ScanLine_Flush ();

  if (modified)
    {
      XCopyArea (display, pixmap, window, gc, 0, 0,
		 window_width, window_height, 0, 0);
    }
#endif

  keypad_trig = 1;

#ifdef XVIEW
#ifdef FPS_MONITOR
  {
    double curtime;

    gettimeofday (&tp, &tzp);
    curtime = tp.tv_sec + (tp.tv_usec / 1000000.0);

    nframes++;

    if ((curtime - basetime) >= 2.0)
      {
	static char gash[64];

	sprintf (gash, "%.2f FPS", (double)nframes / (curtime - basetime));

	xv_set (frame,
		FRAME_LEFT_FOOTER, gash,
		NULL);

	nframes = 0;
	basetime = curtime;
      }
  }
#endif

  notify_dispatch ();
  XFlush (display);
#endif

#ifdef MOTIF
  while (XtAppPending(app))
    {
      static XEvent event;

      XtAppNextEvent (app, &event);
      XtDispatchEvent (&event);
    }
#endif

#ifdef NAS
  NAS_UpdateSound ();
#endif
}

int Atari_Keyboard (void)
{
  int	keycode = AKEY_NONE;

#ifdef XVIEW
  keycode = xview_keycode;
  xview_keycode = AKEY_NONE;
#else
#ifdef MOTIF
  keycode = xview_keycode;
  xview_keycode = AKEY_NONE;
#else
  if (XEventsQueued (display, QueuedAfterFlush) > 0)
    {
      XEvent	event;

      XNextEvent (display, &event);
      keycode = GetKeyCode (&event);
    }
#endif
#endif

  return keycode;
}

mouse_joystick (int mode)
{
  Window root_return;
  Window child_return;
  int root_x_return;
  int root_y_return;
  int win_x_return;
  int win_y_return;
  int mask_return;

  mouse_stick = 0x0f;

  XQueryPointer (display, window, &root_return, &child_return,
		 &root_x_return, &root_y_return,
		 &win_x_return, &win_y_return,
		 &mask_return);

  if (mode < 5)
    {
      int center_x;
      int center_y;
      int threshold;

      if (windowsize == Small)
	{
	  center_x = ATARI_WIDTH / 2;
	  center_y = ATARI_HEIGHT / 2;
	  threshold = 32;
	}
      else if (windowsize == Large)
	{
	  center_x = (ATARI_WIDTH * 2) / 2;
	  center_y = (ATARI_HEIGHT * 2) / 2;
	  threshold = 64;
	}
      else
	{
	  center_x = (ATARI_WIDTH * 3) / 2;
	  center_y = (ATARI_HEIGHT * 3) / 2;
	  threshold = 96;
	}

      if (win_x_return < (center_x - threshold))
	mouse_stick &= 0xfb;
      if (win_x_return > (center_x + threshold))
	mouse_stick &= 0xf7;
      if (win_y_return < (center_y - threshold))
	mouse_stick &= 0xfe;
      if (win_y_return > (center_y + threshold))
	mouse_stick &= 0xfd;
    }
  else
    {
      if (mask_return)
	mouse_stick &= 0xfb;
    }
}

#ifdef LINUX_JOYSTICK

void read_joystick (int js, int centre_x, int centre_y)
{
  const int threshold = 50;
  int status;

  mouse_stick = 0x0f;

  status = read (js, &js_data, JS_RETURN);
  if (status != JS_RETURN)
    {
      perror ("/dev/js");
      exit (1);
    }

  if (js_data.x < (centre_x - threshold))
    mouse_stick &= 0xfb;
  if (js_data.x > (centre_x + threshold))
    mouse_stick &= 0xf7;
  if (js_data.y < (centre_y - threshold))
    mouse_stick &= 0xfe;
  if (js_data.y > (centre_y + threshold))
    mouse_stick &= 0xfd;
}
#endif

int Atari_PORT (int num)
{
  int nibble_0 = 0x0f;
  int nibble_1 = 0x0f;

  if (num == 0)
    {
      if (keypad_mode == 0)
	nibble_0 = keypad_stick;
      else if (keypad_mode == 1)
	nibble_1 = keypad_stick;

      if (mouse_mode == 0)
	{
	  mouse_joystick (mouse_mode);
	  nibble_0 = mouse_stick;
	}
      else if (mouse_mode == 1)
	{
	  mouse_joystick (mouse_mode);
	  nibble_1 = mouse_stick;
	}

#ifdef LINUX_JOYSTICK
      if (js0_mode == 0)
	{
	  read_joystick (js0, js0_centre_x, js0_centre_y);
	  nibble_0 = mouse_stick;
	}
      else if (js0_mode == 1)
	{
	  read_joystick (js0, js0_centre_x, js0_centre_y);
	  nibble_1 = mouse_stick;
	}

      if (js1_mode == 0)
	{
	  read_joystick (js1, js1_centre_x, js1_centre_y);
	  nibble_0 = mouse_stick;
	}
      else if (js1_mode == 1)
	{
	  read_joystick (js1, js1_centre_x, js1_centre_y);
	  nibble_1 = mouse_stick;
	}
#endif
    }
  else
    {
      if (keypad_mode == 2)
	nibble_0 = keypad_stick;
      else if (keypad_mode == 3)
	nibble_1 = keypad_stick;

      if (mouse_mode == 2)
	{
	  mouse_joystick (mouse_mode);
	  nibble_0 = mouse_stick;
	}
      else if (mouse_mode == 3)
	{
	  mouse_joystick (mouse_mode);
	  nibble_1 = mouse_stick;
	}

#ifdef LINUX_JOYSTICK
      if (js0_mode == 2)
	{
	  read_joystick (js0, js0_centre_x, js0_centre_y);
	  nibble_0 = mouse_stick;
	}
      else if (js0_mode == 3)
	{
	  read_joystick (js0, js0_centre_x, js0_centre_y);
	  nibble_1 = mouse_stick;
	}

      if (js1_mode == 2)
	{
	  read_joystick (js1, js1_centre_x, js1_centre_y);
	  nibble_0 = mouse_stick;
	}
      else if (js1_mode == 3)
	{
	  read_joystick (js1, js1_centre_x, js1_centre_y);
	  nibble_1 = mouse_stick;
	}
#endif
    }

  return (nibble_1 << 4) | nibble_0;
}

int Atari_TRIG (int num)
{
  int	trig = 1;	/* Trigger not pressed */

  if (num == keypad_mode)
    {
      trig = keypad_trig;
    }

  if (num == mouse_mode)
    {
      Window	root_return;
      Window	child_return;
      int	root_x_return;
      int	root_y_return;
      int	win_x_return;
      int	win_y_return;
      int	mask_return;

      if (XQueryPointer (display, window, &root_return, &child_return,
			 &root_x_return, &root_y_return,
			 &win_x_return, &win_y_return,
			 &mask_return))
	{
	  if (mask_return)
	    trig = 0;
	}
    }

#ifdef LINUX_JOYSTICK
  if (num == js0_mode)
    {
      int status;

      status = read (js0, &js_data, JS_RETURN);
      if (status != JS_RETURN)
	{
	  perror ("/dev/js0");
	  exit (1);
	}

      if (js_data.buttons & 0x01)
	trig = 0;
      else
	trig = 1;

      if (js_data.buttons & 0x02)
	xview_keycode = AKEY_SPACE;
    }

  if (num == js1_mode)
    {
      int status;

      status = read (js1, &js_data, JS_RETURN);
      if (status != JS_RETURN)
	{
	  perror ("/dev/js1");
	  exit (1);
	}

      trig = (js_data.buttons & 0x0f) ? 0 : 1;
    }
#endif

  return trig;
}

int Atari_POT (int num)
{
  int	pot;

  if (num == (mouse_mode - 4))
    {
      Window root_return;
      Window child_return;
      int root_x_return;
      int root_y_return;
      int win_x_return;
      int win_y_return;
      int mask_return;

      if (XQueryPointer (display, window, &root_return,
			 &child_return, &root_x_return, &root_y_return,
			 &win_x_return, &win_y_return, &mask_return))
	{
	  if (windowsize == Small)
	    {
	      pot = ((float)((ATARI_WIDTH) - win_x_return) /
		     (float)(ATARI_WIDTH)) * 228;
	    }
	  else if (windowsize == Large)
	    {
	      pot = ((float)((ATARI_WIDTH * 2) - win_x_return) /
		     (float)(ATARI_WIDTH * 2)) * 228;
	    }
	  else
	    {
	      pot = ((float)((ATARI_WIDTH * 3) - win_x_return) /
		     (float)(ATARI_WIDTH * 3)) * 228;
	    }
	}
    }
  else
    {
      pot = 228;
    }

  return pot;
}

int Atari_CONSOL (void)
{
  int temp;

  temp = consol;
  consol = 0x07;

  return temp;
}

#ifndef NAS
int Atari_AUDC (int channel, int byte)
{
}

int Atari_AUDF (int channel, int byte)
{
}

int Atari_AUDCTL (int byte)
{
}
#endif
