1
  
2
  
3
  
4
  
5
  
6
  
7
  
8
  
9
  
10
  
11
  
12
  
13
  
14
  
15
  
16
  
17
  
18
  
19
  
20
  
21
  
22
  
23
  
24
  
25
  
26
  
27
  
28
  
29
  
30
  
31
  
32
  
33
  
34
  
35
  
36
  
37
  
38
  
39
  
40
  
41
  
42
  
43
  
44
  
45
  
46
  
47
  
48
  
49
  
50
  
51
  
52
  
53
  
54
  
55
  
56
  
57
  
58
  
59
  
60
  
61
  
62
  
63
  
64
  
65
  
66
  
67
  
68
  
69
  
70
  
71
  
72
  
73
  
74
  
75
  
76
  
77
  
78
  
79
  
80
  
81
  
82
  
83
  
84
  
85
  
86
  
87
  
88
  
89
  
90
  
91
  
92
  
93
  
94
  
95
  
96
  
97
  
98
  
99
  
100
  
101
  
102
  
103
  
104
  
105
  
106
  
107
  
108
  
109
  
110
  
111
  
112
  
113
  
114
  
115
  
116
  
117
  
118
  
119
  
120
  
121
  
122
  
123
  
#include <stdio.h> 
#include "global.h" 
#include "macros.h" 
#include "error.h" 
#include "interpret.h" 
#include "stralloc.h" 
#include "builtin_efuns.h" 
#include "array.h" 
#include "object.h" 
 
char *automatic_fatal; 
JMP_BUF *recoveries; 
 
jmp_buf *init_recovery(JMP_BUF *r) 
{ 
  r->fp=fp; 
  r->sp=sp; 
  r->mark_sp=mark_sp; 
  r->previous=recoveries; 
  recoveries=r; 
  return & r->recovery; 
} 
 
void throw() 
{ 
  if(!recoveries) 
    fatal("No error recovery context.\n"); 
 
#ifdef DEBUG 
  if(sp < recoveries->sp) 
    fatal("Error in error.\n"); 
#endif 
 
  while(fp != recoveries->fp) 
  { 
#ifdef DEBUG 
    if(!fp) 
      fatal("Popped out of stack frames.\n"); 
#endif 
    free_object(fp->current_object); 
    free_program(fp->context.prog); 
     
    fp = fp->parent_frame; 
  } 
 
  pop_n_elems(sp - recoveries->sp); 
  mark_sp = recoveries->mark_sp; 
 
  longjmp(recoveries->recovery,1); 
} 
 
struct svalue throw_value = { T_INT }; 
 
void va_error(char *fmt, va_list args) 
{ 
  char buf[2000]; 
  static int in_error; 
  if(in_error) 
  { 
    in_error=0; 
    fatal("Recursive error() calls.\n"); 
  } 
 
  in_error=1; 
 
  VSPRINTF(buf, fmt, args); 
 
  if(!recoveries || automatic_fatal) 
  { 
    if(!automatic_fatal) 
      automatic_fatal="No error recovery context: "; 
 
    fprintf(stderr,"%s %s",automatic_fatal,buf); 
    abort(); 
  } 
 
  if(strlen(buf) >= sizeof(buf)) 
    fatal("Buffer overflow in error()\n"); 
   
  push_string(make_shared_string(buf)); 
  f_backtrace(0); 
  f_aggregate(2); 
  free_svalue(& throw_value); 
  sp--; 
  throw_value = *sp; 
 
  in_error=0; 
  throw();  /* Hope someone is catching, or we will be out of balls. */ 
} 
 
void error(char *fmt,...) 
{ 
  va_list args; 
  va_start(args,fmt); 
  va_error(fmt,args); 
  va_end(args); 
} 
 
 
void fatal(char *fmt, ...) 
{ 
  va_list args; 
  static int in_fatal = 0; 
 
  va_start(args,fmt); 
  /* Prevent double fatal. */ 
  if (in_fatal) 
  { 
    (void)VFPRINTF(stderr, fmt, args); 
    abort(); 
  } 
  in_fatal = 1; 
#ifdef DEBUG 
  dump_backlog(); 
#endif 
 
  (void)VFPRINTF(stderr, fmt, args); 
  /* Insert dump routine call here */ 
  fflush(stderr); 
  abort(); 
}