gcc/libbanshee/engine/nonspec.c
2004-05-13 02:41:07 -04:00

853 lines
18 KiB
C

/*
* Copyright (c) 2000-2001
* The Regents of the University of California. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of the University nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*
*/
#include "banshee.h"
#include "flowrow-sort.h"
#include "flowrow-var.h"
#include "setif-sort.h"
#include "setif-var.h"
#include "setst-sort.h"
#include "setst-var.h"
#include "term-sort.h"
#include "term-var.h"
struct constructor
{
sort_kind sort;
int type;
int arity;
char *name;
sig_elt *sig;
};
typedef struct constructor *constructor;
typedef enum
{
vnc_pos,
vnc_neg,
vnc_non
} vnc_kind;
struct sig_elt
{
vnc_kind variance;
sort_kind sort;
};
typedef struct sig_elt sig_elt;
typedef struct proj_pat
{
sort_kind sort;
int type;
stamp st;
int i;
gen_e exp;
vnc_kind variance;
constructor c;
} *proj_pat;
typedef struct cons_expr
{
sort_kind sort;
int type;
stamp st;
int arity;
char *name;
sig_elt *sig;
gen_e *exps;
} * cons_expr;
static int new_type()
{
static int type = 10;
int ret = type;
if (type > 2000)
{
fprintf(stderr, "Exceeded maximum number of constructors\n");
assert(0);
}
type += 2;
return ret;
}
static bool fixed_sort(sort_kind s)
{
return !(s == sort_term || s == sort_set);
}
/*
Convention : constructor types are even, pats are odd.
The smallest specialized type is 10.
*/
static bool setif_is_pat(gen_e e)
{
int type = ((setif_term)e)->type;
return ( (type & 1) && (type > 10) );
}
static bool setst_is_pat(gen_e e)
{
int type = ((setst_term)e)->type;
return ( (type & 1) && (type > 10) );
}
static gen_e get_proj_var(sort_kind s, bool large)
{
switch (s)
{
case setif_sort:
{
if (large)
return (gen_e)sv_fresh_large(get_sort_region(setif_sort),NULL);
else return (gen_e)sv_fresh(get_sort_region(setif_sort),NULL);
}
break;
case setst_sort:
{
if (large)
return (gen_e)st_fresh_large(get_sort_region(setst_sort),NULL);
else return (gen_e)st_fresh(get_sort_region(setst_sort),NULL);
}
break;
case flowrow_sort:
{
if (large)
return (gen_e)fv_fresh_large(get_sort_region(flowrow_sort),NULL);
else return (gen_e)fv_fresh(get_sort_region(flowrow_sort),NULL);
}
break;
case term_sort:
{
if (large)
return (gen_e)tv_fresh_large(get_sort_region(term_sort),NULL);
else return (gen_e)tv_fresh(get_sort_region(term_sort),NULL)
}
break;
default:
{
fail("Unmatched sort in get_proj_var\n");
return NULL;
}
break;
}
return NULL;
}
static gen_e get_sort_zero(sort_kind s)
{
switch (s)
{
case setif_sort:
return setif_zero();
case setst_sort:
return setst_zero();
case flowrow_sort:
return flowrow_zero();
case term_sort:
return term_zero();
default:
fail("Unmatched sort in get_sort_zero\n");
return NULL;
}
return NULL;
}
static gen_e get_sort_one(sort_kind s)
{
switch (s)
{
case setif_sort:
return setif_one();
case setst_sort:
return setst_one();
case flowrow_sort:
return flowrow_one();
case term_sort:
return term_one();
default:
fail("Unmatched sort in get_sort_zero\n");
return NULL;
}
return NULL;
}
static region get_sort_region(sort s)
{
switch (s)
{
case setif_sort:
return setif_region;
case setst_sort:
return setst_region;
case flowrow_sort:
return flowrow_region;
case term_sort:
return term_region:
default:
fail("Unmatched sort in get_sort_region\n");
return NULL;
}
return NULL;
}
static term_hash get_sort_hash(sort s)
{
switch (s)
{
case setif_sort:
return setif_hash;
case setst_sort:
return setst_hash;
case flowrow_sort:
return flowrow_hash;
case term_sort:
return term_hash:
default:
fail("Unmatched sort in get_sort_hash\n");
return NULL;
}
return NULL;
}
constructor make_constructor(const char *name,sort_kind sort, sig_elt s[],
int arity)
{
constructor c = ralloc(expr_region,struct constructor);
sig_elt *sig = rarrayalloc(expr_region,arity,sig_elt);
c->type = new_type();
if (arity)
{
memcpy(sig,s,sizeof(sig_elt)*arity);
}
if ( fixed_sort(sort) )
failure("Specified sort does not allow constructor types\n");
c->sort = sort;
c->arity = arity;
c->name = rstrdup(expr_region,name);
c->sig = sig;
return c;
}
gen_e constructor_expr(constructor c, gen_e exps[], int arity)
{
cons_expr result;
int i;
region sort_region = get_sort_region(c->sort);
term_hash sort_hash = get_sort_hash(c->sort);
stamp *st = rarrayalloc(sort_region,arity + 1,stamp);
st[0] = c->type;
// Dynamic arity check
if(arity != c->arity)
failure("Signature mismatch\n");
// Dynamic sort checks
for (i = 0; i < arity; i++)
{
if ( c->sig[i].sort != exps[i]->sort)
failure(stderr,"Signature mismatch\n");
st[i+1] = exps[i]->st;
}
// Hash-consing of terms
if (!(result = term_hash_find(sort_hash,st,arity+1)) || arity == 0 )
{
gen_e *e = rarrayalloc(sort_region,arity,gen_e);
if (arity)
memcpy(e,exps,sizeof(gen_e)*arity);
else
e = NULL;
result = ralloc(sort_region,struct cons_expr);
result->type = st[0];
result->st = stamp_fresh();
result->sort = c->sort;
result->arity = c->arity;
result->name = c->name;
result->sig = c->sig;
result->exps = e;
term_hash_insert(expr_hash,result,st,arity+1);
}
return (gen_e)result;
}
static gen_e proj_pat(constructor c, int i, gen_e e)
{
proj_pat pat;
region sort_region = get_sort_region(e->sort);
term_hash sort_hash = get_sort_hash(e->sort);
stamp s[3];
s[0] = c->type + 1;
s[1] = e->st;
s[2] = i;
if (! (pat = term_hash_find(sort_hash,s,3)) )
{
pat = ralloc(sort_region,struct proj_pat);
pat->type = s[0];
pat->st = stamp_fresh();
pat->sort = c->sort;
pat->exp = e;
pat->variance = c->sig[i].variance;
pat->c = c;
pat->i = i;
term_hash_insert(sort_hash,pat,s,3);
}
return (gen_e)pat;
}
gen_e setif_proj_pat(constructor c,int i,gen_e e)
{
return proj_pat(c,i,e);
}
gen_e setst_proj_pat(constructor c, int i, gen_e e)
{
return proj_pat(c,i,e);
}
gen_e setif_proj(constructor c, int i, gen_e e)
{
setif_var v;
gen_e proj_var, proj;
gen_e nonspec_get_proj(gen_e_list arg1)
{
proj_pat pat;
gen_e_list_scanner scan;
gen_e temp;
gen_e_list_scan(arg1,&scan);
while (gen_e_list_next(&scan,&temp))
{
pat = (proj_pat)temp;
if ( pat_match(pat->type,c->type) && i == pat->i )
return pat->exp;
}
return NULL;
}
if (e->sort != setif_sort)
{
failure("Sort check : setif_proj\n");
}
else if (i < 0 || i > c->arity)
{
failure("Signature mismatch\n");
}
else if (setif_is_zero(e))
return get_sort_zero(c->sig[i].sort);
else if ( ((setif_term)e)->type == c->type )
{
cons_expr constructed = (cons_expr)e;
return constructed->exps[i];
}
else if (setif_is_var(e))
{
v = (setif_var)e;
if ( (proj = sv_get_ub_proj(v,nonspec_get_proj)) )
{
return proj;
}
else
{
gen_e pat;
gen_e_list_scanner scan;
gen_e lb;
proj_var = get_proj_var(c->sig[i].sort,FALSE);
pat = setif_proj_pat(c,i,proj_var);
sv_add_ub_proj(sort_region,v,pat);
gen_e_list_scan(sv_get_lbs(v),&scan);
while (gen_e_list_next(&scan,&lb))
{
setif_inclusion(lb,pat);
}
return proj_var;
}
}
else if (setif_is_union(e))
{
if( (proj = nonspec_get_proj(setif_get_proj_cache(e))) )
return proj;
else
{
gen_e pat;
proj_var = get_proj_var(c->sig[i].sort,FALSE);
pat = setif_proj_pat(c,i,proj_var);
setif_set_proj_cache(e,pat);
setif_inclusion(e,pat);
return proj_var;
}
}
else
{
gen_e pat;
proj_var = get_proj_var(c->sig[i].sort,FALSE);
pat = setif_proj_pat(c,i,proj_var);
setif_inclusion(e,pat);
return proj_var;
}
}
gen_e setst_proj(constructor c, int i, gen_e e)
{
setst_var v;
gen_e proj_var, proj;
gen_e nonspec_get_proj(gen_e_list arg1)
{
proj_pat pat;
gen_e_list_scanner scan;
gen_e temp;
gen_e_list_scan(arg1,&scan);
while (gen_e_list_next(&scan,&temp))
{
pat = (proj_pat)temp;
if ( pat_match(pat->type,c->type) && i == pat->i )
return pat->exp;
}
return NULL;
}
if (e->sort != setst_sort)
{
failure("Sort check : setst_proj\n");
}
else if (i < 0 || i > c->arity)
{
failure("Signature mismatch\n");
}
else if (setst_is_zero(e))
return get_sort_zero(c->sig[i].sort);
else if ( ((setst_term)e)->type == c->type )
{
cons_expr constructed = (cons_expr)e;
return constructed->exps[i];
}
else if (setst_is_var(e))
{
v = (setst_var)e;
if ( (proj = sv_get_ub_proj(v,nonspec_get_proj)) )
{
return proj;
}
else
{
gen_e pat;
gen_e_list_scanner scan;
gen_e lb;
proj_var = get_proj_var(c->sig[i].sort,FALSE);
pat = setst_proj_pat(c,i,proj_var);
sv_add_ub_proj(sort_region,v,pat);
gen_e_list_scan(sv_get_lbs(v),&scan);
while (gen_e_list_next(&scan,&lb))
{
setst_inclusion(lb,pat);
}
return proj_var;
}
}
else if (setst_is_union(e))
{
if( (proj = nonspec_get_proj(setst_get_proj_cache(e))) )
return proj;
else
{
gen_e pat;
proj_var = get_proj_var(c->sig[i].sort,FALSE);
pat = setst_proj_pat(c,i,proj_var);
setst_set_proj_cache(e,pat);
setst_inclusion(e,pat);
return proj_var;
}
}
else
{
gen_e pat;
proj_var = get_proj_var(c->sig[i].sort,FALSE);
pat = setst_proj_pat(c,i,proj_var);
setst_inclusion(e,pat);
return proj_var;
}
}
static void setif_con_match(gen_e e1, gen_e e2)
{
// Case where e1 is a constructor expression and e2 is a proj_pat
if (pat_match(((setif_term)e2)->type,((setif_term)e1)->type))
{
cons_expr c = (cons_expr)e1;
proj_pat p = (proj_pat)e2;
int i = p->i;
if (c->sig[i].variance == vnc_pos)
call_inclusion_fn(c->exps[i],p->exp);
else if (c->sig[i].variance == vnc_neg)
call_inclusion_fn(p->exp,c->exps[i]);
else
call_unify_fn(c->exps[i],p->exp);
}
else if (setif_is_pat(e2))
{
return;
}
// Case where e1 and e2 are constructor expressions
else
{
cons_expr c1 = (cons_expr)e1,
c2 = (cons_expr)e2;
if (c1->type != c2->type)
failure("Constructor mismatch\n");
else
{
int i;
for (i = 0; i < c1->arity; i++)
{
if (c1->sig[i].variance == vnc_pos)
call_inclusion_fn(e1,e2);
else if (c1->sig[i].variance == vnc_neg)
call_inclusion_fn(e2,e1);
else
call_unify_fn(e1,e2);
}
}
}
}
static void setst_con_match(gen_e e1, gen_e e2)
{
// Case where e1 is a constructor expression and e2 is a proj_pat
if (pat_match(((setst_term)e2)->type,((setst_term)e1)->type))
{
cons_expr c = (cons_expr)e1;
proj_pat p = (proj_pat)e2;
int i = p->i;
if (c->sig[i].variance == vnc_pos)
call_inclusion_fn(c->exps[i],p->exp);
else if (c->sig[i].variance == vnc_neg)
call_inclusion_fn(p->exp,c->exps[i]);
else
call_unify_fn(c->exps[i],p->exp);
}
else if (setst_is_pat(e2))
{
return;
}
// Case where e1 and e2 are constructor expressions
else
{
cons_expr c1 = (cons_expr)e1,
c2 = (cons_expr)e2;
if (c1->type != c2->type)
failure("Constructor mismatch\n");
else
{
int i;
for (i = 0; i < c1->arity; i++)
{
if (c1->sig[i].variance == vnc_pos)
call_inclusion_fn(e1,e2);
else if (c1->sig[i].variance == vnc_neg)
call_inclusion_fn(e2,e1);
else
call_unify_fn(e1,e2);
}
}
}
}
// given x <= proj(c,i,e)
// proj_merge(region,e,get_proj_i_arg,fresh_large_fn_ptr,
// sort_inclusion_fn_ptr,set_inclusion)
static bool nonspec_res_proj(set_var v1,gen_e e2)
{
proj_pat projection_pat = (proj_pat)e2;
gen_e setif_get_proj(gen_e_list arg1)
{
gen_e_list_scanner scan;
gen_e temp;
proj_pat pat;
gen_e_list_scan(arg1,&scan);
while(gen_e_list_next(&scan,&temp))
{
pat = (proj_pat)temp;
if ( pat->type == ((setif_term)e2)->type &&
pat->i == ((proj_pat)e2)->i)
return pat->exp;
}
return NULL;
}
gen_e fresh_large(void)
{
return get_proj_var( ((proj_pat)e2)->exp->sort,TRUE);
}
bool sort_inclusion(gen_e e1, gen_e e2)
{
if ( projection_pat->variance == vnc_pos )
return call_inclusion_fn(e1,e2);
else if ( projection_pat->variance == vnc_neg)
return call_inclusion_fn(e2,e1);
else
return call_unify_fn(e1,e2);
}
gen_e proj_con(gen_e e)
{
return make_proj_pat( ((proj_pat)e2)->c, ((proj_pat)e2)->i,e);
}
return setif_proj_merge(setif_region,v1,((proj_pat)e2)->exp,
setif_get_proj,proj_con,
fresh_large,sort_inclusion,
call_setif_inclusion);
}
void call_setif_inclusion(gen_e e1,gen_e e2)
{
setif_inclusion(setif_con_match,setif_res_proj,e1,e2);
}
void call_setif_unify(gen_e e1, gen_e e2)
{
setif_inclusion(setif_con_match,setif_res_proj,e1,e2);
setif_inclusion(setif_con_match,setif_res_proj,e2,e1);
}
void call_setst_inclusion(gen_e e1, gen_e e2)
{
setst_inclusion(setst_con_match,e1,e2);
}
void call_setst_unify(gen_e e1, gen_e e2)
{
setst_inclusion(setst_con_match,e1,e2);
setst_inclusion(setst_con_match,e2,e1);
}
void call_flowrow_inclusion(gen_e e1,gen_e e2)
{
if ( (e1->sort != flowrow_sort) || (e2->sort != flowrow_sort) )
failure("Constraint system is not well-sorted\n");
if ( flowrow_base_sort(e1) != flowrow_base_sert(e2))
failure("Constraint system is not well-sorted\n");
flowrow_inclusion(fresh,get_stamp,field_incl,zero_elem,e1,e2);
}
void call_flowrow_unify(gen_e e1, gen_e e2)
{
flowrow_inclusion(fresh,get_stamp,field_incl,zero_elem,e1,e2);
flowrow_inclusion(fresh,get_stamp,field_incl,zero_elem,e2,e1);
}
static void term_con_match(gen_e e1, gen_e e2)
{
cons_expr c1 = (cons_expr)e1,
c2 = (cons_expr)e2;
if (c1->type != c2->type)
failure("Constructor mismatch\n");
else
{
int i;
for (i = 0; i < c1->arity; i++)
{
call_unify_fn(e1,e2);
}
}
}
static void term_occurs(term_var v, gen_e e)
{
gen_e ecr = term_get_ecr(e);
if (((gen_term)ecr)->type == VAR_TYPE)
return ( term_get_stamp((gen_e)v) == term_get_stamp(e) );
else if (((gen_term)ecr)->type >= 10)
{
cons_expr c_e = (cons_expr) e;
int i;
for (int i = 0; i < arity; i++)
{
if (term_occurs(v,c->exps[i]))
return TRUE;
}
}
return FALSE;
}
void call_term_unify(gen_e e1, gen_e e2)
{
term_unify(term_con_match,term_occurs,e1,e2);
}
void call_term_cunify(gen_e e1, gen_e e2)
{
term_cunify(term_con_match,term_occurs,e1,e2);
}
static void call_inclusion_fn(gen_e e1, gen_e e2)
{
switch (e1->sort)
{
case sort_setif:
{
setif_inclusion(setif_con_match,setif_res_proj,e1,e2);
}
break;
case sort_setst:
{
setst_inclusion(setst_con_match,e1,e2);
}
break;
case sort_term:
{
term_unify(term_con_match,term_occurs,e1,e2);
}
break;
case sort_row:
{
/* TODO */
flowrow_inclusion(fresh,get_stamp,field_incl,zero_elem,e1,e2);
}
break;
default :
fail("Unmatched sort in call inclusion\n");
}
}
static bool call_unify_fn(gen_e e1, gen_e e2)
{
switch (e1->sort)
{
case sort_setif:
{
setif_inclusion(setif_con_match,setif_res_proj,e1,e2);
setif_inclusion(setif_con_match,setif_res_proj,e2,e1);
}
break;
case sort_setst:
{
setst_inclusion(setst_con_match,e1,e2);
setst_inclusion(setst_con_match,e2,e1);
}
break;
case sort_term:
{
term_unify(term_con_match,term_occurs,e1,e2);
}
break;
case sort_row:
{
/* TODO */
flowrow_inclusion(fresh,get_stamp,field_incl,zero_elem,e1,e2);
flowrow_inclusion(fresh,get_stamp,field_incl,zero_elem,e2,e1);
}
break;
default :
fail("Unmatched sort in call inclusion\n");
}
}
void nonspec_init(void)
{
banshee_init();
setif_init();
setst_init();
flowrow_init();
}
void nonspec_reset(void)
{
flowrow_reset();
setst_reset();
setif_reset();
banshee_reset();
}
void expr_print(FILE *f,gen_e e)
{
}