[BACK]Return to strobj.c CVS log [TXT][DIR] Up to [local] / OpenXM_contrib2 / asir2000 / builtin

Diff for /OpenXM_contrib2/asir2000/builtin/strobj.c between version 1.51 and 1.69

version 1.51, 2004/08/05 00:56:54 version 1.69, 2005/10/05 08:57:25
Line 45 
Line 45 
  * DEVELOPER SHALL HAVE NO LIABILITY IN CONNECTION WITH THE USE,   * DEVELOPER SHALL HAVE NO LIABILITY IN CONNECTION WITH THE USE,
  * PERFORMANCE OR NON-PERFORMANCE OF THE SOFTWARE.   * PERFORMANCE OR NON-PERFORMANCE OF THE SOFTWARE.
  *   *
  * $OpenXM: OpenXM_contrib2/asir2000/builtin/strobj.c,v 1.50 2004/07/13 10:57:26 ohara Exp $   * $OpenXM: OpenXM_contrib2/asir2000/builtin/strobj.c,v 1.68 2005/10/05 07:38:08 noro Exp $
 */  */
 #include "ca.h"  #include "ca.h"
 #include "parse.h"  #include "parse.h"
Line 77  void Pquotetotex_tb();
Line 77  void Pquotetotex_tb();
 void Pquotetotex();  void Pquotetotex();
 void Pquotetotex_env();  void Pquotetotex_env();
 void Pflatten_quote();  void Pflatten_quote();
 void Pquote_to_funargs(),Pfunargs_to_quote();  
   void Pquote_is_integer(),Pquote_is_rational(),Pquote_is_number();
   void Pquote_is_dependent(),Pquote_is_function();
   
   void Pquote_to_funargs(),Pfunargs_to_quote(),Pget_function_name();
   void Pquote_unify(),Pget_quote_id(),Pquote_match_rewrite();
   void Pquote_to_nary(),Pquote_to_bin();
   void do_assign(NODE arg);
 void fnodetotex_tb(FNODE f,TB tb);  void fnodetotex_tb(FNODE f,TB tb);
 char *symbol_name(char *name);  char *symbol_name(char *name);
 char *conv_rule(char *name);  char *conv_rule(char *name);
