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
  
124
  
125
  
126
  
127
  
128
  
129
  
130
  
131
  
132
  
133
  
134
  
135
  
136
  
137
  
138
  
139
  
140
  
141
  
142
  
143
  
144
  
145
  
146
  
147
  
/*\ 
||| This file a part of Pike, and is copyright by Fredrik Hubinette 
||| Pike is distributed as GPL (General Public License) 
||| See the files COPYING and DISCLAIMER for more information. 
\*/ 
#ifndef INTERPRET_H 
#define INTERPRET_H 
 
#include "global.h" 
#include "program.h" 
 
#ifndef STRUCT_FRAME_DECLARED 
#define STRUCT_FRAME_DECLARED 
#endif 
struct frame 
{ 
  unsigned char *pc; 
  struct frame *parent_frame; 
  struct svalue *locals; 
  INT32 args; 
  struct object *current_object; 
  struct inherit context; 
  char *current_storage; 
  INT32 fun; 
  INT16 num_locals; 
  INT16 num_args; 
}; 
 
#ifdef DEBUG 
#define debug_check_stack() do{if(sp<evaluator_stack)fatal("Stack error.\n");}while(0) 
#else 
#define debug_check_stack()  
#endif 
 
#define pop_stack() do{ free_svalue(--sp); debug_check_stack(); }while(0) 
 
#define push_program(P) do{ struct program *_=(P); sp->u.program=_; sp++->type=T_PROGRAM; }while(0) 
#define push_int(I) do{ INT32 _=(I); sp->u.integer=_;sp->type=T_INT;sp++->subtype=NUMBER_NUMBER; }while(0) 
#define push_mapping(M) do{ struct mapping *_=(M); sp->u.mapping=_; sp++->type=T_MAPPING; }while(0) 
#define push_array(A) do{ struct array *_=(A); sp->u.array=_ ;sp++->type=T_ARRAY; }while(0) 
#define push_multiset(L) do{ struct multiset *_=(L); sp->u.multiset=_; sp++->type=T_MULTISET; }while(0) 
#define push_string(S) do{ struct pike_string *_=(S); sp->subtype=0; sp->u.string=_; sp++->type=T_STRING; }while(0) 
#define push_object(O) do{ struct object  *_=(O); sp->u.object=_; sp++->type=T_OBJECT; }while(0) 
#define push_float(F) do{ float _=(F); sp->u.float_number=_; sp++->type=T_FLOAT; }while(0) 
#define push_text(T) push_string(make_shared_string((T))) 
#define push_constant_text(T) do{ sp->subtype=0; MAKE_CONSTANT_SHARED_STRING(sp->u.string,T); sp++->type=T_STRING; }while(0) 
 
#define ref_push_program(P) do{ struct program *_=(P); _->refs++; sp->u.program=_; sp++->type=T_PROGRAM; }while(0) 
#define ref_push_mapping(M) do{ struct mapping *_=(M); _->refs++; sp->u.mapping=_; sp++->type=T_MAPPING; }while(0) 
#define ref_push_array(A) do{ struct array *_=(A); _->refs++; sp->u.array=_ ;sp++->type=T_ARRAY; }while(0) 
#define ref_push_multiset(L) do{ struct multiset *_=(L); _->refs++; sp->u.multiset=_; sp++->type=T_MULTISET; }while(0) 
#define ref_push_string(S) do{ struct pike_string *_=(S); _->refs++; sp->subtype=0; sp->u.string=_; sp++->type=T_STRING; }while(0) 
#define ref_push_object(O) do{ struct object  *_=(O); _->refs++; sp->u.object=_; sp++->type=T_OBJECT; }while(0) 
 
#define push_svalue(S) do { struct svalue *_=(S); assign_svalue_no_free(sp,_); sp++; }while(0) 
 
enum apply_type 
{ 
  APPLY_STACK, /* The function is the first argument */ 
  APPLY_SVALUE, /* arg1 points to an svalue containing the function */ 
   APPLY_LOW    /* arg1 is the object pointer,(int)arg2 the function */ 
}; 
 
#define apply_low(O,FUN,ARGS) \ 
  mega_apply(APPLY_LOW, (ARGS), (void*)(O),(void*)(FUN)) 
 
#define strict_apply_svalue(SVAL,ARGS) \ 
  mega_apply(APPLY_SVALUE, (ARGS), (void*)(SVAL),0) 
 
#define APPLY_MASTER(FUN,ARGS) \ 
do{ \ 
  static int fun_,master_cnt=0; \ 
  struct object *master_ob=master(); \ 
  if(master_cnt != master_ob->prog->id) \ 
  { \ 
    fun_=find_identifier(FUN,master_ob->prog); \ 
    master_cnt = master_ob->prog->id; \ 
  } \ 
  apply_low(master_ob, fun_, ARGS); \ 
}while(0) 
 
#define SAFE_APPLY_MASTER(FUN,ARGS) \ 
do{ \ 
  static int fun_,master_cnt=0; \ 
  struct object *master_ob=master(); \ 
  if(master_cnt != master_ob->prog->id) \ 
  { \ 
    fun_=find_identifier(FUN,master_ob->prog); \ 
    master_cnt = master_ob->prog->id; \ 
  } \ 
  safe_apply_low(master_ob, fun_, ARGS); \ 
}while(0) 
 
#define check_threads_etc() \ 
  call_callback(& evaluator_callbacks, (void *)0) 
 
#ifdef DEBUG 
#define fast_check_threads_etc(X) do { \ 
  static int div_; if(d_flag || !(div_++& ((1<<(X))-1))) check_threads_etc(); } while(0) 
 
#else 
#define fast_check_threads_etc(X) do { \ 
  static int div_; if(!(div_++& ((1<<(X))-1))) check_threads_etc(); } while(0) 
#endif 
 
/* Prototypes begin here */ 
void push_sp_mark(void); 
int pop_sp_mark(void); 
void init_interpreter(void); 
void check_stack(INT32 size); 
void check_mark_stack(INT32 size); 
void lvalue_to_svalue_no_free(struct svalue *to,struct svalue *lval); 
void assign_lvalue(struct svalue *lval,struct svalue *from); 
union anything *get_pointer_if_this_type(struct svalue *lval, TYPE_T t); 
void print_return_value(void); 
void pop_n_elems(INT32 x); 
void reset_evaluator(void); 
struct backlog; 
void dump_backlog(void); 
void mega_apply(enum apply_type type, INT32 args, void *arg1, void *arg2); 
int apply_low_safe_and_stupid(struct object *o, INT32 offset); 
void safe_apply_low(struct object *o,int fun,int args); 
void safe_apply(struct object *o, char *fun ,INT32 args); 
void apply_lfun(struct object *o, int fun, int args); 
void apply_shared(struct object *o, 
                  struct pike_string *fun, 
                  int args); 
void apply(struct object *o, char *fun, int args); 
void apply_svalue(struct svalue *s, INT32 args); 
void slow_check_stack(void); 
void cleanup_interpret(void); 
/* Prototypes end here */ 
 
extern struct svalue *sp; 
extern struct svalue **mark_sp; 
extern struct svalue *evaluator_stack; 
extern struct svalue **mark_stack; 
extern struct frame *fp; /* frame pointer */ 
extern int stack_size; 
extern int evaluator_stack_malloced, mark_stack_malloced; 
struct callback; 
extern struct callback_list evaluator_callbacks; 
extern void call_callback(struct callback_list *, void *); 
#endif