/* --------------------------------------------------------------------------
 * machdep.c:   Copyright (c) Mark P Jones 1991-1998.   All rights reserved.
 *              See NOTICE for details and conditions of use etc...
 *              Hugs version 1.3b, January 1998
 *
 * Machine dependent code
 * RISCOS specific code provided by Bryan Scatergood, JBS
 * ------------------------------------------------------------------------*/

#if UNIX
#include <signal.h>
#include <sys/ioctl.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <time.h>
#endif

#if MSWIN
#include <dos.h>
#include <io.h>
#include <dir.h>
#include <stdlib.h>
#include <mem.h>
#include <sys\stat.h>
#include <time.h>
extern HCURSOR HandCursor;	      /* Forward references to cursors   */
extern HCURSOR GarbageCursor;
extern HCURSOR SaveCursor;
static void    local DrawStatusLine	Args((HWND));
#endif

#if BCC
#include <dos.h>
#include <conio.h>
#include <io.h>
#include <stdlib.h>
#include <mem.h>
#include <sys\stat.h>
#include <time.h>
extern unsigned _stklen = 8000;		/* Allocate an 8k stack segment	   */
#endif

#if VC32
#include <dos.h>
#include <conio.h>
#include <stdlib.h>
#include <sys\stat.h>
#include <signal.h>
#include <time.h>
#endif

#if DJGPP2
#include <dos.h>
#include <stdlib.h>
#include <std.h>
#include <signal.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <time.h>
#endif

#if RISCOS
#include <assert.h>
#include <signal.h>
#include "swis.h"
#include "os.h"
#endif

#if ATARI
#include <signal.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#endif

#if MACWP
#include <SIOUX.h>
#include <signal.h>
#include <sys/types.h>
//#include <stat.h>
#include <sys/stat.h>
#include <unistd.h>
#endif

/* --------------------------------------------------------------------------
 * Find information about a file:
 * ------------------------------------------------------------------------*/

#if RISCOS
typedef struct { unsigned hi, lo; } Time;
#define timeChanged(now,thn)	(now.hi!=thn.hi || now.lo!=thn.lo)
#define timeSet(var,tm)		var.hi = tm.hi; var.lo = tm.lo
#else
typedef time_t Time;
#define timeChanged(now,thn)	(now!=thn)
#define timeSet(var,tm)		var = tm
#endif

static Void local getFileInfo	Args((String, Time *, Long *));

static Void local getFileInfo(s,tm,sz)	/* find time stamp and size of file*/
String s;
Time   *tm;
Long   *sz; {
#if RISCOS				/* get file info for RISCOS -- JBS */
    os_regset r;			/* RISCOS PRM p.850 and p.837	   */
    r.r[0] = 17;			/* Read catalogue, no path	   */
    r.r[1] = (int)s;
    os_swi(OS_File, &r);
    if(r.r[0] == 1 && (r.r[2] & 0xFFF00000) == 0xFFF00000) {
	tm->hi = r.r[2] & 0xFF;		/* Load address (high byte)	   */
	tm->lo = r.r[3];		/* Execution address (low 4 bytes) */
    }
    else				/* Not found, or not time-stamped  */
	tm->hi = tm->lo = 0;
    *sz = (Long)(r.r[0] == 1 ? r.r[4] : 0);
#else					/* normally just use stat()	   */
    static struct stat scbuf;
    stat(s,&scbuf);
    *tm = scbuf.st_mtime;
    *sz = (Long)(scbuf.st_size);
#endif
}

#if RISCOS				/* RISCOS needs access()	   */
int access(char *s, int dummy) {	/* Give 1 iff cannot access file s */
    os_regset r;			/* RISCOS PRM p.850	-- JBS	   */
    assert(dummy == 0);
    r.r[0] = 17; /* Read catalogue, no path */
    r.r[1] = (int)s;
    os_swi(OS_File, &r);
    return r.r[0] != 1;
}

int namecmp(char *filename, char *spec){/* For filename extension hacks	   */
    while(*spec)
        if  (tolower(*filename) != *spec++)
	    return 0;
	else
	    ++filename;
    return *filename == '.';
}
#endif

