2000-05-28 13:40:48 +00:00
|
|
|
/*****************************************************************************/
|
|
|
|
/* */
|
2000-06-03 11:15:11 +00:00
|
|
|
/* istack.c */
|
2000-05-28 13:40:48 +00:00
|
|
|
/* */
|
2000-06-03 11:15:11 +00:00
|
|
|
/* Input stack for the scanner */
|
2000-05-28 13:40:48 +00:00
|
|
|
/* */
|
|
|
|
/* */
|
|
|
|
/* */
|
2000-06-03 11:15:11 +00:00
|
|
|
/* (C) 2000 Ullrich von Bassewitz */
|
2000-05-28 13:40:48 +00:00
|
|
|
/* Wacholderweg 14 */
|
|
|
|
/* D-70597 Stuttgart */
|
|
|
|
/* EMail: uz@musoftware.de */
|
|
|
|
/* */
|
|
|
|
/* */
|
|
|
|
/* This software is provided 'as-is', without any expressed or implied */
|
|
|
|
/* warranty. In no event will the authors be held liable for any damages */
|
|
|
|
/* arising from the use of this software. */
|
|
|
|
/* */
|
|
|
|
/* Permission is granted to anyone to use this software for any purpose, */
|
|
|
|
/* including commercial applications, and to alter it and redistribute it */
|
|
|
|
/* freely, subject to the following restrictions: */
|
|
|
|
/* */
|
|
|
|
/* 1. The origin of this software must not be misrepresented; you must not */
|
|
|
|
/* claim that you wrote the original software. If you use this software */
|
|
|
|
/* in a product, an acknowledgment in the product documentation would be */
|
|
|
|
/* appreciated but is not required. */
|
|
|
|
/* 2. Altered source versions must be plainly marked as such, and must not */
|
|
|
|
/* be misrepresented as being the original software. */
|
|
|
|
/* 3. This notice may not be removed or altered from any source */
|
|
|
|
/* distribution. */
|
|
|
|
/* */
|
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
2000-08-01 15:17:43 +00:00
|
|
|
/* common */
|
|
|
|
#include "check.h"
|
|
|
|
#include "xmalloc.h"
|
|
|
|
|
|
|
|
/* ca65 */
|
2000-06-03 11:15:11 +00:00
|
|
|
#include "error.h"
|
|
|
|
#include "istack.h"
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
2000-06-03 11:15:11 +00:00
|
|
|
/* Data */
|
2000-05-28 13:40:48 +00:00
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
2000-06-03 11:15:11 +00:00
|
|
|
/* Size of the stack (== maximum nested macro or repeat count) */
|
|
|
|
#define ISTACK_MAX 256
|
2000-05-28 13:40:48 +00:00
|
|
|
|
2000-06-03 11:15:11 +00:00
|
|
|
/* Structure holding a stack element */
|
|
|
|
typedef struct IElement IElement;
|
|
|
|
struct IElement {
|
|
|
|
IElement* Next; /* Next stack element */
|
|
|
|
int (*Func)(void*); /* Function called for input */
|
|
|
|
void* Data; /* User data given as argument */
|
|
|
|
const char* Desc; /* Description */
|
2000-05-28 13:40:48 +00:00
|
|
|
};
|
|
|
|
|
2000-06-03 11:15:11 +00:00
|
|
|
/* The stack */
|
|
|
|
static IElement* IStack = 0; /* Input stack pointer */
|
|
|
|
static unsigned ICount = 0; /* Number of items on the stack */
|
|
|
|
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
2000-06-03 11:15:11 +00:00
|
|
|
/* Code */
|
2000-05-28 13:40:48 +00:00
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
2000-06-03 11:15:11 +00:00
|
|
|
void PushInput (int (*Func) (void*), void* Data, const char* Desc)
|
|
|
|
/* Push an input function onto the input stack */
|
|
|
|
{
|
|
|
|
IElement* E;
|
|
|
|
|
|
|
|
/* Check for a stack overflow */
|
|
|
|
if (ICount > ISTACK_MAX) {
|
|
|
|
Fatal (FAT_NESTING);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Create a new stack element */
|
2000-06-14 09:32:22 +00:00
|
|
|
E = xmalloc (sizeof (*E));
|
2000-06-03 11:15:11 +00:00
|
|
|
|
|
|
|
/* Initialize it */
|
|
|
|
E->Func = Func;
|
|
|
|
E->Data = Data;
|
|
|
|
E->Desc = Desc;
|
|
|
|
|
|
|
|
/* Push it */
|
|
|
|
E->Next = IStack;
|
|
|
|
IStack = E;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void PopInput (void)
|
|
|
|
/* Pop the current input function from the input stack */
|
|
|
|
{
|
|
|
|
IElement* E;
|
|
|
|
|
|
|
|
/* We cannot pop from an empty stack */
|
|
|
|
PRECONDITION (IStack != 0);
|
|
|
|
|
|
|
|
/* Remember the last element */
|
|
|
|
E = IStack;
|
|
|
|
|
|
|
|
/* Pop it */
|
|
|
|
IStack = IStack->Next;
|
2000-05-28 13:40:48 +00:00
|
|
|
|
2000-06-03 11:15:11 +00:00
|
|
|
/* And delete it */
|
2000-06-14 09:32:22 +00:00
|
|
|
xfree (E);
|
2000-06-03 11:15:11 +00:00
|
|
|
}
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2000-06-03 11:15:11 +00:00
|
|
|
int InputFromStack (void)
|
|
|
|
/* Try to get input from the input stack. Return true if we had such input,
|
|
|
|
* return false otherwise.
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
/* Repeatedly call the TOS routine until we have a token or if run out of
|
|
|
|
* routines.
|
|
|
|
*/
|
|
|
|
while (IStack) {
|
|
|
|
if (IStack->Func (IStack->Data) != 0) {
|
|
|
|
/* We have a token */
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
2000-05-28 13:40:48 +00:00
|
|
|
|
2000-06-03 11:15:11 +00:00
|
|
|
/* Nothing is on the stack */
|
|
|
|
return 0;
|
|
|
|
}
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2000-07-09 13:19:25 +00:00
|
|
|
int HavePushedInput (void)
|
|
|
|
/* Return true if we have stacked input available, return false if not */
|
|
|
|
{
|
|
|
|
return (IStack != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2000-06-03 11:15:11 +00:00
|
|
|
void CheckInputStack (void)
|
|
|
|
/* Called from the scanner before closing an input file. Will check for any
|
|
|
|
* stuff on the input stack.
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
if (IStack) {
|
|
|
|
Error (ERR_OPEN_STMT, IStack->Desc);
|
|
|
|
}
|
|
|
|
}
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
|