575 lines
14 KiB
C
575 lines
14 KiB
C
|
/*************************************************************************
|
||
|
* *
|
||
|
* BEAM -> Basic Extended Andorra Model *
|
||
|
* BEAM extends the YAP Prolog system to support the EAM *
|
||
|
* *
|
||
|
* Copyright: Ricardo Lopes and NCC - University of Porto, Portugal *
|
||
|
* *
|
||
|
**************************************************************************
|
||
|
* comments: abstract machine assembler *
|
||
|
*************************************************************************/
|
||
|
|
||
|
#ifdef BEAM
|
||
|
|
||
|
#include "Yap.h"
|
||
|
#include "compile.h"
|
||
|
#include "clause.h"
|
||
|
#include "eam.h"
|
||
|
#include "eamamasm.h"
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
|
||
|
Cell *inst_code;
|
||
|
int pass=0;
|
||
|
Cell *labels[1000];
|
||
|
|
||
|
Cell *Code_Start;
|
||
|
Cell Area_Code[200000];
|
||
|
Cell area_code=0;
|
||
|
|
||
|
extern Int inst_am(int n);
|
||
|
void emit_inst(long int i);
|
||
|
void emit_par(long int i);
|
||
|
void emit_upar(Cell i);
|
||
|
Cell *get_addr(void);
|
||
|
int Is_X_Var(Ventry *ve);
|
||
|
int Is_P_Var(Ventry *ve);
|
||
|
int X_Var(Ventry *ve);
|
||
|
int Y_Var(Ventry *ve);
|
||
|
void eam_pass(CInstr *ppc);
|
||
|
Cell *eam_assemble(CInstr *code);
|
||
|
int next_not_nop_inst(CInstr *ppc);
|
||
|
extern void *alloc_mem(Cell);
|
||
|
|
||
|
void emit_inst(long int i)
|
||
|
{
|
||
|
if (pass) *inst_code=inst_am(i);
|
||
|
inst_code++;
|
||
|
}
|
||
|
|
||
|
void emit_par(long int i)
|
||
|
{
|
||
|
if (pass) *inst_code=i;
|
||
|
inst_code++;
|
||
|
}
|
||
|
|
||
|
void emit_upar(Cell i)
|
||
|
{
|
||
|
if (pass) *inst_code=i;
|
||
|
inst_code++;
|
||
|
}
|
||
|
|
||
|
|
||
|
Cell *get_addr(void)
|
||
|
{
|
||
|
return(inst_code);
|
||
|
}
|
||
|
|
||
|
|
||
|
int Is_P_Var(Ventry *ve)
|
||
|
{
|
||
|
if (ve->FirstOfVE>0) return (1); /* var aparece pela primeira no corpo da clausula */
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
int Is_X_Var(Ventry *ve)
|
||
|
{
|
||
|
if (ve->KindOfVE == PermVar) return(0);
|
||
|
if (ve->KindOfVE == VoidVar) return(0);
|
||
|
|
||
|
return(1);
|
||
|
}
|
||
|
|
||
|
int X_Var(Ventry *ve)
|
||
|
{
|
||
|
int var;
|
||
|
|
||
|
if (ve->KindOfVE == PermVar || ve->KindOfVE == VoidVar ) {
|
||
|
printf("Erro no tipo de variavel X ->eamamas.c \n");
|
||
|
exit(1);
|
||
|
}
|
||
|
var = ((ve->NoOfVE) & MaskVarAdrs);
|
||
|
|
||
|
return (var);
|
||
|
}
|
||
|
|
||
|
extern int nperm;
|
||
|
|
||
|
int Y_Var(Ventry *ve)
|
||
|
{
|
||
|
int var;
|
||
|
if (ve->KindOfVE != PermVar) {
|
||
|
printf("Erro no tipo de variavel Y ->eamamas.c \n");
|
||
|
exit(1);
|
||
|
}
|
||
|
var = ((ve->NoOfVE) & MaskVarAdrs);
|
||
|
return (var);
|
||
|
}
|
||
|
|
||
|
|
||
|
int next_not_nop_inst(CInstr *ppc) {
|
||
|
while(ppc) {
|
||
|
if ((int) ppc->op!=nop_op) return ((int) ppc->op);
|
||
|
ppc = ppc->nextInst;
|
||
|
}
|
||
|
return exit_op;
|
||
|
}
|
||
|
|
||
|
void eam_pass(CInstr *ppc)
|
||
|
{
|
||
|
int alloc_found=0;
|
||
|
int body=0;
|
||
|
|
||
|
while (ppc) {
|
||
|
switch ((int) ppc->op) {
|
||
|
|
||
|
case get_var_op:
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_get_var_X_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
} else {
|
||
|
emit_inst(_get_var_Y_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
}
|
||
|
break;
|
||
|
case get_val_op:
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_get_val_X_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
} else {
|
||
|
emit_inst(_get_val_Y_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case get_num_op:
|
||
|
case get_atom_op:
|
||
|
emit_inst(_get_atom_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(ppc->new4);
|
||
|
break;
|
||
|
|
||
|
case get_list_op:
|
||
|
emit_inst(_get_list_op);
|
||
|
emit_par(ppc->new1);
|
||
|
break;
|
||
|
case get_struct_op:
|
||
|
emit_inst(_get_struct_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(ppc->new4);
|
||
|
emit_par(ArityOfFunctor((Functor ) ppc->new4));
|
||
|
break;
|
||
|
|
||
|
case unify_last_local_op:
|
||
|
case unify_local_op:
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_unify_local_X_op);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
} else {
|
||
|
emit_inst(_unify_local_Y_op);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case unify_last_val_op:
|
||
|
case unify_val_op:
|
||
|
if (((Ventry *)(ppc->new4))->KindOfVE!=VoidVar) {
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_unify_val_X_op);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
} else {
|
||
|
emit_inst(_unify_val_Y_op);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
|
||
|
}
|
||
|
} else { emit_inst(_unify_void_op); }
|
||
|
break;
|
||
|
|
||
|
case unify_last_var_op:
|
||
|
case unify_var_op:
|
||
|
if (((Ventry *)(ppc->new4))->KindOfVE!=VoidVar) {
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_unify_var_X_op);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
} else {
|
||
|
emit_inst(_unify_var_Y_op);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
}
|
||
|
} else { emit_inst(_unify_void_op); }
|
||
|
break;
|
||
|
|
||
|
case unify_last_atom_op:
|
||
|
case unify_last_num_op:
|
||
|
emit_inst(_unify_last_atom_op);
|
||
|
emit_par(ppc->new4);
|
||
|
break;
|
||
|
case unify_num_op:
|
||
|
case unify_atom_op:
|
||
|
emit_inst(_unify_atom_op);
|
||
|
emit_par(ppc->new4);
|
||
|
break;
|
||
|
case unify_list_op:
|
||
|
emit_inst(_unify_list_op);
|
||
|
break;
|
||
|
case unify_last_list_op:
|
||
|
emit_inst(_unify_last_list_op);
|
||
|
break;
|
||
|
case unify_struct_op:
|
||
|
emit_inst(_unify_struct_op);
|
||
|
emit_par(ppc->new4);
|
||
|
emit_par(ArityOfFunctor((Functor )ppc->new4));
|
||
|
break;
|
||
|
case unify_last_struct_op:
|
||
|
emit_inst(_unify_last_struct_op);
|
||
|
emit_par(ppc->new4);
|
||
|
emit_par(ArityOfFunctor((Functor )ppc->new4));
|
||
|
break;
|
||
|
|
||
|
case put_var_op:
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_put_var_X_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
} else {
|
||
|
if (Is_P_Var((Ventry *) ppc->new4)) emit_inst(_put_var_P_op);
|
||
|
else emit_inst(_put_var_Y_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
}
|
||
|
break;
|
||
|
case put_val_op:
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_put_val_X_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
break;
|
||
|
} else { /* else put_val_Y */
|
||
|
emit_inst(_put_val_Y_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case put_unsafe_op:
|
||
|
emit_inst(_put_unsafe_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
break;
|
||
|
|
||
|
case put_num_op:
|
||
|
case put_atom_op:
|
||
|
emit_inst(_put_atom_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(ppc->new4);
|
||
|
break;
|
||
|
case put_list_op:
|
||
|
emit_inst(_put_list_op);
|
||
|
emit_par(ppc->new1);
|
||
|
break;
|
||
|
case put_struct_op:
|
||
|
emit_inst(_put_struct_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(ppc->new4);
|
||
|
emit_par(ArityOfFunctor((Functor )ppc->new4));
|
||
|
break;
|
||
|
|
||
|
case write_var_op:
|
||
|
if (((Ventry *)(ppc->new4))->KindOfVE!=VoidVar) {
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_write_var_X_op);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
} else {
|
||
|
if (Is_P_Var((Ventry *) ppc->new4)) emit_inst(_write_var_P_op);
|
||
|
else emit_inst(_write_var_Y_op);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
}
|
||
|
} else emit_inst(_write_void);
|
||
|
break;
|
||
|
|
||
|
case write_local_op:
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_write_local_X_op);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
} else {
|
||
|
emit_inst(_write_local_Y_op);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case write_val_op:
|
||
|
if (((Ventry *)(ppc->new4))->KindOfVE!=VoidVar) {
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_write_val_X_op);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
} else {
|
||
|
emit_inst(_write_val_Y_op);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
}
|
||
|
} else emit_inst(_write_void);
|
||
|
break;
|
||
|
case write_num_op:
|
||
|
case write_atom_op:
|
||
|
emit_inst(_write_atom_op);
|
||
|
emit_par(ppc->new4);
|
||
|
break;
|
||
|
case write_list_op:
|
||
|
emit_inst(_write_list_op);
|
||
|
break;
|
||
|
case write_last_list_op:
|
||
|
emit_inst(_write_last_list_op);
|
||
|
break;
|
||
|
case write_struct_op:
|
||
|
emit_inst(_write_struct_op);
|
||
|
emit_par(ppc->new4);
|
||
|
emit_par(ArityOfFunctor((Functor )ppc->new4));
|
||
|
break;
|
||
|
case write_last_struct_op:
|
||
|
emit_inst(_write_last_struct_op);
|
||
|
emit_par(ppc->new4);
|
||
|
emit_par(ArityOfFunctor((Functor )ppc->new4));
|
||
|
break;
|
||
|
|
||
|
case fail_op:
|
||
|
emit_inst(_fail_op);
|
||
|
break;
|
||
|
case cutexit_op:
|
||
|
printf("cutexit \n");
|
||
|
exit(1);
|
||
|
break;
|
||
|
|
||
|
case cut_op:
|
||
|
emit_inst(_cut_op);
|
||
|
break;
|
||
|
case commit_op:
|
||
|
emit_inst(_commit_op);
|
||
|
break;
|
||
|
|
||
|
case procceed_op:
|
||
|
emit_inst(_proceed_op);
|
||
|
break;
|
||
|
case pop_op:
|
||
|
emit_inst(_pop_op);
|
||
|
emit_par(ppc->new4);
|
||
|
break;
|
||
|
case save_b_op:
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_save_b_X_op);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
} else {
|
||
|
emit_inst(_save_b_Y_op);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
}
|
||
|
break;
|
||
|
case save_pair_op:
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_save_pair_X_op);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
} else {
|
||
|
emit_inst(_save_pair_Y_op);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
}
|
||
|
break;
|
||
|
case save_appl_op:
|
||
|
if (Is_X_Var((Ventry *) ppc->new4)) {
|
||
|
emit_inst(_save_appl_X_op);
|
||
|
emit_par(X_Var((Ventry *) ppc->new4));
|
||
|
} else {
|
||
|
emit_inst(_save_appl_Y_op);
|
||
|
emit_par(Y_Var((Ventry *) ppc->new4));
|
||
|
}
|
||
|
break;
|
||
|
case std_base_op:
|
||
|
emit_inst(_std_base+ppc->new4);
|
||
|
break;
|
||
|
|
||
|
case safe_call_op:
|
||
|
if (ppc->new1==1) {
|
||
|
emit_inst(_safe_call_unary_op);
|
||
|
} else if (ppc->new1==2) {
|
||
|
emit_inst(_safe_call_binary_op);
|
||
|
} else {
|
||
|
emit_inst(_safe_call_op);
|
||
|
}
|
||
|
emit_par(ppc->new4);
|
||
|
break;
|
||
|
|
||
|
case direct_safe_call_op:
|
||
|
if (ppc->new1==1) {
|
||
|
emit_inst(_direct_safe_call_unary_op);
|
||
|
} else if (ppc->new1==2) {
|
||
|
emit_inst(_direct_safe_call_binary_op);
|
||
|
} else {
|
||
|
emit_inst(_direct_safe_call_op);
|
||
|
}
|
||
|
emit_par(ppc->new4);
|
||
|
break;
|
||
|
|
||
|
case call_op:
|
||
|
emit_inst(_call_op);
|
||
|
emit_par(ppc->new4);
|
||
|
break;
|
||
|
|
||
|
case skip_while_var_op:
|
||
|
emit_inst(_skip_while_var);
|
||
|
break;
|
||
|
case wait_while_var_op:
|
||
|
emit_inst(_wait_while_var);
|
||
|
break;
|
||
|
case force_wait_op:
|
||
|
emit_inst(_force_wait);
|
||
|
break;
|
||
|
case write_op:
|
||
|
if (ppc->new1=='\n') {
|
||
|
static Atom a=NULL;
|
||
|
if (a==NULL) a=Yap_LookupAtom("\n");
|
||
|
emit_inst(_put_atom_op);
|
||
|
emit_par(1);
|
||
|
emit_par((Cell) MkAtomTerm(a));
|
||
|
}
|
||
|
emit_inst(_write_call);
|
||
|
break;
|
||
|
case is_op:
|
||
|
emit_inst(_is_call);
|
||
|
break;
|
||
|
case equal_op:
|
||
|
emit_inst(_equal_call);
|
||
|
break;
|
||
|
|
||
|
case either_op:
|
||
|
emit_inst(_either_op);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_upar((Cell) Code_Start+ (Cell) labels[ppc->new4]);
|
||
|
break;
|
||
|
case orelse_op:
|
||
|
emit_inst(_orelse_op);
|
||
|
emit_upar((Cell) Code_Start+ (Cell) labels[ppc->new4]);
|
||
|
break;
|
||
|
case orlast_op:
|
||
|
emit_inst(_orlast_op);
|
||
|
break;
|
||
|
|
||
|
case create_first_box_op:
|
||
|
case create_box_op:
|
||
|
case create_last_box_op:
|
||
|
emit_upar((Cell) Code_Start+ (Cell) labels[ppc->new4]);
|
||
|
alloc_found=1;
|
||
|
break;
|
||
|
|
||
|
case remove_box_op:
|
||
|
case remove_last_box_op:
|
||
|
break;
|
||
|
|
||
|
case jump_op:
|
||
|
emit_inst(_jump_op);
|
||
|
emit_upar((Cell) Code_Start+ (Cell) labels[ppc->new4]);
|
||
|
break;
|
||
|
case label_op:
|
||
|
if (pass==0) labels[ppc->new4] = get_addr();
|
||
|
break;
|
||
|
|
||
|
|
||
|
case only_1_clause_op:
|
||
|
emit_inst(_only_1_clause_op);
|
||
|
emit_par(ppc->new4);
|
||
|
emit_par(((struct Clauses *)ppc->new4)->predi->arity);
|
||
|
emit_par(((struct Clauses *)ppc->new4)->nr_vars);
|
||
|
emit_par(0); /* Nr da alternativa */
|
||
|
break;
|
||
|
case try_me_op:
|
||
|
emit_inst(_try_me_op);
|
||
|
emit_par(ppc->new4);
|
||
|
emit_par(((struct Clauses *)ppc->new4)->predi->arity);
|
||
|
emit_par(((struct Clauses *)ppc->new4)->nr_vars);
|
||
|
emit_par(0); /* Nr da alternativa */
|
||
|
break;
|
||
|
case retry_me_op:
|
||
|
emit_inst(_retry_me_op);
|
||
|
emit_par(ppc->new4);
|
||
|
emit_par(((struct Clauses *)ppc->new4)->predi->arity);
|
||
|
emit_par(((struct Clauses *)ppc->new4)->nr_vars);
|
||
|
emit_par(ppc->new1);
|
||
|
break;
|
||
|
case trust_me_op:
|
||
|
emit_inst(_trust_me_op);
|
||
|
emit_par(ppc->new4);
|
||
|
emit_par(((struct Clauses *)ppc->new4)->predi->arity);
|
||
|
emit_par(((struct Clauses *)ppc->new4)->nr_vars);
|
||
|
emit_par(ppc->new1);
|
||
|
break;
|
||
|
|
||
|
case body_op:
|
||
|
if (next_not_nop_inst(ppc->nextInst)==procceed_op) {
|
||
|
//emit_inst(_proceed_op);
|
||
|
break;
|
||
|
} else if (next_not_nop_inst(ppc->nextInst)==fail_op) {
|
||
|
//emit_inst(_fail_op);
|
||
|
break;
|
||
|
}
|
||
|
if (ppc->new4!=0) {
|
||
|
emit_inst(_prepare_calls);
|
||
|
emit_par(ppc->new4); /* nr_calls */
|
||
|
}
|
||
|
body=1;
|
||
|
break;
|
||
|
|
||
|
case prepare_tries:
|
||
|
emit_inst(_prepare_tries);
|
||
|
emit_par(ppc->new1);
|
||
|
emit_par(ppc->new4);
|
||
|
break;
|
||
|
|
||
|
case exit_op:
|
||
|
emit_inst(_exit_eam);
|
||
|
break;
|
||
|
|
||
|
case mark_initialised_pvars_op:
|
||
|
break;
|
||
|
case fetch_args_for_bccall:
|
||
|
case bccall_op:
|
||
|
printf("[ Fatal Error: fetch and bccall instructions not supported ]\n");
|
||
|
exit(1);
|
||
|
break;
|
||
|
|
||
|
case endgoal_op:
|
||
|
case nop_op:
|
||
|
case name_op:
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
if (pass) {
|
||
|
printf("[ Sorry, there is at least one unsupported instruction in your code... %3d] %d\n",ppc->op,exit_op);
|
||
|
printf("[ please note that beam still does not support a lot of builtins ]\n");
|
||
|
}
|
||
|
emit_inst(_fail_op);
|
||
|
|
||
|
}
|
||
|
ppc = ppc->nextInst;
|
||
|
}
|
||
|
emit_inst(_exit_eam);
|
||
|
emit_par(-1);
|
||
|
}
|
||
|
|
||
|
|
||
|
Cell *eam_assemble(CInstr *code)
|
||
|
{
|
||
|
|
||
|
Code_Start=0;
|
||
|
pass=0;
|
||
|
inst_code=0;
|
||
|
eam_pass(code);
|
||
|
|
||
|
pass=1;
|
||
|
Code_Start=alloc_mem((Cell) inst_code);
|
||
|
inst_code=Code_Start;
|
||
|
eam_pass(code);
|
||
|
|
||
|
return(Code_Start);
|
||
|
}
|
||
|
|
||
|
|
||
|
#endif /* BEAM */
|