#if MACWP				/* access() is missing in CW libs */
int access(char *s, int dummy) {
    struct stat scbuf;
    return stat(s,&scbuf);
}
#endif

/* --------------------------------------------------------------------------
 * Search for script files on the HUGS path:
 * ------------------------------------------------------------------------*/

static String local normPath	Args((String));
static Void   local searchChr 	Args((Int));
static Void   local searchStr 	Args((String));
static Bool   local tryEndings	Args((String));

#if (MSWIN | DJGPP | VC32 | BCC)
#define SLASH			'\\'
#define PATHSEP			';'
#else
#if MACWP
#define SLASH			':'
#define PATHSEP			';'
#else
#define SLASH			'/'
#define PATHSEP			':'
#endif
#endif
#ifndef valid
#define valid(file)		(access(file,0)==0)
#endif

static String endings[] = { "", ".hs", ".lhs", 0 };
static char   searchBuf[FILENAME_MAX+1];
static Int    searchPos;

#if MACWP
static char   hugsdir[FILENAME_MAX+1];	/* Directory where Hugs was found */
#endif

#define searchReset(n)		searchBuf[searchPos=(n)]='\0'

static local String normPath(s)	/* Try, as much as possible, to normalize  */
String s; {			/* a pathname in some appropriate manner.  */
#if (MSWIN | BCC)
    static char path[FILENAME_MAX+1];
    _fullpath(path,s,FILENAME_MAX+1);		/* Find absolute pathname, */
    strlwr(path);				/* and convert to lowercase*/
    return path;
#else
#if DJGPP2
    static char path[FILENAME_MAX+1];
    strcpy(path,s);
    strlwr(path);
    return path;
#else
    return s;
#endif
#endif
}

static Void local searchChr(c)	/* Add single character to search buffer   */
Int c; {
    if (searchPos<FILENAME_MAX) {
	searchBuf[searchPos++] = (char)c;
	searchBuf[searchPos]   = '\0';
    }
}

static Void local searchStr(s)	/* Add string to search buffer		   */
String s; {
    while (*s && searchPos<FILENAME_MAX)
	searchBuf[searchPos++] = *s++;
    searchBuf[searchPos] = '\0';
}

static Bool local tryEndings(s)	/* Try each of the listed endings	   */
String s; {
    Int i = 0;
    searchStr(s);
    for (; endings[i]; ++i) {
	Int save = searchPos;
	searchStr(endings[i]);
	if (valid(searchBuf))
	    return TRUE;
	searchReset(save);
    }
    return FALSE;
}

String findPathname(along,name)	/* Look for a file along specified path	   */
String along;			/* If nonzero, a path prefix from along is */
String name; {			/* used as the first prefix in the search. */
    String pathpt = hugsPath;

    searchReset(0);
    if (along) {		/* Was a path for an existing file given?  */
	Int last = (-1);
	Int i    = 0;
	for (; along[i]; i++) {
	    searchChr(along[i]);
	    if (along[i]==SLASH)
		last = i;
	}
	searchReset(last+1);
    }
    if (tryEndings(name))
	return normPath(searchBuf);

    if (pathpt && *pathpt) {	/* Otherwise, we look along the OHUGSPATH   */
	Bool more = TRUE;
	do {
	    searchReset(0);
	    if (*pathpt) {
		if (*pathpt!=PATHSEP) {
#if MACWP			/* Pre-define one MPW-style "shell-variable" */
		    if (strncmp(pathpt,"{OHUGS}",7)==0) {
		    	searchStr(hugsdir);
		    	pathpt += 7;
		    }
#endif
		    do
			searchChr(*pathpt++);
		    while (*pathpt && *pathpt!=PATHSEP);
		    searchChr(SLASH);
		}
		if (*pathpt==PATHSEP)
		    pathpt++;
		else
		    more = FALSE;
	    }
	    else
		more = FALSE;
	    if (tryEndings(name))
		return normPath(searchBuf);
	} while (more);
    }

    searchReset(0);		/* As a last resort, look for file in the  */
    tryEndings(name);		/* current directory			   */
    return normPath(searchBuf);
}