Line 87  void tb_to_string(TB tb,STRING *rp);
Line 94  void tb_to_string(TB tb,STRING *rp);
 void fnodenodetotex_tb(NODE n,TB tb);  void fnodenodetotex_tb(NODE n,TB tb);
 void fargstotex_tb(char *opname,FNODE f,TB tb);  void fargstotex_tb(char *opname,FNODE f,TB tb);
 int top_is_minus(FNODE f);  int top_is_minus(FNODE f);
   int quote_unify(Obj f,Obj pat,NODE *rp);
   
 struct ftab str_tab[] = {  struct ftab str_tab[] = {
         {"sprintf",Psprintf,-99999999},          {"sprintf",Psprintf,-99999999},
Line 102  struct ftab str_tab[] = {
Line 110  struct ftab str_tab[] = {
         {"clear_tb",Pclear_tb,1},          {"clear_tb",Pclear_tb,1},
         {"tb_to_string",Ptb_to_string,1},          {"tb_to_string",Ptb_to_string,1},
         {"string_to_tb",Pstring_to_tb,1},          {"string_to_tb",Pstring_to_tb,1},
           {"get_quote_id",Pget_quote_id,1},
   
           {"quote_is_number",Pquote_is_number,1},
           {"quote_is_rational",Pquote_is_rational,1},
           {"quote_is_integer",Pquote_is_integer,1},
           {"quote_is_function",Pquote_is_function,1},
           {"quote_is_dependent",Pquote_is_dependent,2},
   
           {"quote_to_nary",Pquote_to_nary,1},
           {"quote_to_bin",Pquote_to_bin,2},
         {"quotetotex_tb",Pquotetotex_tb,2},          {"quotetotex_tb",Pquotetotex_tb,2},
         {"quotetotex",Pquotetotex,1},          {"quotetotex",Pquotetotex,1},
         {"quotetotex_env",Pquotetotex_env,-99999999},          {"quotetotex_env",Pquotetotex_env,-99999999},
         {"flatten_quote",Pflatten_quote,2},          {"flatten_quote",Pflatten_quote,-2},
         {"quote_to_funargs",Pquote_to_funargs,1},          {"quote_to_funargs",Pquote_to_funargs,1},
           {"quote_unify",Pquote_unify,2},
           {"quote_match_rewrite",Pquote_match_rewrite,-4},
         {"funargs_to_quote",Pfunargs_to_quote,1},          {"funargs_to_quote",Pfunargs_to_quote,1},
           {"get_function_name",Pget_function_name,1},
         {0,0,0},          {0,0,0},
 };  };
   
Line 503  void Pwrite_to_tb(NODE arg,Q *rp)
Line 524  void Pwrite_to_tb(NODE arg,Q *rp)
         *rp = 0;          *rp = 0;
 }  }
   
   FNODE partial_eval(FNODE), quote_to_nary(FNODE), quote_to_bin(FNODE,int);
   
   void Pquote_to_nary(NODE arg,QUOTE *rp)
   {
           FNODE f;
   
           f = quote_to_nary(BDY((QUOTE)ARG0(arg)));
           MKQUOTE(*rp,f);
   }
   
   void Pquote_to_bin(NODE arg,QUOTE *rp)
   {
           FNODE f;
           int direction;
   
           direction = QTOS((Q)ARG1(arg));
           f = quote_to_bin(BDY((QUOTE)ARG0(arg)),direction);
   
           MKQUOTE(*rp,f);
   }
   
   void Pquote_is_number(NODE arg,Q *rp)
   {
           QUOTE q;
           int ret;
   
           q = (QUOTE)ARG0(arg);
           asir_assert(q,O_QUOTE,"quote_is_number");
           ret = fnode_is_number(BDY(q));
           STOQ(ret,*rp);
   }
   
   void Pquote_is_rational(NODE arg,Q *rp)
   {
           QUOTE q;
           int ret;
   
           q = (QUOTE)ARG0(arg);
           asir_assert(q,O_QUOTE,"quote_is_rational");
           ret = fnode_is_rational(BDY(q));
           STOQ(ret,*rp);
   }
   
   void Pquote_is_integer(NODE arg,Q *rp)
   {
           QUOTE q;
           int ret;
   
           q = (QUOTE)ARG0(arg);
           asir_assert(q,O_QUOTE,"quote_is_integer");
           ret = fnode_is_integer(BDY(q));
           STOQ(ret,*rp);
   }
   
   void Pquote_is_function(NODE arg,Q *rp)
   {
           QUOTE q;
           int ret;
   
           q = (QUOTE)ARG0(arg);
           asir_assert(q,O_QUOTE,"quote_is_function");
           if ( q->id == I_FUNC || q->id == I_IFUNC )
                   ret = 1;
           else
                   ret = 0;
           STOQ(ret,*rp);
   }
   
   void Pquote_is_dependent(NODE arg,Q *rp)
   {
           P x;
           QUOTE q,v;
           int ret;
           V var;
   
           q = (QUOTE)ARG0(arg);
           v = (QUOTE)ARG1(arg);
           asir_assert(q,O_QUOTE,"quote_is_dependent");
           asir_assert(v,O_QUOTE,"quote_is_dependent");
           x = (P)eval(BDY(v));
           if ( !x || OID(x) != O_P )
                   *rp = 0;
           var = VR(x);
           ret = fnode_is_dependent(BDY(q),var);
           STOQ(ret,*rp);
   }
   
   
   void Pquote_unify(NODE arg,Q *rp)
   {
           FNODE f,g;
           Obj obj;
           QUOTE q;
           NODE r;
           int ret;
   
   #if 0
           g = partial_eval(BDY(((QUOTE)ARG0(arg))));
           MKQUOTE(q,g);
           ret = quote_unify((Obj)q,(Obj)ARG1(arg),&r);
   #else
           obj = (Obj)ARG0(arg);
           ret = quote_unify(obj,(Obj)ARG1(arg),&r);
   #endif
           if ( ret ) {
                   do_assign(r);
                   *rp = ONE;
           } else
                   *rp = 0;
   }
   
   FNODE rewrite_fnode(FNODE,NODE);
   
   extern Obj VOIDobj;
   
   void Pquote_match_rewrite(NODE arg,Obj *rp)
   {
           FNODE f,g,h,c,value;
           Obj obj;
           QUOTE q;
           NODE r,t,s,s0,pair;
           int ret,ind,ac;
   
           obj = (Obj)ARG0(arg);
           ret = quote_unify(obj,(Obj)ARG1(arg),&r);
           if ( ret ) {
                   for ( t = r, s0 = 0; t; t = NEXT(t) ) {
                           NEXTNODE(s0,s);
                           pair = BDY((LIST)BDY(t));
                           ind = (int)FA0((FNODE)BDY((QUOTE)BDY(pair)));
                           value = mkfnode(1,I_FORMULA,BDY(NEXT(pair)));
                           BDY(s) = mknode(2,ind,value);
                   }
                   if ( s0 ) NEXT(s) = 0;
                   switch ( ac = argc(arg) ) {
                           case 3:
                                   h = rewrite_fnode(BDY((QUOTE)ARG2(arg)),s0);
                                   *rp = eval(h);
                                   break;
                           case 4:
                                   c = rewrite_fnode(BDY((QUOTE)ARG2(arg)),s0);
                                   if ( eval(c) ) {
                                           h = rewrite_fnode(BDY((QUOTE)ARG3(arg)),s0);
                                           *rp = eval(h);
                                   } else
                                           *rp = VOIDobj;
                                   break;
                           default:
                                   error("quote_match_rewrite : invalid argument");
                   }
           } else
                   *rp = VOIDobj;
   }
   
   void do_assign(NODE arg)
   {
           NODE t,pair;
           int pv;
   
           QUOTE value;
   
           for ( t = arg; t; t = NEXT(t) ) {
                   pair = BDY((LIST)BDY(t));
                   pv = (int)FA0((FNODE)BDY((QUOTE)BDY(pair)));
                   value = (QUOTE)(BDY(NEXT(pair)));
                   ASSPV(pv,value);
           }
   }
   
   /*
   /* consistency check and merge
    */
   
   int merge_matching_node(NODE n,NODE a,NODE *rp)
   {
           NODE ta,ba,tn,bn;
           QUOTE pa,va,pn,vn;
   
           if ( !n ) {
                   *rp = a;
                   return 1;
           }
           for ( ta = a; ta; ta = NEXT(ta) ) {
                   ba = BDY((LIST)BDY(ta));
                   if ( !ba ) continue;
                   pa = (QUOTE)BDY(ba); va = (QUOTE)BDY(NEXT(ba));
                   for ( tn = n; tn; tn = NEXT(tn) ) {
                           bn = BDY((LIST)BDY(tn));
                           if ( !bn ) continue;
                           pn = (QUOTE)BDY(bn); vn = (QUOTE)BDY(NEXT(bn));
                           if ( !compquote(CO,pa,pn) ) {
                                   if ( !compquote(CO,va,vn) ) break;
                                   else return 0;
                           }
                   }
                   if ( !tn ) {
                           MKNODE(tn,(pointer)BDY(ta),n);
                           n = tn;
                   }
           }
           *rp = n;
           return 1;
   }
   
   int quote_unify_node(NODE f,NODE pat,NODE *rp) {
           NODE r,a,tf,tp,r1;
           int ret;
   
           if ( length(f) != length(pat) ) return 0;
           r = 0;
           for ( tf = f, tp = pat; tf; tf = NEXT(tf), tp = NEXT(tp) ) {
                   ret = quote_unify((Obj)BDY(tf),(Obj)BDY(tp),&a);
                   if ( !ret ) return 0;
                   ret = merge_matching_node(r,a,&r1);
                   if ( !ret ) return 0;
                   else r = r1;
           }
           *rp = r;
           return 1;
   }
   
   /* f = [a,b,c,...] pat = [X,Y,...] rpat matches the rest of f */
   
   int quote_unify_cons(NODE f,NODE pat,Obj rpat,NODE *rp) {
           QUOTE q;
           Q id;
           FNODE fn;
           NODE r,a,tf,tp,r1,arg;
           int ret;
           LIST list,alist;
   
           /* matching of the head part */
           if ( length(f) < length(pat) ) return 0;
           r = 0;
           for ( tf = f, tp = pat; tp; tf = NEXT(tf), tp = NEXT(tp) ) {
                   ret = quote_unify((Obj)BDY(tf),(Obj)BDY(tp),&a);
                   if ( !ret ) return 0;
                   ret = merge_matching_node(r,a,&r1);
                   if ( !ret ) return 0;
                   else r = r1;
           }
           /* matching of the rest */
           MKLIST(list,tf);
           STOQ(I_LIST,id); a = mknode(2,id,list);
           MKLIST(alist,a);
           arg = mknode(1,alist);
           Pfunargs_to_quote(arg,&q);
           ret = quote_unify((Obj)q,rpat,&a);
           if ( !ret ) return 0;
           ret = merge_matching_node(r,a,&r1);
           if ( !ret ) return 0;
           *rp = r1;
           return 1;
   }
   
   void get_quote_id_arg(QUOTE f,int *id,NODE *r)
   {
           LIST fa;
           NODE arg,fab;
   
           arg = mknode(1,f); Pquote_to_funargs(arg,&fa); fab = BDY((LIST)fa);
           *id = QTOS((Q)BDY(fab)); *r = NEXT(fab);
   }
   
   /* *rp : [[quote(A),quote(1)],...] */
   
   int quote_unify(Obj f, Obj pat, NODE *rp)
   {
           NODE tf,tp,head,body;
           NODE parg,farg,r;
           Obj rpat;
           LIST fa,l;
           int pid,id;
           FUNC ff,pf;
           int ret;
           QUOTE q;
           FNODE g;
   
           if ( !f )
                   if ( !pat ) {
                           *rp = 0; return 1;
                   } else
                           return 0;
           else if ( OID(pat) == O_LIST ) {
                   if ( OID(f) == O_LIST )
                           return quote_unify_node(BDY((LIST)f),BDY((LIST)pat),rp);
                   else
                           return 0;
           } else if ( OID(pat) == O_QUOTE ) {
                   pid = ((FNODE)BDY((QUOTE)pat))->id;
                   switch ( pid ) {
                           case I_FORMULA:
                                   if ( compquote(CO,f,pat) )
                                           return 0;
                                   else {
                                           *rp = 0; return 1;
                                   }
                                   break;
   
                           case I_LIST: case I_CONS:
                                   get_quote_id_arg((QUOTE)pat,&pid,&parg);
                                   if ( OID(f) == O_LIST )
                                           tf = BDY((LIST)f);
                                   else if ( OID(f) == O_QUOTE
                                           && ((FNODE)BDY((QUOTE)f))->id == pid ) {
                                           get_quote_id_arg((QUOTE)f,&id,&farg);
                                           tf = BDY((LIST)BDY(farg));
                                   } else
                                           return 0;
   
                                   tp = BDY((LIST)BDY(parg));
                                   if ( pid == I_LIST )
                                           return quote_unify_node(tf,tp,rp);
                                   else {
                                           rpat = (Obj)BDY(NEXT(parg));
                                           return quote_unify_cons(tf,tp,rpat,rp);
                                   }
   
                           case I_PVAR:
                                   /* [[pat,f]] */
                                   r = mknode(2,pat,f); MKLIST(l,r);
                                   *rp =  mknode(1,l);
                                   return 1;
   
                           case I_IFUNC:
                                   /* F(X,Y,...) = ... */
                                   get_quote_id_arg((QUOTE)f,&id,&farg);
                                   get_quote_id_arg((QUOTE)pat,&pid,&parg);
                                   if ( id == I_FUNC ) {
                                           r = mknode(2,BDY(parg),BDY(farg)); MKLIST(l,r);
                                           head = mknode(1,l);
                                           ret = quote_unify(BDY(NEXT(farg)),
                                                                   BDY(NEXT(parg)),&body);
                                           if ( !ret ) return 0;
                                           else return merge_matching_node(head,body,rp);
                                   } else
                                           return 0;
   
                           case I_NARYOP: case I_BOP: case I_FUNC:
                                   /* X+Y = ... */
                                   /* f(...) = ... */
                                   if ( OID(f) != O_QUOTE ) return 0;
                                   id = ((FNODE)BDY((QUOTE)f))->id;
                                   if ( pid == I_FUNC )
                                           ;
                                   else {
                                           /* XXX converting to I_BOP */
                                           if ( pid == I_NARYOP ) {
                                                   g = quote_to_bin(BDY((QUOTE)pat),1);
                                                   MKQUOTE(q,g); pat = (Obj)q;
                                           }
                                           if ( id == I_NARYOP ) {
                                                   g = quote_to_bin(BDY((QUOTE)f),1);
                                                   MKQUOTE(q,g); f = (Obj)q;
                                           }
                                   }
                                   get_quote_id_arg((QUOTE)pat,&pid,&parg);
                                   get_quote_id_arg((QUOTE)f,&id,&farg);
                                   if ( compqa(CO,BDY(farg),BDY(parg)) ) return 0;
                                   return quote_unify_node(NEXT(farg),NEXT(parg),rp);
   
                           default:
                                   if ( OID(f) != O_QUOTE ) return 0;
                                   id = ((FNODE)BDY((QUOTE)f))->id;
                                   if ( id != pid ) return 0;
                                   get_quote_id_arg((QUOTE)pat,&pid,&parg);
                                   get_quote_id_arg((QUOTE)f,&id,&farg);
                                   return quote_unify_node(farg,parg,rp);
                   }
           }
   }
   
 void Pquotetotex(NODE arg,STRING *rp)  void Pquotetotex(NODE arg,STRING *rp)
 {  {
         TB tb;          TB tb;
Line 844  char *symbol_name(char *name)
Line 1237  char *symbol_name(char *name)
         return 0;          return 0;
 }  }
   
   void Pget_function_name(NODE arg,STRING *rp)
   {
                   QUOTEARG qa;
                   ARF f;
                   char *opname;
   
                   qa = (QUOTEARG)BDY(arg);
                   if ( !qa || OID(qa) != O_QUOTEARG || qa->type != A_arf )
                           *rp = 0;
                   else {
                           f = (ARF)BDY(qa);
                           opname = f->name;
                           MKSTR(*rp,opname);
                   }
   }
   
 FNODE strip_paren(FNODE);  FNODE strip_paren(FNODE);
   
 void fnodetotex_tb(FNODE f,TB tb)  void fnodetotex_tb(FNODE f,TB tb)
