diff --git a/compiler2/ttcn3/AST_ttcn3.cc b/compiler2/ttcn3/AST_ttcn3.cc index 75ec17a8a0571d31da58318cd2e57ec5d0c2c534..1ea4a48d0db0c7290442ae0cc7b516bbcb5c2ec4 100644 --- a/compiler2/ttcn3/AST_ttcn3.cc +++ b/compiler2/ttcn3/AST_ttcn3.cc @@ -6625,6 +6625,31 @@ namespace Ttcn { if (!checked) chk(); return output_type; } + + bool Def_Function_Base::is_identical(Def_Function_Base* p_other) + { + if (asstype != p_other->get_asstype()) { + return false; + } + else if (return_type != NULL && + !p_other->return_type->is_identical(return_type)) { + return false; + } + FormalParList* other_fp_list = p_other->get_FormalParList(); + if (other_fp_list->get_nof_fps() != fp_list->get_nof_fps()) { + return false; + } + for (size_t i = 0; i < fp_list->get_nof_fps(); ++i) { + FormalPar* fp1 = fp_list->get_fp_byIndex(i); + FormalPar* fp2 = other_fp_list->get_fp_byIndex(i); + if (fp1->get_asstype() != fp2->get_asstype() || + !fp1->get_Type()->is_identical(fp2->get_Type()) || + fp1->get_id().get_name() != fp2->get_id().get_name()) { + return false; + } + } + return true; + } // ================================= @@ -7475,7 +7500,7 @@ namespace Ttcn { checked = true; Error_Context cntxt(this, "In external function definition `%s'", id->get_dispname().c_str()); - if (!ext_keyword && my_scope->get_scope_class()->is_external()) { + if (!ext_keyword && !my_scope->get_scope_class()->is_external()) { error("Missing function body or `external' keyword"); } fp_list->chk(asstype); @@ -8087,59 +8112,6 @@ namespace Ttcn { // TODO: with attributes } - void Def_AbsFunction::chk_implementation(Common::Assignment* p_ass, Location* p_loc) - { - switch (p_ass->get_asstype()) { - case A_FUNCTION: - case A_FUNCTION_RVAL: - case A_FUNCTION_RTEMP: { - Def_Function* def_func = dynamic_cast<Def_Function*>(p_ass); - if (def_func == NULL) { - // it's an abstract function, which means it hasn't been implemented - p_loc->error("Missing implementation of abstract method `%s'", - get_fullname().c_str()); - } - else { - // check whether they're identical - bool match = true; - if (asstype != p_ass->get_asstype()) { - match = false; - } - else if (return_type != NULL && - !def_func->get_return_type()->is_identical(return_type)) { - match = false; - } - else { - FormalParList* other_fp_list = def_func->get_FormalParList(); - if (other_fp_list->get_nof_fps() != fp_list->get_nof_fps()) { - match = false; - } - else { - for (size_t i = 0; i < fp_list->get_nof_fps(); ++i) { - FormalPar* fp1 = fp_list->get_fp_byIndex(i); - FormalPar* fp2 = other_fp_list->get_fp_byIndex(i); - if (fp1->get_asstype() != fp2->get_asstype() || - !fp1->get_Type()->is_identical(fp2->get_Type()) || - fp1->get_id().get_name() != fp2->get_id().get_name()) { - match = false; - } - } - } - } - if (!match) { - p_ass->error("The prototype of method `%s' is not identical to that " - "of inherited abstract method `%s'", - def_func->get_id().get_dispname().c_str(), get_fullname().c_str()); - } - } - break; } - default: - p_ass->error("%s shadows inherited abstract method `%s'", - p_ass->get_description().c_str(), get_fullname().c_str()); - break; - } - } - void Def_AbsFunction::generate_code(output_struct* target, bool) { const string& t_genname = get_genname(); diff --git a/compiler2/ttcn3/AST_ttcn3.hh b/compiler2/ttcn3/AST_ttcn3.hh index f72f0825238ecaecda6e1a660992c7629384474a..9341701f39d9715fc6bfd9561ec812cc3fb48662 100644 --- a/compiler2/ttcn3/AST_ttcn3.hh +++ b/compiler2/ttcn3/AST_ttcn3.hh @@ -1390,6 +1390,7 @@ namespace Ttcn { Type *get_output_type(); Type* get_return_type() const { return return_type; } //virtual Type *get_RunsOnType(); + bool is_identical(Def_Function_Base* p_other); template_restriction_t get_template_restriction() { return template_restriction; } /** Checks and returns whether the function is startable. @@ -1620,7 +1621,6 @@ namespace Ttcn { virtual ~Def_AbsFunction(); virtual Definition* clone() const; virtual void chk(); - void chk_implementation(Common::Assignment* p_ass, Location* p_loc); virtual void generate_code(output_struct* target, bool clean_up = false); }; diff --git a/compiler2/ttcn3/Ttcnstuff.cc b/compiler2/ttcn3/Ttcnstuff.cc index 1a1355fca76a684c380c35d915833e110e5b1bfb..bc17b0353b326014ab366172006b6c7585d8776c 100644 --- a/compiler2/ttcn3/Ttcnstuff.cc +++ b/compiler2/ttcn3/Ttcnstuff.cc @@ -3200,7 +3200,7 @@ namespace Ttcn { if (members->has_local_ass_withId(p_id)) { return true; } - if (base_type != NULL) { + if (base_class != NULL) { return base_class->has_local_ass_withId(p_id); } else { @@ -3415,7 +3415,56 @@ namespace Ttcn { } } - if (constructor == NULL) { + bool name_clash = false; + if (base_class != NULL) { + for (size_t i = 0; i < members->get_nof_asss(); ++i) { + Common::Assignment* local_def = members->get_ass_byIndex(i, false); + const Common::Identifier& local_id = local_def->get_id(); + if (local_def->get_asstype() != Common::Assignment::A_CONSTRUCTOR && + base_class->has_local_ass_withId(local_id)) { + Common::Assignment* base_def = base_class->get_local_ass_byId(local_id); + switch (local_def->get_asstype()) { + case Common::Assignment::A_FUNCTION: + case Common::Assignment::A_FUNCTION_RVAL: + case Common::Assignment::A_FUNCTION_RTEMP: + case Common::Assignment::A_EXT_FUNCTION: + case Common::Assignment::A_EXT_FUNCTION_RVAL: + case Common::Assignment::A_EXT_FUNCTION_RTEMP: + switch (base_def->get_asstype()) { + case Common::Assignment::A_FUNCTION: + case Common::Assignment::A_FUNCTION_RVAL: + case Common::Assignment::A_FUNCTION_RTEMP: + case Common::Assignment::A_EXT_FUNCTION: + case Common::Assignment::A_EXT_FUNCTION_RVAL: + case Common::Assignment::A_EXT_FUNCTION_RTEMP: { + Def_Function_Base* local_func = dynamic_cast<Def_Function_Base*>(local_def); + Def_Function_Base* base_func = dynamic_cast<Def_Function_Base*>(base_def); + if (!local_func->is_identical(base_func)) { + local_def->error("The prototype of method `%s' is not identical " + "to that of inherited method `%s'", + local_id.get_dispname().c_str(), base_def->get_fullname().c_str()); + } + break; } + default: + local_def->error("%s shadows inherited member `%s'", + local_def->get_description().c_str(), base_def->get_fullname().c_str()); + name_clash = true; + break; + } + break; + default: + local_def->error("%s shadows inherited %s `%s'", + local_def->get_description().c_str(), + dynamic_cast<Def_Function_Base*>(base_def) != NULL ? "method" : "member", + base_def->get_fullname().c_str()); + name_clash = true; + break; + } + } + } + } + + if (constructor == NULL && !name_clash) { // create a default constructor Reference* base_call = NULL; FormalParList* fp_list = NULL; @@ -3521,10 +3570,7 @@ namespace Ttcn { Def_AbsFunction* def_abs_func = dynamic_cast<Def_AbsFunction*>(ass); if (def_abs_func != NULL) { const string& def_name = def_abs_func->get_id().get_name(); - if (abstract_functions.has_key(def_name)) { - // TODO - } - else { + if (!abstract_functions.has_key(def_name)) { abstract_functions.add(def_name, def_abs_func); } } @@ -3539,10 +3585,25 @@ namespace Ttcn { // all abstract methods from the base class have to be implemented in this class for (size_t i = 0; i < base_class->abstract_functions.size(); ++i) { Def_AbsFunction* def_abs_func = base_class->abstract_functions.get_nth_elem(i); - def_abs_func->chk_implementation(get_local_ass_byId(def_abs_func->get_id()), this); + Common::Assignment* ass = get_local_ass_byId(def_abs_func->get_id()); + switch (ass->get_asstype()) { + case Common::Assignment::A_FUNCTION: + case Common::Assignment::A_FUNCTION_RVAL: + case Common::Assignment::A_FUNCTION_RTEMP: { + if (dynamic_cast<Def_AbsFunction*>(ass) != NULL) { + error("Missing implementation of abstract method `%s'", + def_abs_func->get_fullname().c_str()); + } + // whether the new function is identical to the abstract one has + // already been checked + break; } + default: + // it's either an external function (which is OK), or + // it's shadowed by a member (error has already been reported) + break; + } } } - // todo: name clashes with defs in the base class? } void ClassTypeBody::chk_recursions(ReferenceChain& refch) diff --git a/compiler2/ttcn3/compiler.y b/compiler2/ttcn3/compiler.y index 046d41c420c4626cffcc2e6020f7d461d5e8f7be..c100f7e74d3a1e78167d5dc0d47248cdc71b8e6c 100644 --- a/compiler2/ttcn3/compiler.y +++ b/compiler2/ttcn3/compiler.y @@ -3578,6 +3578,7 @@ optAbstractModifier: optExtendsClassDef: /* empty */ { $$ = NULL; } | ExtendsKeyword ReferencedType { $$ = $2; } +| ExtendsKeyword ObjectKeyword { $$ = NULL; } ; optFinallyDef: diff --git a/function_test/Semantic_Analyser/oop/oop_SE.ttcn b/function_test/Semantic_Analyser/oop/oop_SE.ttcn index 9fdeb7c3a209ceaa043fa3ede9e2db5ace8e9547..063482daae89b416ab9a19ea041f7c0995fec02f 100644 --- a/function_test/Semantic_Analyser/oop/oop_SE.ttcn +++ b/function_test/Semantic_Analyser/oop/oop_SE.ttcn @@ -114,15 +114,11 @@ type class C6 { //^In type definition// } -type class C7 extends C9 { //^In type definition// //^In superclass definition// - public const integer x1 := 1; -} +type class C7 extends C9 { } //^In type definition// //^In superclass definition// -type class C8 extends C7 { //^In type definition// //While checking embedded recursions\: Circular reference\: `@oop_SE.C8' -> `@oop_SE.C8.<superclass>' -> `@oop_SE.C7' -> `@oop_SE.C7.<superclass>' -> `@oop_SE.C9' -> `@oop_SE.C9.<superclass>' -> `@oop_SE.C8'// - private var integer x2; -} +type class C8 extends C7 { } //^In type definition// //While checking embedded recursions\: Circular reference\: `@oop_SE.C8' -> `@oop_SE.C8.<superclass>' -> `@oop_SE.C7' -> `@oop_SE.C7.<superclass>' -> `@oop_SE.C9' -> `@oop_SE.C9.<superclass>' -> `@oop_SE.C8'// -type class C9 extends C8 {} //^In type definition// //^In superclass definition// +type class C9 extends C8 { } //^In type definition// //^In superclass definition// external const C0 ec_c0; //^In external constant definition// //External constant cannot be defined for class type `@oop_SE.C0'// external const object ec_obj; //^In external constant definition// //External constant cannot be defined for class type `object'// @@ -343,42 +339,42 @@ function f_inst_abstract() { //^In function definition// } type class @abstract C25 { - public function @abstract f_abs() return integer; + public function @abstract f_abs(inout template integer p) return integer; } type class @abstract C26 extends C25 { - public function @abstract f_abs2(inout template integer p); + public function @abstract f_abs2(); } type class C27 extends C26 { } //^In type definition// //Missing implementation of abstract method `@oop_SE.C25.f_abs'// //Missing implementation of abstract method `@oop_SE.C26.f_abs2'// type class C28 extends C25 { //^In type definition// - public var integer f_abs; //variable `@oop_SE.C28.f_abs' shadows inherited abstract method `@oop_SE.C25.f_abs'// + public var integer f_abs; //variable `@oop_SE.C28.f_abs' shadows inherited method `@oop_SE.C25.f_abs'// } type class C29 extends C26 { //^In type definition// //Missing implementation of abstract method `@oop_SE.C25.f_abs'// - public function f_abs2(inout template integer p) { } + public function f_abs2() { } } type class C30 extends C26 { - public function f_abs() return integer { return 1; } - public function f_abs2(inout template integer p) { } + public function f_abs(inout template integer p) return integer { return 1; } + public function f_abs2() { } } type class C31 extends C25 { //^In type definition// - public function f_abs(in template integer p) { } //The prototype of method `f_abs' is not identical to that of inherited abstract method `@oop_SE.C25.f_abs'// + public function f_abs(in template integer p) { } //The prototype of method `f_abs' is not identical to that of inherited method `@oop_SE.C25.f_abs'// } type class C32 extends C25 { //^In type definition// - public function f_abs(inout integer p) { } //The prototype of method `f_abs' is not identical to that of inherited abstract method `@oop_SE.C25.f_abs'// + public function f_abs(inout integer p) { } //The prototype of method `f_abs' is not identical to that of inherited method `@oop_SE.C25.f_abs'// } type class C33 extends C25 { //^In type definition// - public function f_abs(inout template integer q) { } //The prototype of method `f_abs' is not identical to that of inherited abstract method `@oop_SE.C25.f_abs'// + public function f_abs(inout template integer q) { } //The prototype of method `f_abs' is not identical to that of inherited method `@oop_SE.C25.f_abs'// } type class C34 extends C25 { //^In type definition// - public function f_abs(inout template integer p) return boolean { return false; } //The prototype of method `f_abs' is not identical to that of inherited abstract method `@oop_SE.C25.f_abs'// + public function f_abs(inout template integer p) return boolean { return false; } //The prototype of method `f_abs' is not identical to that of inherited method `@oop_SE.C25.f_abs'// } type class @abstract C35 { @@ -433,8 +429,8 @@ function f_default() { //^In function definition// } -type external class C40 { //^In type definition// - function f1(in integer p) return octetstring; //^In external function definition// //Missing function body or `external' keyword// +type external class C40 { + function f1(in integer p) return octetstring; external function f2(); } @@ -459,6 +455,26 @@ type class C44 extends C40 { external function f(); } +type class C45 { //^In type definition// + function f(); //^In external function definition// //Missing function body or `external' keyword// +} + + +type class C46 extends object { } + + +type class C47 { + const integer m1 := 3; + function m2() return boolean { return true; } + function m3(in integer p) return charstring { return int2str(p); } +} + +type class C48 extends C47 { //^In type definition// + var template octetstring m1; //template variable `@oop_SE.C48.m1' shadows inherited member `@oop_SE.C47.m1'// + var integer m2; //variable `@oop_SE.C48.m2' shadows inherited method `@oop_SE.C47.m2'// + function m3() return charstring { return "1"; } //The prototype of method `m3' is not identical to that of inherited method `@oop_SE.C47.m3'// +} + control { //^In control part// var C11 x := C11.create; //^In variable definition// //A definition without `runs on' clause cannot create a value of class type `@oop_SE.C11', which runs on component type `@oop_SE.CT_RunsOn'// //Cannot create value of class type `@oop_SE.C11', which has an `mtc' clause, in the control part.// //Cannot create value of class type `@oop_SE.C11', which has a `system' clause, in the control part.//