/* --------------------------------------------------------------------------
 * Get time/date stamp for inclusion in compiled files:
 * ------------------------------------------------------------------------*/

#if PROFILING
String timeString() {			/* return time&date string	   */
    time_t clock;			/* must end with '\n' character	   */
    time(&clock);
    return(ctime(&clock));
}
#endif

/* --------------------------------------------------------------------------
 * Garbage collection notification:
 * ------------------------------------------------------------------------*/

Bool gcMessages = FALSE;		/* TRUE => print GC messages	   */

Void gcStarted() {			/* notify garbage collector start  */
#if MSWIN
    SaveCursor = SetCursor(GarbageCursor);
#endif
    if (gcMessages) {
	printf("{{Gc");
	fflush(stdout);
    }
}

Void gcScanning() {			/* notify garbage collector scans  */
    if (gcMessages) {
	putchar(':');
	fflush(stdout);
    }
}

Void gcRecovered(recovered,flatrec)	/* notify garbage collection done  */
Int recovered, flatrec; {
    if (gcMessages) {
#ifdef SHOW_FLATREC
	printf("%d,%d}}",recovered,flatrec);
#else
	printf("%d}}",recovered);
#endif
	fflush(stdout);
    }
#if MSWIN
    SetCursor(SaveCursor);
#endif
}

Cell *CStackBase;			/* Retain start of C control stack */

#if RISCOS				/* Stack traversal for RISCOS	   */

/* Warning: The following code is specific to the Acorn ARM under RISCOS
   (and C4).  We must explicitly walk back through the stack frames, since
   the stack is extended from the heap. (see PRM pp. 1757).  gcCStack must
   not be modified, since the offset '5' assumes that only v1 is used inside
   this function. Hence we do all the real work in gcARM.
*/
		  
#define spreg 13 /* C3 has SP=R13 */

#define previousFrame(fp)	((int *)((fp)[-3]))
#define programCounter(fp)	((int *)((*(fp)-12) & ~0xFC000003))
#define isSubSPSP(w)		(((w)&dontCare) == doCare)
#define doCare			(0xE24DD000)  /* SUB r13,r13,#0 */
#define dontCare		(~0x00100FFF) /* S and # bits   */
#define immediateArg(x)		( ((x)&0xFF) << (((x)&0xF00)>>7) )

static void gcARM(int *fp) {
    int si = *programCounter(fp);	/* Save instruction indicates how */
					/* many registers in this frame   */
    int *regs = fp - 4;
    if (si & (1<<0)) markWithoutMove(*regs--);
    if (si & (1<<1)) markWithoutMove(*regs--);
    if (si & (1<<2)) markWithoutMove(*regs--);
    if (si & (1<<3)) markWithoutMove(*regs--);
    if (si & (1<<4)) markWithoutMove(*regs--);
    if (si & (1<<5)) markWithoutMove(*regs--);
    if (si & (1<<6)) markWithoutMove(*regs--);
    if (si & (1<<7)) markWithoutMove(*regs--);
    if (si & (1<<8)) markWithoutMove(*regs--);
    if (si & (1<<9)) markWithoutMove(*regs--);
    if (previousFrame(fp)) {
	/* The non-register stack space is for the previous frame is above
	   this fp, and not below the previous fp, because of the way stack
	   extension works. It seems the only way of discovering its size is
	   finding the SUB sp, sp, #? instruction by walking through the code
	   following the entry point.
	*/
	int *oldpc = programCounter(previousFrame(fp));
	int fsize = 0, i;
	for(i = 1; i < 6; ++i)
	    if(isSubSPSP(oldpc[i])) fsize += immediateArg(oldpc[i]) / 4;
	for(i=1; i<=fsize; ++i)
	    markWithoutMove(fp[i]);
    }
}

