diff --git a/src/compilation.h b/src/compilation.h index 51cd82409547567fcd6485caa24722d8474979e8..6c614ed7fecbf76ff2adc8dc149654a8f480b07b 100644 --- a/src/compilation.h +++ b/src/compilation.h @@ -75,18 +75,6 @@ ZMEMBER(struct pike_string *,X) #endif -#ifdef FILE_STATE - SNAME(file_state,previous_file_state) - ZMEMBER(INT32,old_line) - ZMEMBER(INT32,current_line) - ZMEMBER(INT32,nexpands) - ZMEMBER(int,pragma_all_inline) - ZMEMBER(struct inputstate *,istate) - ZMEMBER(struct hash_table *,defines) - STRMEMBER(current_file,"current_file") - SEND -#endif - #ifdef PROGRAM_STATE SNAME(program_state,previous_program_state) ZMEMBER(INT32,last_line) @@ -97,11 +85,26 @@ ZMEMBER(int,num_parse_error) ZMEMBER(struct locals *,local_variables) ZMEMBER(dynamic_buffer,inherit_names) + ZMEMBER(dynamic_buffer,used_modules) ZMEMBER2(dynamic_buffer,areas,[NUM_AREAS]) IMEMBER(int,comp_stackp) SEND #endif +#ifdef FILE_STATE + SNAME(file_state,previous_file_state) + ZMEMBER(INT32,old_line) + ZMEMBER(INT32,current_line) + ZMEMBER(INT32,nexpands) + ZMEMBER(int,pragma_all_inline) + ZMEMBER(struct inputstate *,istate) + ZMEMBER(struct hash_table *,defines) + ZMEMBER(struct program_state *,previous_program_state) + STRMEMBER(current_file,"current_file") + SEND +#endif + + #undef PCODE #undef STRMEMBER #undef IMEMBER diff --git a/src/las.c b/src/las.c index 49a8ab05f381f3f9fc5c76c7de9d625d46097c6b..1fa17efff8171f9e15b32b9931b7bec783cd6ffc 100644 --- a/src/las.c +++ b/src/las.c @@ -4,7 +4,7 @@ ||| See the files COPYING and DISCLAIMER for more information. \*/ #include "global.h" -RCSID("$Id: las.c,v 1.12 1997/01/19 09:08:00 hubbe Exp $"); +RCSID("$Id: las.c,v 1.13 1997/01/22 05:19:45 hubbe Exp $"); #include "language.h" #include "interpret.h" @@ -1138,40 +1138,56 @@ void fix_type_field(node *n) break; case F_INDEX: + type_a=CAR(n)->type; + type_b=CDR(n)->type; + if(!check_indexing(type_a, type_b, n)) + my_yyerror("Indexing on illegal type."); + n->type=index_type(type_a,n); + break; + case F_ARROW: type_a=CAR(n)->type; type_b=CDR(n)->type; - if(!check_indexing(type_a, type_b)) + if(!check_indexing(type_a, type_b, n)) my_yyerror("Indexing on illegal type."); - n->type=index_type(type_a); + n->type=index_type(type_a,n); break; case F_APPLY: { struct pike_string *s; + struct pike_string *f; push_type(T_MIXED); /* match any return type, even void */ push_type(T_VOID); /* not varargs */ push_type(T_MANY); low_build_function_type(CDR(n)); push_type(T_FUNCTION); s=pop_type(); - - n->type=check_call(s,CAR(n)->type?CAR(n)->type:mixed_type_string); + f=CAR(n)->type?CAR(n)->type:mixed_type_string; + n->type=check_call(s,f); if(!n->type) { + char *name; + int args; switch(CAR(n)->token) { case F_IDENTIFIER: setup_fake_program(); - my_yyerror("Bad argument %d to '%s'.", - max_correct_args+1, - ID_FROM_INT(& fake_program, CAR(n)->u.number)->name->str); + name=ID_FROM_INT(&fake_program, CAR(n)->u.number)->name->str; break; - + case F_CONSTANT: default: - my_yyerror("Bad argument %d to function call.",max_correct_args+1); + name="function call"; + } + + if(max_correct_args == count_arguments(s)) + { + my_yyerror("To few arguments to %s.\n",name); + }else{ + my_yyerror("Bad argument %d to %s.", + max_correct_args+1, name); } copy_shared_string(n->type, mixed_type_string); } diff --git a/src/pike_types.c b/src/pike_types.c index a2450b2e9793f96e20ff3bb2fb8257a16cb5f76c..4d2f950a186b86d577bf495ed2ea71e3a541643d 100644 --- a/src/pike_types.c +++ b/src/pike_types.c @@ -4,7 +4,7 @@ ||| See the files COPYING and DISCLAIMER for more information. \*/ #include "global.h" -RCSID("$Id: pike_types.c,v 1.12 1997/01/04 05:09:25 hubbe Exp $"); +RCSID("$Id: pike_types.c,v 1.13 1997/01/22 05:19:45 hubbe Exp $"); #include <ctype.h> #include "svalue.h" #include "pike_types.h" @@ -18,6 +18,8 @@ RCSID("$Id: pike_types.c,v 1.12 1997/01/04 05:09:25 hubbe Exp $"); #include "mapping.h" #include "macros.h" #include "error.h" +#include "las.h" +#include "language.h" int max_correct_args; @@ -890,10 +892,44 @@ int match_types(struct pike_string *a,struct pike_string *b) /* FIXME, add the index */ -static struct pike_string *low_index_type(char *t) +static struct pike_string *low_index_type(char *t, node *n) { switch(EXTRACT_UCHAR(t++)) { + case T_OBJECT: + { + struct program *p=id_to_program(EXTRACT_INT(t)); + if(p) + { + if(n->token == F_ARROW) + { + if(find_identifier("`->",p) != -1 || find_identifier("`->=",p) != -1) + { + reference_shared_string(mixed_type_string); + return mixed_type_string; + } + }else{ + if(find_identifier("`[]",p) != -1 || find_identifier("`[]=",p) != -1) + { + reference_shared_string(mixed_type_string); + return mixed_type_string; + } + } + if(CDR(n)->token == F_CONSTANT && CDR(n)->u.sval.type==T_STRING) + { + INT32 i; + i=find_shared_string_identifier(CDR(n)->u.sval.u.string, p); + if(i==-1) + { + reference_shared_string(int_type_string); + return int_type_string; + }else{ + reference_shared_string(ID_FROM_INT(p, i)->type); + return ID_FROM_INT(p, i)->type; + } + } + } + } default: reference_shared_string(mixed_type_string); return mixed_type_string; @@ -901,9 +937,9 @@ static struct pike_string *low_index_type(char *t) case T_OR: { struct pike_string *a,*b; - a=low_index_type(t); + a=low_index_type(t,n); t+=type_length(t); - b=low_index_type(t); + b=low_index_type(t,n); if(!b) return a; if(!a) return b; push_finished_type(b); @@ -913,7 +949,7 @@ static struct pike_string *low_index_type(char *t) } case T_AND: - return low_index_type(t+type_length(t)); + return low_index_type(t+type_length(t),n); case T_STRING: /* always int */ case T_MULTISET: /* always int */ @@ -928,35 +964,51 @@ static struct pike_string *low_index_type(char *t) } } -struct pike_string *index_type(struct pike_string *type) +struct pike_string *index_type(struct pike_string *type, node *n) { struct pike_string *t; - t=low_index_type(type->str); + t=low_index_type(type->str,n); if(!t) copy_shared_string(t,mixed_type_string); return t; } -static int low_check_indexing(char *type, char *index_type) +static int low_check_indexing(char *type, char *index_type, node *n) { switch(EXTRACT_UCHAR(type++)) { case T_OR: - return low_check_indexing(type,index_type) || - low_check_indexing(type+type_length(type),index_type); + return low_check_indexing(type,index_type,n) || + low_check_indexing(type+type_length(type),index_type,n); case T_AND: - return low_check_indexing(type,index_type) && - low_check_indexing(type+type_length(type),index_type); + return low_check_indexing(type,index_type,n) && + low_check_indexing(type+type_length(type),index_type,n); case T_NOT: - return !low_check_indexing(type,index_type); + return !low_check_indexing(type,index_type,n); case T_STRING: case T_ARRAY: return !!low_match_types(int_type_string->str, index_type,0); case T_OBJECT: - return !!low_match_types(string_type_string->str, index_type,0); + { + struct program *p=id_to_program(EXTRACT_INT(type)); + if(p) + { + if(n->token == F_ARROW) + { + if(find_identifier("`->",p) != -1 || find_identifier("`->=",p) != -1) + return 1; + }else{ + if(find_identifier("`[]",p) != -1 || find_identifier("`[]=",p) != -1) + return 1; + } + return !!low_match_types(string_type_string->str, index_type,0); + }else{ + return 1; + } + } case T_MULTISET: case T_MAPPING: @@ -971,12 +1023,13 @@ static int low_check_indexing(char *type, char *index_type) } int check_indexing(struct pike_string *type, - struct pike_string *index_type) + struct pike_string *index_type, + node *n) { CHECK_TYPE(type); CHECK_TYPE(index_type); - return low_check_indexing(type->str, index_type->str); + return low_check_indexing(type->str, index_type->str, n); } /* Count the number of arguments for a funciton type. diff --git a/src/pike_types.h b/src/pike_types.h index 77bd058263d49255db3319d59c84ab5a71288aff..50a0af8c677dbf9547f9e4be18946c1b3825291d 100644 --- a/src/pike_types.h +++ b/src/pike_types.h @@ -6,6 +6,8 @@ #ifndef PIKE_TYPES_H #define PIKE_TYPES_H +#include "las.h" + extern int max_correct_args; extern struct pike_string *string_type_string; extern struct pike_string *int_type_string; @@ -29,6 +31,7 @@ void pop_type_stack(); void type_stack_pop_to_mark(); void type_stack_reverse(); void push_type(unsigned char tmp); +void push_type_int(unsigned INT32 i); void push_unfinished_type(char *s); void push_finished_type(struct pike_string *type); struct pike_string *pop_unfinished_type(); @@ -40,12 +43,13 @@ char *low_describe_type(char *t); struct pike_string *describe_type(struct pike_string *type); TYPE_T compile_type_to_runtime_type(struct pike_string *s); int match_types(struct pike_string *a,struct pike_string *b); -struct pike_string *index_type(struct pike_string *type); +struct pike_string *index_type(struct pike_string *type, node *n); int check_indexing(struct pike_string *type, - struct pike_string *index_type); + struct pike_string *index_type, + node *n); int count_arguments(struct pike_string *s); struct pike_string *check_call(struct pike_string *args, - struct pike_string *type); + struct pike_string *type); void check_array_type(struct array *a); struct pike_string *get_type_of_svalue(struct svalue *s); char *get_name_of_type(int t); diff --git a/src/program.c b/src/program.c index 72c2eeda446c95e0df25fbbd90ff79a846010ec2..9b9cd5d668c59b0ef0d8173a48d3c406a0d1f548 100644 --- a/src/program.c +++ b/src/program.c @@ -4,7 +4,7 @@ ||| See the files COPYING and DISCLAIMER for more information. \*/ #include "global.h" -RCSID("$Id: program.c,v 1.13 1997/01/19 09:08:03 hubbe Exp $"); +RCSID("$Id: program.c,v 1.14 1997/01/22 05:19:46 hubbe Exp $"); #include "program.h" #include "object.h" #include "dynamic_buffer.h" @@ -85,6 +85,9 @@ void use_module(struct svalue *s) } +static int low_find_shared_string_identifier(struct pike_string *name, + struct program *prog); + int find_module_identifier(struct pike_string *ident) { JMP_BUF tmp; @@ -116,6 +119,45 @@ int find_module_identifier(struct pike_string *ident) } } UNSETJMP(tmp); + + { + struct program_state *p; + for(p=previous_program_state;p;p=p->previous) + { + INT32 i; + if(previous_file_state && + previous_file_state->previous_program_state==p->previous) + break; + + i=low_find_shared_string_identifier(ident, &p->fake_program); + if(i!=-1) + { + struct identifier *id; + id=ID_FROM_INT(&p->fake_program, i); + if(IDENTIFIER_IS_CONSTANT(id->flags)) + { + push_svalue(PROG_FROM_INT(&p->fake_program, i)->constants+ + id->func.offset); + return 1; + }else{ + yyerror("Identifier is not a constant"); + return 0; + } + } + } + } + + return 0; +} + +/* This should be optimized */ +struct program *id_to_program(INT32 id) +{ + struct program *p; + if(id) + for(p=first_program;p;p=p->next) + if(id==p->id) + return p; return 0; } @@ -161,12 +203,18 @@ void start_new_program() struct pike_string *name; threads_disabled++; + if(local_variables) + setup_fake_program(); #define PROGRAM_STATE #define PUSH #include "compilation.h" #undef PUSH #undef PROGRAM_STATE + if(previous_program_state->fake_program.num_inherits) + previous_program_state->fake_program.inherits[0].prog= + &previous_program_state->fake_program; + for(e=0; e<NUM_AREAS; e++) low_reinit_buf(areas + e); low_reinit_buf(& inherit_names); low_reinit_buf(& used_modules); @@ -569,6 +617,8 @@ struct program *end_program() #include "compilation.h" #undef POP #undef PROGRAM_STATE + if(fake_program.num_inherits) + fake_program.inherits[0].prog=&fake_program; threads_disabled--; free_all_nodes(); return prog; diff --git a/src/program.h b/src/program.h index 115911a3155a610e32a45efd3bf8c0b6efc5227c..c90a9240adac64fa069dc6a04d50cb02551be4d7 100644 --- a/src/program.h +++ b/src/program.h @@ -184,6 +184,7 @@ extern struct program fake_program; /* Prototypes begin here */ void use_module(struct svalue *s); int find_module_identifier(struct pike_string *ident); +struct program *id_to_program(INT32 id); void setup_fake_program(); void start_new_program(); void really_free_program(struct program *p); diff --git a/src/testsuite.in b/src/testsuite.in index 5247de7b7d5532b5dc2ab2686ea7fe9fcd7b9579..ed6e1c0d4e55e3dfe207059cbd60bbb8c03a379d 100644 --- a/src/testsuite.in +++ b/src/testsuite.in @@ -1,10 +1,20 @@ +test_compile_error(int foo() { LJjjjjJJJ ; }) +test_true(clone(class { constant i=1; })->i) +test_true(clone(class { constant i=0; mixed `->(string s) { if(s=="i") return 1; }})->i) +test_true(clone(class { constant i=1; mixed `->(string s) { return 0; }})["i"]) +test_true(clone(class { constant i=0; mixed `[](string s) { if(s=="i") return 1; }})["i"]) +test_true(clone(class { mixed `[]=(mixed a, mixed b) { if(a!=b) throw(1); }})[1]=1) +test_true(clone(class { mixed `->=(mixed a, mixed b) { if(a!=b) throw(1); }})->i="i") + +test_compile(class A {}; class B { inherit A; }) + test_true(mappingp(_memory_usage())) test_true(objectp( _next(this_object()) || _prev(this_object()))) test_true(arrayp( _next(({})) || _prev(({})))) test_any(object o=this_object(); while(o=_next(o))); test_any(object o=this_object(); while(o=_prev(o))); -test_true([["$Id: testsuite.in,v 1.15 1996/12/07 23:41:06 hubbe Exp $"]]) +test_true([["$Id: testsuite.in,v 1.16 1997/01/22 05:19:47 hubbe Exp $"]]) test_any([[object(File) o=File(); return objectp(o);]],1) test_any([[object o=Regexp("foo"); return objectp(o);]],1) test_any([[object o=Regexp("foo"); return object_program(o);]],Regexp) diff --git a/src/version.c b/src/version.c index 41fd1f63e4dffe4eb489999f60066c539c829f8e..ee9d1c6b82c6ce1f5d49053a30a7190ce0a9eaee 100644 --- a/src/version.c +++ b/src/version.c @@ -1,3 +1,9 @@ +/*\ +||| 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. +\*/ + #include "global.h" #include "svalue.h" #include "interpret.h" @@ -6,5 +12,5 @@ void f_version(INT32 args) { pop_n_elems(args); - push_text("Pike v0.4pl4"); + push_text("Pike v0.4pl5"); } diff --git a/src/version.h b/src/version.h index 301b8aad4f3f9404d1dc14515fea3481f4fa62af..f982194f24bb071287dd6efde8bd93fd4c886a7c 100644 --- a/src/version.h +++ b/src/version.h @@ -1,3 +1,9 @@ +/*\ +||| 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. +\*/ + /* Prototypes begin here */ void f_version(INT32 args); /* Prototypes end here */