Line 1333  void Pflatten_quote(NODE arg,Obj *rp)
Line 1742  void Pflatten_quote(NODE arg,Obj *rp)
   
         if ( !ARG0(arg) || OID((Obj)ARG0(arg)) != O_QUOTE )          if ( !ARG0(arg) || OID((Obj)ARG0(arg)) != O_QUOTE )
                 *rp = (Obj)ARG0(arg);                  *rp = (Obj)ARG0(arg);
         else {          else if ( argc(arg) == 1 ) {
                   f = flatten_fnode(BDY((QUOTE)ARG0(arg)),"+");
                   f = flatten_fnode(f,"*");
                   MKQUOTE(q,f);
                   *rp = (Obj)q;
           } else {
                 f = flatten_fnode(BDY((QUOTE)ARG0(arg)),BDY((STRING)ARG1(arg)));                  f = flatten_fnode(BDY((QUOTE)ARG0(arg)),BDY((STRING)ARG1(arg)));
                 MKQUOTE(q,f);                  MKQUOTE(q,f);
                 *rp = (Obj)q;                  *rp = (Obj)q;
         }          }
 }  }
   
   void Pget_quote_id(NODE arg,Q *rp)
   {
           FNODE f;
           QUOTE q;
   
           q = (QUOTE)ARG0(arg);
           if ( !q || OID(q) != O_QUOTE )
                   error("get_quote_id : invalid argument");
           f = BDY(q);
           STOQ((int)f->id,*rp);
   }
   
 void Pquote_to_funargs(NODE arg,LIST *rp)  void Pquote_to_funargs(NODE arg,LIST *rp)
 {  {
         fid_spec_p spec;          fid_spec_p spec;
Line 1478  void Pfunargs_to_quote(NODE arg,QUOTE *rp)
Line 1904  void Pfunargs_to_quote(NODE arg,QUOTE *rp)
                 }                  }
         }          }
         MKQUOTE(*rp,f);          MKQUOTE(*rp,f);
   }
   
   int fnode_is_number(FNODE f)
   {
           Obj obj;
   
           switch ( f->id ) {
                   case I_MINUS: case I_PAREN:
                           return fnode_is_number(FA0(f));
   
                   case I_FORMULA:
                           obj = FA0(f);
                           if ( !obj ) return 1;
                           else if ( OID(obj) == O_QUOTE )
                                   return fnode_is_number(BDY((QUOTE)obj));
                           else if ( NUM(obj) ) return 1;
                           else return 0;
   
                   case I_BOP:
                           return fnode_is_number(FA1(f)) && fnode_is_number(FA2(f));
   
                   default:
                           return 0;
           }
   }
   
   int fnode_is_rational(FNODE f)
   {
           Obj obj;
   
           switch ( f->id ) {
                   case I_MINUS: case I_PAREN:
                           return fnode_is_number(FA0(f));
   
                   case I_FORMULA:
                           obj = FA0(f);
                           if ( !obj ) return 1;
                           else if ( OID(obj) == O_QUOTE )
                                   return fnode_is_rational(BDY((QUOTE)obj));
                           else if ( NUM(obj) && RATN(obj) ) return 1;
                           else return 0;
   
                   case I_BOP:
                           if ( !strcmp(((ARF)FA0(f))->name,"^")  )
                                   return fnode_is_rational(FA1(f)) && fnode_is_integer(FA2(f));
                           else
                                   return fnode_is_rational(FA1(f)) && fnode_is_rational(FA2(f));
   
                   default:
                           return 0;
           }
   }
   
   int fnode_is_integer(FNODE f)
   {
           Obj obj;
   
           switch ( f->id ) {
                   case I_MINUS: case I_PAREN:
                           return fnode_is_integer(FA0(f));
   
                   case I_FORMULA:
                           obj = FA0(f);
                           if ( !obj ) return 1;
                           else if ( OID(obj) == O_QUOTE )
                                   return fnode_is_integer(BDY((QUOTE)obj));
                           else if ( INT(obj)) return 1;
                           else return 0;
   
                   case I_BOP:
                           if ( !strcmp(((ARF)FA0(f))->name,"^")  )
                                   return fnode_is_integer(FA1(f))
                                           && fnode_is_nonnegative_integer(FA2(f));
                           else if ( !strcmp(((ARF)FA0(f))->name,"/")  )
                                   return fnode_is_integer(FA1(f)) &&
                                           ( fnode_is_one(FA2(f)) || fnode_is_minusone(FA2(f)) );
                           else
                                   return fnode_is_integer(FA1(f)) && fnode_is_integer(FA2(f));
   
                   default:
                           return 0;
           }
   }
   
   int fnode_is_nonnegative_integer(FNODE f)
   {
           Q n;
   
           n = eval(f);
           if ( !n || (INT(n) && SGN(n) > 0) ) return 1;
           else return 0;
   }
   
   int fnode_is_one(FNODE f)
   {
           Q n;
   
           n = eval(f);
           if ( UNIQ(n) ) return 1;
           else return 0;
   }
   
   int fnode_is_minusone(FNODE f)
   {
           Q n;
   
           n = eval(f);
           if ( MUNIQ(n) ) return 1;
           else return 0;
   }
   
   int fnode_is_dependent(FNODE f,V v)
   {
           Obj obj;
           FNODE arg;
           NODE t;
   
           switch ( f->id ) {
                   case I_MINUS: case I_PAREN:
                           return fnode_is_dependent(FA0(f),v);
   
                   case I_FORMULA:
                           obj = FA0(f);
                           if ( !obj ) return 0;
                           else if ( OID(obj) == O_QUOTE )
                                   return fnode_is_dependent(BDY((QUOTE)obj),v);
                           else if ( obj_is_dependent(obj,v) ) return 1;
                           else return 0;
   
                   case I_BOP:
                           return fnode_is_dependent(FA1(f),v) || fnode_is_dependent(FA2(f),v);
   
                   case I_FUNC:
                           arg = (FNODE)FA1(f);
                           for ( t = FA0(arg); t; t = NEXT(t) )
                                   if ( fnode_is_dependent(BDY(t),v) ) return 1;
                           return 0;
   
                   default:
                           return 0;
           }
 }  }

Legend:
Removed from v.1.51  
changed lines
  Added in v.1.69

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>