void gcCStack() {
    int dummy;
    int *fp = 5 + &dummy;
    while (fp) {
	gcARM(fp);
	fp = previousFrame(fp);
    }
}

#else			/* Garbage collection for standard stack machines  */

Void gcCStack() {			/* Garbage collect elements off    */
    Cell stackTop = NIL;		/* C stack			   */
    Cell *ptr = &stackTop;
#if SMALL_HUGS || (MACWP && __MC68K__)	/* 68K Mac stacks are 2-byte aligned */
    if (((long)(ptr) - (long)(CStackBase))&1)
	fatal("gcCStack");
#else 
    if (((long)(ptr) - (long)(CStackBase))&3)
	fatal("gcCStack");
#endif

#define StackGrowsDown	while (ptr<=CStackBase) markWithoutMove(*ptr++)
#define StackGrowsUp	while (ptr>=CStackBase) markWithoutMove(*ptr--)
#define GuessDirection	if (ptr>CStackBase) StackGrowsUp; else StackGrowsDown
#if HPUX
    GuessDirection;
#else
    StackGrowsDown;
#if MACWP && __MC68K__				/* Search stack twice to find */
    ptr = (Cell *)((long)(&stackTop) + 2);	/* all Cells on a 68K Mac     */
    StackGrowsDown;
#endif
#endif
#undef  StackGrowsDown
#undef  StackGrowsUp
#undef  GuessDirection
}
#endif

/* --------------------------------------------------------------------------
 * Terminal dependent stuff:
 * ------------------------------------------------------------------------*/

#if   (TERMIO_IO | SGTTY_IO | TERMIOS_IO)

#if TERMIO_IO
#include <termio.h>
typedef  struct termio   TermParams;
#define  getTerminal(tp) ioctl(fileno(stdin),TCGETA,&tp)
#define  setTerminal(tp) ioctl(fileno(stdin),TCSETAF,&tp)
#define  noEcho(tp)      tp.c_lflag    &= ~(ICANON | ECHO); \
			 tp.c_cc[VMIN]  = 1;		    \
			 tp.c_cc[VTIME] = 0;
#endif

#if SGTTY_IO
#include <sgtty.h>
typedef  struct sgttyb   TermParams;
#define  getTerminal(tp) ioctl(fileno(stdin),TIOCGETP,&tp)
#define  setTerminal(tp) ioctl(fileno(stdin),TIOCSETP,&tp)
#if HPUX
#define  noEcho(tp)      tp.sg_flags |= RAW; tp.sg_flags &= (~ECHO);
#else
#define  noEcho(tp)      tp.sg_flags |= CBREAK; tp.sg_flags &= (~ECHO);
#endif
#endif

#if TERMIOS_IO
#include <termios.h>
typedef  struct termios  TermParams;
#define  getTerminal(tp) tcgetattr(fileno(stdin), &tp)
#define  setTerminal(tp) tcsetattr(fileno(stdin), TCSAFLUSH, &tp)
#define  noEcho(tp)      tp.c_lflag    &= ~(ICANON | ECHO); \
			 tp.c_cc[VMIN]  = 1;                \
			 tp.c_cc[VTIME] = 0;
#endif

static Bool messedWithTerminal = FALSE;
static TermParams originalSettings;

Void normalTerminal() {			/* restore terminal initial state  */
    if (messedWithTerminal)
	setTerminal(originalSettings);
}

Void noechoTerminal() {			/* set terminal into noecho mode   */
    TermParams settings;

    if (!messedWithTerminal) {
	getTerminal(originalSettings);
	messedWithTerminal = TRUE;
    }
    getTerminal(settings);
    noEcho(settings);
    setTerminal(settings);
}

Int getTerminalWidth() {		/* determine width of terminal	   */
#ifdef TIOCGWINSZ
#ifdef _M_UNIX				/* SCO Unix 3.2.4 defines TIOCGWINSZ*/
#include <sys/stream.h>			/* Required by sys/ptem.h	   */
#include <sys/ptem.h>			/* Required to declare winsize	   */
#endif
    static struct winsize terminalSize;
    ioctl(fileno(stdout),TIOCGWINSZ,&terminalSize);
    return (terminalSize.ws_col==0)? 80 : terminalSize.ws_col;
#else
    return 80;
#endif
}

