First step to actually handling scanner overflows the right way

git-svn-id: https://yap.svn.sf.net/svnroot/yap/trunk@1185 b08c6af1-5177-4d33-ba66-4b1c6b8b522a
This commit is contained in:
vsc 2004-11-22 22:28:06 +00:00
parent 36878b9a2b
commit c083c838bb
2 changed files with 96 additions and 23 deletions

View File

@ -2967,7 +2967,7 @@ do_read(int inp_stream)
/* Scans the term using stack space */ /* Scans the term using stack space */
Yap_eot_before_eof = FALSE; Yap_eot_before_eof = FALSE;
tokstart = Yap_tokptr = Yap_toktide = Yap_tokenizer (inp_stream); tokstart = Yap_tokptr = Yap_toktide = Yap_tokenizer(inp_stream);
/* preserve value of H after scanning: otherwise we may lose strings /* preserve value of H after scanning: otherwise we may lose strings
and floats */ and floats */
old_H = H; old_H = H;
@ -2976,7 +2976,7 @@ do_read(int inp_stream)
/* next read should give out an end of file */ /* next read should give out an end of file */
Stream[inp_stream].status |= Push_Eof_Stream_f; Stream[inp_stream].status |= Push_Eof_Stream_f;
} else { } else {
if (tokstart != NIL && tokstart->Tok != Ord (eot_tok)) { if (tokstart != NULL && tokstart->Tok != Ord (eot_tok)) {
/* we got the end of file from an abort */ /* we got the end of file from an abort */
if (Yap_ErrorMessage == "Abort") { if (Yap_ErrorMessage == "Abort") {
Yap_clean_tokenizer(tokstart, Yap_VarTable, Yap_AnonVarTable); Yap_clean_tokenizer(tokstart, Yap_VarTable, Yap_AnonVarTable);

View File

@ -57,7 +57,7 @@
STATIC_PROTO(int my_getch, (int (*) (int))); STATIC_PROTO(int my_getch, (int (*) (int)));
STATIC_PROTO(Term float_send, (char *)); STATIC_PROTO(Term float_send, (char *));
STATIC_PROTO(Term get_num, (int *, int *, int, int (*) (int), int (*) (int))); STATIC_PROTO(Term get_num, (int *, int *, int, int (*) (int), int (*) (int),UInt));
/* token table with some help from Richard O'Keefe's PD scanner */ /* token table with some help from Richard O'Keefe's PD scanner */
static char chtype0[NUMBER_OF_CHARS+1] = static char chtype0[NUMBER_OF_CHARS+1] =
@ -135,7 +135,7 @@ AllocScannerMemory(unsigned int size)
ScannerStack = AuxSpScan+size; ScannerStack = AuxSpScan+size;
if (Yap_TrailTop <= ScannerStack) { if (Yap_TrailTop <= ScannerStack) {
if(!Yap_growtrail (sizeof(CELL) * 16 * 1024L)) { if(!Yap_growtrail (sizeof(CELL) * 16 * 1024L)) {
return(NULL); return NULL;
} }
} }
return AuxSpScan; return AuxSpScan;
@ -377,7 +377,7 @@ read_quoted_char(int *scan_nextp, int inp_stream, int (*QuotedNxtch)(int))
/* reads a number, either integer or float */ /* reads a number, either integer or float */
static Term static Term
get_num(int *chp, int *chbuffp, int inp_stream, int (*Nxtch) (int), int (*QuotedNxtch) (int)) get_num(int *chp, int *chbuffp, int inp_stream, int (*Nxtch) (int), int (*QuotedNxtch) (int), UInt max_size)
{ {
char *s = (char *)ScannerStack, *sp = s; char *s = (char *)ScannerStack, *sp = s;
int ch = *chp; int ch = *chp;
@ -392,6 +392,10 @@ get_num(int *chp, int *chbuffp, int inp_stream, int (*Nxtch) (int), int (*Quoted
*/ */
if (chtype[ch] == NU) { if (chtype[ch] == NU) {
*sp++ = ch; *sp++ = ch;
if (--max_size == 0) {
Yap_ErrorMessage = "Number Too Long";
return (TermNil);
}
base = 10 * base + ch - '0'; base = 10 * base + ch - '0';
ch = Nxtch(inp_stream); ch = Nxtch(inp_stream);
} }
@ -401,6 +405,10 @@ get_num(int *chp, int *chbuffp, int inp_stream, int (*Nxtch) (int), int (*Quoted
return (TermNil); return (TermNil);
} }
might_be_float = FALSE; might_be_float = FALSE;
if (--max_size == 0) {
Yap_ErrorMessage = "Number Too Long";
return (TermNil);
}
*sp++ = ch; *sp++ = ch;
ch = Nxtch(inp_stream); ch = Nxtch(inp_stream);
if (base == 0) { if (base == 0) {
@ -421,6 +429,10 @@ get_num(int *chp, int *chbuffp, int inp_stream, int (*Nxtch) (int), int (*Quoted
while (my_isxdigit(ch, upper_case, lower_case)) { while (my_isxdigit(ch, upper_case, lower_case)) {
Int oval = val; Int oval = val;
if (--max_size == 0) {
Yap_ErrorMessage = "Number Too Long";
return (TermNil);
}
*sp++ = ch; *sp++ = ch;
val = val * base + (chtype[ch] == NU ? ch - '0' : val = val * base + (chtype[ch] == NU ? ch - '0' :
(my_isupper(ch) ? ch - 'A' : ch - 'a') + 10); (my_isupper(ch) ? ch - 'A' : ch - 'a') + 10);
@ -431,10 +443,18 @@ get_num(int *chp, int *chbuffp, int inp_stream, int (*Nxtch) (int), int (*Quoted
} }
} else if ((ch == 'x' || ch == 'X') && base == 0) { } else if ((ch == 'x' || ch == 'X') && base == 0) {
might_be_float = FALSE; might_be_float = FALSE;
if (--max_size == 0) {
Yap_ErrorMessage = "Number Too Long";
return (TermNil);
}
*sp++ = ch; *sp++ = ch;
ch = Nxtch(inp_stream); ch = Nxtch(inp_stream);
while (my_isxdigit(ch, 'F', 'f')) { while (my_isxdigit(ch, 'F', 'f')) {
Int oval = val; Int oval = val;
if (--max_size == 0) {
Yap_ErrorMessage = "Number Too Long";
return (TermNil);
}
*sp++ = ch; *sp++ = ch;
val = val * 16 + (chtype[ch] == NU ? ch - '0' : val = val * 16 + (chtype[ch] == NU ? ch - '0' :
(my_isupper(ch) ? ch - 'A' : ch - 'a') + 10); (my_isupper(ch) ? ch - 'A' : ch - 'a') + 10);
@ -447,6 +467,10 @@ get_num(int *chp, int *chbuffp, int inp_stream, int (*Nxtch) (int), int (*Quoted
else if ((ch == 'o') && base == 0) { else if ((ch == 'o') && base == 0) {
might_be_float = FALSE; might_be_float = FALSE;
base = 8; base = 8;
if (--max_size == 0) {
Yap_ErrorMessage = "Number Too Long";
return (TermNil);
}
*sp++ = ch; *sp++ = ch;
*chp = Nxtch(inp_stream); *chp = Nxtch(inp_stream);
} }
@ -457,6 +481,10 @@ get_num(int *chp, int *chbuffp, int inp_stream, int (*Nxtch) (int), int (*Quoted
while (chtype[ch] == NU) { while (chtype[ch] == NU) {
Int oval = val; Int oval = val;
if (!(val == 0 && ch == '0')) { if (!(val == 0 && ch == '0')) {
if (--max_size == 0) {
Yap_ErrorMessage = "Number Too Long";
return (TermNil);
}
*sp++ = ch; *sp++ = ch;
} }
if (ch - '0' >= base) if (ch - '0' >= base)
@ -468,6 +496,10 @@ get_num(int *chp, int *chbuffp, int inp_stream, int (*Nxtch) (int), int (*Quoted
} }
if (might_be_float && (ch == '.' || ch == 'e' || ch == 'E')) { if (might_be_float && (ch == '.' || ch == 'e' || ch == 'E')) {
if (ch == '.') { if (ch == '.') {
if (--max_size == 0) {
Yap_ErrorMessage = "Number Too Long";
return (TermNil);
}
*sp++ = '.'; *sp++ = '.';
if (chtype[ch = Nxtch(inp_stream)] != NU) { if (chtype[ch = Nxtch(inp_stream)] != NU) {
*chbuffp = '.'; *chbuffp = '.';
@ -477,17 +509,31 @@ get_num(int *chp, int *chbuffp, int inp_stream, int (*Nxtch) (int), int (*Quoted
return(read_int_overflow(s,base,val)); return(read_int_overflow(s,base,val));
return (MkIntegerTerm(val)); return (MkIntegerTerm(val));
} }
do do {
if (--max_size == 0) {
Yap_ErrorMessage = "Number Too Long";
return (TermNil);
}
*sp++ = ch; *sp++ = ch;
}
while (chtype[ch = Nxtch(inp_stream)] == NU); while (chtype[ch = Nxtch(inp_stream)] == NU);
} }
if (ch == 'e' || ch == 'E') { if (ch == 'e' || ch == 'E') {
char *sp0 = sp; char *sp0 = sp;
char cbuff = ch; char cbuff = ch;
if (--max_size == 0) {
Yap_ErrorMessage = "Number Too Long";
return (TermNil);
}
*sp++ = ch; *sp++ = ch;
ch = Nxtch(inp_stream); ch = Nxtch(inp_stream);
if (ch == '-') { if (ch == '-') {
cbuff = '-'; cbuff = '-';
if (--max_size == 0) {
Yap_ErrorMessage = "Number Too Long";
return (TermNil);
}
*sp++ = '-'; *sp++ = '-';
ch = Nxtch(inp_stream); ch = Nxtch(inp_stream);
} else if (ch == '+') { } else if (ch == '+') {
@ -515,6 +561,10 @@ get_num(int *chp, int *chbuffp, int inp_stream, int (*Nxtch) (int), int (*Quoted
return(MkIntegerTerm(val)); return(MkIntegerTerm(val));
} }
do { do {
if (--max_size == 0) {
Yap_ErrorMessage = "Number Too Long";
return (TermNil);
}
*sp++ = ch; *sp++ = ch;
} while (chtype[ch = Nxtch(inp_stream)] == NU); } while (chtype[ch = Nxtch(inp_stream)] == NU);
} }
@ -546,9 +596,18 @@ Yap_scan_num(int (*Nxtch) (int))
Term out; Term out;
int sign = 1; int sign = 1;
int ch, cherr; int ch, cherr;
UInt tsize;
Yap_ErrorMessage = NULL; Yap_ErrorMessage = NULL;
ScannerStack = (char *)TR; ScannerStack = (char *)TR;
tsize = Yap_TrailTop-ScannerStack;
if (tsize < 4096) {
if(!Yap_growtrail (sizeof(CELL) * 16 * 1024L)) {
printf("vsc: Hello\n");
Yap_ErrorMessage = "Trail Overflow";
return TermNil;
}
}
ch = Nxtch(-1); ch = Nxtch(-1);
if (ch == '-') { if (ch == '-') {
sign = -1; sign = -1;
@ -560,7 +619,7 @@ Yap_scan_num(int (*Nxtch) (int))
return(TermNil); return(TermNil);
} }
cherr = 0; cherr = 0;
out = get_num(&ch, &cherr, -1, Nxtch, Nxtch); out = get_num(&ch, &cherr, -1, Nxtch, Nxtch, tsize);
if (sign == -1) { if (sign == -1) {
if (IsIntegerTerm(out)) if (IsIntegerTerm(out))
out = MkIntegerTerm(-IntegerOfTerm(out)); out = MkIntegerTerm(-IntegerOfTerm(out));
@ -587,8 +646,8 @@ Yap_tokenizer(int inp_stream)
Yap_AnonVarTable = NULL; Yap_AnonVarTable = NULL;
Yap_eot_before_eof = FALSE; Yap_eot_before_eof = FALSE;
ScannerStack = (char *)TR; ScannerStack = (char *)TR;
l = NIL; l = NULL;
p = NIL; /* Just to make lint happy */ p = NULL; /* Just to make lint happy */
ch = Nxtch(inp_stream); ch = Nxtch(inp_stream);
do { do {
int och, quote, isvar; int och, quote, isvar;
@ -600,11 +659,11 @@ Yap_tokenizer(int inp_stream)
t = (TokEntry *) AllocScannerMemory(sizeof(TokEntry)); t = (TokEntry *) AllocScannerMemory(sizeof(TokEntry));
t->TokNext = NULL; t->TokNext = NULL;
if (t == NULL) { if (t == NULL) {
Yap_ErrorMessage = "not enough stack space to read in term"; Yap_ErrorMessage = "Trail Overflow";
if (p != NIL) if (p)
p->TokInfo = eot_tok; p->TokInfo = eot_tok;
/* serious error now */ /* serious error now */
return(l); return l;
} }
if (l == NIL) if (l == NIL)
l = t; l = t;
@ -659,8 +718,18 @@ Yap_tokenizer(int inp_stream)
case NU: case NU:
{ {
int cherr, cha = ch; int cherr, cha = ch;
UInt tsize = Yap_TrailTop-ScannerStack;
cherr = 0; cherr = 0;
t->TokInfo = get_num(&cha,&cherr,inp_stream,Nxtch,QuotedNxtch); if (tsize < 4096) {
if(!Yap_growtrail (sizeof(CELL) * 16 * 1024L)) {
Yap_ErrorMessage = "Trail Overflow";
if (p)
t->TokInfo = eot_tok;
/* serious error now */
return l;
}
}
t->TokInfo = get_num(&cha,&cherr,inp_stream,Nxtch,QuotedNxtch,tsize);
ch = cha; ch = cha;
if (cherr) { if (cherr) {
TokEntry *e; TokEntry *e;
@ -668,10 +737,11 @@ Yap_tokenizer(int inp_stream)
t->TokPos = GetCurInpPos(inp_stream); t->TokPos = GetCurInpPos(inp_stream);
e = (TokEntry *) AllocScannerMemory(sizeof(TokEntry)); e = (TokEntry *) AllocScannerMemory(sizeof(TokEntry));
if (e == NULL) { if (e == NULL) {
Yap_ErrorMessage = "not enough stack space to read in term"; Yap_ErrorMessage = "Trail Overflow";
p->TokInfo = eot_tok; if (p)
p->TokInfo = eot_tok;
/* serious error now */ /* serious error now */
return(l); return l;
} else { } else {
e->TokNext = NULL; e->TokNext = NULL;
} }
@ -695,10 +765,11 @@ Yap_tokenizer(int inp_stream)
t->TokPos = GetCurInpPos(inp_stream); t->TokPos = GetCurInpPos(inp_stream);
e2 = (TokEntry *) AllocScannerMemory(sizeof(TokEntry)); e2 = (TokEntry *) AllocScannerMemory(sizeof(TokEntry));
if (e2 == NULL) { if (e2 == NULL) {
Yap_ErrorMessage = "not enough stack space to read in term"; Yap_ErrorMessage = "Trail Overflow";
p->TokInfo = eot_tok; if (p)
p->TokInfo = eot_tok;
/* serious error now */ /* serious error now */
return(l); return l;
} else { } else {
e2->TokNext = NULL; e2->TokNext = NULL;
} }
@ -724,10 +795,10 @@ Yap_tokenizer(int inp_stream)
t->TokPos = GetCurInpPos(inp_stream); t->TokPos = GetCurInpPos(inp_stream);
e2 = (TokEntry *) AllocScannerMemory(sizeof(TokEntry)); e2 = (TokEntry *) AllocScannerMemory(sizeof(TokEntry));
if (e2 == NULL) { if (e2 == NULL) {
Yap_ErrorMessage = "not enough stack space to read in term"; Yap_ErrorMessage = "Trail Overflow";
p->TokInfo = eot_tok; p->TokInfo = eot_tok;
/* serious error now */ /* serious error now */
return(l); return l;
} else { } else {
e2->TokNext = NULL; e2->TokNext = NULL;
} }
@ -789,6 +860,7 @@ Yap_tokenizer(int inp_stream)
/* serious error now */ /* serious error now */
Yap_ReleasePreAllocCodeSpace((CODEADDR)TokImage); Yap_ReleasePreAllocCodeSpace((CODEADDR)TokImage);
t->Tok = Ord(kind = eot_tok); t->Tok = Ord(kind = eot_tok);
return l;
} }
} }
*charp = '\0'; *charp = '\0';
@ -798,6 +870,7 @@ Yap_tokenizer(int inp_stream)
Yap_ErrorMessage = "not enough stack space to read in string or quoted atom"; Yap_ErrorMessage = "not enough stack space to read in string or quoted atom";
Yap_ReleasePreAllocCodeSpace((CODEADDR)TokImage); Yap_ReleasePreAllocCodeSpace((CODEADDR)TokImage);
t->Tok = Ord(kind = eot_tok); t->Tok = Ord(kind = eot_tok);
return l;
} }
strcpy(mp, TokImage); strcpy(mp, TokImage);
t->TokInfo = Unsigned(mp); t->TokInfo = Unsigned(mp);
@ -900,10 +973,10 @@ Yap_tokenizer(int inp_stream)
/* insert an error token to inform the system of what happened */ /* insert an error token to inform the system of what happened */
TokEntry *e = (TokEntry *) AllocScannerMemory(sizeof(TokEntry)); TokEntry *e = (TokEntry *) AllocScannerMemory(sizeof(TokEntry));
if (e == NULL) { if (e == NULL) {
Yap_ErrorMessage = "not enough stack space to read in term"; Yap_ErrorMessage = "Trail Overflow";
p->TokInfo = eot_tok; p->TokInfo = eot_tok;
/* serious error now */ /* serious error now */
return(l); return l;
} }
p->TokNext = e; p->TokNext = e;
e->Tok = Error_tok; e->Tok = Error_tok;