Int readTerminalChar() {		/* read character from terminal	   */
    return getchar();			/* without echo, assuming that	   */
}					/* noechoTerminal() is active...   */
#endif

#if DOS_IO
static Bool terminalEchoReqd = TRUE;

Int getTerminalWidth() {		/* PC screen is fixed 80 chars	   */
    return 80;
}

Void normalTerminal() {			/* restore terminal initial state  */
    terminalEchoReqd = TRUE;
}

Void noechoTerminal() {			/* turn terminal echo on/off	   */
    terminalEchoReqd = FALSE;
}

Int readTerminalChar() {		/* read character from terminal	   */
    if (terminalEchoReqd)
	return getchar();
    else {
	Int c = getch();
	return c=='\r' ? '\n' : c;
    }
}
#endif

#if RISCOS
Int getTerminalWidth() {
    int dummy, width;
    (void) os_swi3r(OS_ReadModeVariable, -1, 1, 0, &dummy, &dummy, &width);
    return width+1;
}

Void normalTerminal() {			/* restore terminal initial state  */
}					/* (not yet implemented)	   */

Void noechoTerminal() {			/* turn terminal echo on/off	   */
}					/* (not yet implemented)	   */

Int readTerminalChar() {		/* read character from terminal	   */
    return getchar();
}
#endif

#if MACWP
Main hugsmain Args((Int, String []));
Void hugsexit Args((Int));

/* DropUNIX lib calls this main1() after startup.                	   */
/* Do some initialization here, before calling main == hugsmain in hugs.c. */
void main1     Args((Int,String []));
void main1(argc,argv)
Int    argc;
String argv[]; {
    int i;

    getcwd(hugsdir,FILENAME_MAX);	/* Initialize */
    for (i = 1; i < argc; i++) {
	char *p = strrchr(argv[i], SLASH);
	if (p) {
 	    int tmpc = *p; *p = 0;
	    chdir(argv[i]);		/* Most reasonable wd */
	    *p = tmpc;
	    break;
	}
    }
    SIOUXSettings.asktosaveonclose = FALSE;
    SIOUXSettings.tabspaces = 8;
    hugsmain(argc,argv);		/* Start REAL hugs */
}

Void hugsexit(n)	/* Catch normal exit calls so that the         */
Int n; {		/* console window may be closed automatically. */
    SIOUXSettings.autocloseonquit = !n;
    exit(n);
}

#define main hugsmain		/* Effective scope is remainder of hugs.c */
#define exit hugsexit		/* Effective scope is remainder of hugs.c */


Int getTerminalWidth() {	/* Default setting for SIOUX.        */
    return 80;			/* Dynamic value cannot be obtained. */
}

Void normalTerminal() {
}

Void noechoTerminal() {		/* Cannot turn off echo using SIOUX. */
}

Int readTerminalChar() {
    return getc(stdin);
}
#endif

/* --------------------------------------------------------------------------
 * Interrupt handling:
 * ------------------------------------------------------------------------*/

Bool    broken         = FALSE;
static  Bool breakReqd = FALSE;
static  sigProto(ignoreBreak);

Bool breakOn(reqd)			/* set break trapping on if reqd,  */
Bool reqd; {				/* or off otherwise, returning old */
    Bool old  = breakReqd;

    breakReqd = reqd;
    if (reqd) {
	if (broken) {			/* repond to break signal received */
	    broken = FALSE;		/* whilst break trap disabled	   */
	    sigRaise(breakHandler);
	}
#if VC32
	ctrlbrk(ignoreBreak);
#else
	ctrlbrk(breakHandler);
#endif
    }
    else
	ctrlbrk(ignoreBreak);

    return old;
}

static sigHandler(ignoreBreak) {	/* record but don't respond to break*/
    ctrlbrk(ignoreBreak);
    broken = TRUE;
    sigResume;
}

/* --------------------------------------------------------------------------
 * Shell escapes:
 * ------------------------------------------------------------------------*/

Bool startEdit(line,name) 		/* Start editor on file name at	   */
Int    line;				/* given line.  Both name and line */
String name; {				/* or just line may be zero	   */
    static char editorCmd[FILENAME_MAX+1];

    if (hugsEdit && *hugsEdit) {	/* Check that editor configured	   */
	Int n     = FILENAME_MAX;
	String he = hugsEdit;
	String ec = editorCmd;
	String rd = NULL;		/* Set to nonnull to redo ...	   */

	for (; n>0 && *he && *he!=' '; n--)
	    *ec++ = *he++;		/* Copy editor name to buffer	   */

	if (name && line && n>1 && *he){/* Name, line, and enough space    */
	    rd = ec;			/* save, in case we don't find name*/
	    while (n>0 && *he) {
		if (*he=='%') {
		    if (*++he=='d' && n>10) {
			sprintf(ec,"%d",line);
			he++;
		    }
		    else if (*he=='s' && n>strlen(name)) {
			strcpy(ec,name);
			rd = NULL;
			he++;
		    }
		    else if (*he=='%' && n>1) {
			strcpy(ec,"%");
			he++;
		    }
		    else		/* Ignore % char if not followed   */
			*ec = '\0';	/* by one of d, s, or %,	   */
		    for (; *ec && n>0; n--)
			ec++;
		}   /* ignore % followed by anything other than d, s, or % */
		else {			/* Copy other characters across	   */
		    *ec++ = *he++;
		    n--;
		}
	    }
	}
	else
	    line = 0;

	if (rd) {			/* If file name was not included   */
	    ec   = rd;
	    line = 0;
	}

	if (name && line==0 && n>1) {	/* Name, but no line ...	   */
	    *ec++ = ' ';
	    for (; n>0 && *name; n--)	/* ... just copy file name	   */
		*ec++ = *name++;
	}

	*ec = '\0';				/* Add terminating null byte	   */
    }
    else {
	ERRMSG(0) "Hugs is not configured to use an editor"
	EEND;
    }

#if MSWIN
    WinExec(editorCmd, SW_SHOW);
    return FALSE;
#else
    if (shellEsc(editorCmd))
	printf("Warning: Editor terminated abnormally\n");
    return TRUE;
#endif
}

Int shellEsc(s)				/* run a shell command (or shell)  */
String s; {
#if UNIX
    if (s[0]=='\0')
	s = fromEnv("SHELL","/bin/sh");
#endif
    return system(s);
}

#if RISCOS				/* RISCOS also needs a chdir()	   */
int chdir(char *s) {			/* RISCOS PRM p. 885	-- JBS	   */
    return os_swi2(OS_FSControl + XOS_Bit, 0, (int)s) != NULL;
}
#endif

/* --------------------------------------------------------------------------
 * Floating point support:
 * ------------------------------------------------------------------------*/

#if HAS_FLOATS
#if BREAK_FLOATS
static union {
    Float  flVal;
    struct {
	Cell flPart1,flPart2;
    }	   clVal;
} fudgeCoerce;

Cell part1Float(fl)
FloatPro fl; {
    fudgeCoerce.flVal = fl;
    return fudgeCoerce.clVal.flPart1;
}

Cell part2Float(fl)
FloatPro fl; {
    fudgeCoerce.flVal = fl;
    return fudgeCoerce.clVal.flPart2;
}

FloatPro floatFromParts(c1,c2)
Cell c1, c2; {
    fudgeCoerce.clVal.flPart1 = c1;
    fudgeCoerce.clVal.flPart2 = c2;
    return fudgeCoerce.flVal;
}

Cell bfTemp = NIL;

Cell mkFloat(fl)
FloatPro fl; {
    Cell p1,p2;
    fudgeCoerce.flVal = fl;
    bfTemp = mkInt(fudgeCoerce.clVal.flPart1);
    p2     = mkInt(fudgeCoerce.clVal.flPart2);
    p1     = bfTemp;
    bfTemp = NIL;
    return pair(FLOATCELL,pair(p1,p2));
}

FloatPro floatOf(c)
Cell c; {
    fudgeCoerce.clVal.flPart1 = intOf(fst(snd(c)));
    fudgeCoerce.clVal.flPart2 = intOf(snd(snd(c)));
    return fudgeCoerce.flVal;
}

#else /* !BREAK_FLOATS */
static union {
    Float flVal;
    Cell  clVal;
} fudgeCoerce;

Cell mkFloat(fl)
FloatPro fl; {
    fudgeCoerce.flVal = fl;
    return pair(FLOATCELL,fudgeCoerce.clVal);
}

FloatPro floatOf(c)
Cell c; {
    fudgeCoerce.clVal = snd(c);
    return fudgeCoerce.flVal;
}
#endif

String floatToString(fl)			/* Make sure that floating   */
FloatPro fl; {					/* point values print out in */
    static char buffer1[32];			/* a form in which they could*/
    static char buffer2[32];			/* also be entered as floats */
    Int i=0, j=0;

    sprintf(buffer1,FloatFMT,fl);
    while (buffer1[i] && strchr("eE.",buffer1[i])==0)
	buffer2[j++] = buffer1[i++];
    if (buffer1[i]!='.') {
        buffer2[j++] = '.';
	buffer2[j++] = '0';
    }
    while (buffer2[j++]=buffer1[i++])
	;
    return buffer2;
}

FloatPro stringToFloat(s)
String s; {
    return atof(s);
}
#else
Cell mkFloat(fl)
FloatPro fl; {
    internal("mkFloat");
    return 0;/*NOTREACHED*/
}

FloatPro floatOf(c)
Cell c; {
    internal("floatOf");
    return 0;/*NOTREACHED*/
}

String floatToString(fl)
FloatPro fl; {
    internal("floatToString");
    return "";/*NOTREACHED*/
}

FloatPro stringToFloat(s)
String s; {
    internal("stringToFloat");
    return 0;
}
#endif

/* --------------------------------------------------------------------------
 * Machine dependent control:
 * ------------------------------------------------------------------------*/

/* Exit as quickly as possible, but at least restore terminal */
static doexit(int i) {
  machdep(EXIT);
  exit(i);
}

#if UNIX
static sigHandler(panic) {		/* exit in a panic, on receipt of  */
    everybody(EXIT);			/* an unexpected signal		   */
    fprintf(stderr,"\nUnexpected signal\n");
    doexit(1);
    sigResume;/*NOTREACHED*/
}
#endif

Void machdep(what)			/* Handle machine specific	   */
Int what; {				/* initialisation etc..		   */
    switch (what) {
        case MARK    :
#if HAS_FLOATS && BREAK_FLOATS
		       mark(bfTemp);
#endif
		       break;
#if UNIX
        case INSTALL :
#ifdef SIGHUP
		       signal(SIGHUP,panic);
#endif
#ifdef SIGQUIT
		       signal(SIGQUIT,panic);
#endif
#ifdef SIGTERM
		       signal(SIGTERM,panic);
#endif
#ifdef SIGSEGV
		       signal(SIGSEGV,panic);
#endif
#ifdef SIGBUS
		       signal(SIGBUS,panic);
#endif
		       break;
#endif
        case RESET   :
#if HAS_FLOATS && BREAK_FLOATS
		       bfTemp = NIL;
#endif
	case BREAK   :
	case EXIT    : normalTerminal();
#if MSWIN
		       if (what==EXIT)
			   DestroyWindow(hWndMain);
		       else
			   SetCursor(LoadCursor(NULL,IDC_ARROW));
#endif
		       break;
    }
}

/* Elegant fix for a needed eight bit character... */
#if VC32
# define OE "\224"
#endif
#if MACWP
# define OE "\232"
#endif
#ifndef OE
# define OE "ö"
#endif

/*-------------------------------------------------------------------------*/


syntax highlighted by Code2HTML, v. 0.9.1