diff --git a/compiler2/ttcn3/Statement.cc b/compiler2/ttcn3/Statement.cc
index fb922fd6a808ca533844bb25569ae03342b8e155..5622e8b44cd2d7b9487cb219dee4bb2a2ed04527 100644
--- a/compiler2/ttcn3/Statement.cc
+++ b/compiler2/ttcn3/Statement.cc
@@ -588,115 +588,11 @@ namespace Ttcn {
       str = mputstr(str, "TTCN_TryBlock try_block;\n");
     }
     if (finally_block != NULL) {
-      string tmp_id = get_scope_mod_gen()->get_temporary_id();
 #if __cplusplus < 201103L
-      // the finally block's code is generated into the destructor of a newly created class;
-      // all local definitions, parameters and the TTCN_Location object are added to the class as members,
-      // so the destructor can reach them
-      str = mputprintf(str,
-        "class %s_finally {\n"
-        "public:\n", tmp_id.c_str());
-      
-      if (include_location_info) {
-        str = mputstr(str, "TTCN_Location& current_location;\n");
-      }
-      for (size_t i = 0; i < finally_block->refd_local_defs.size(); ++i) {
-        bool is_const = false;
-        bool is_template = false;
-        Common::Assignment* def = finally_block->refd_local_defs.get_nth_key(i);
-        switch (def->get_asstype()) {
-        case Common::Assignment::A_PAR_TEMPL_IN:
-        case Common::Assignment::A_TEMPLATE:
-          is_template = true;
-          // fall through
-        case Common::Assignment::A_PAR_VAL:
-        case Common::Assignment::A_PAR_VAL_IN:
-        case Common::Assignment::A_CONST:
-          is_const = true;
-          break;
-        case Common::Assignment::A_VAR_TEMPLATE:
-        case Common::Assignment::A_PAR_TEMPL_INOUT:
-        case Common::Assignment::A_PAR_TEMPL_OUT:
-          is_template = true;
-          break;
-        default:
-          break;
-        }
-        str = mputprintf(str, "%s%s& %s;\n",
-          is_const ? "const " : "",
-          is_template ? def->get_Type()->get_genname_template(my_sb).c_str() : def->get_Type()->get_genname_value(my_sb).c_str(),
-          def->get_id().get_name().c_str());
-      }
-      if (include_location_info || finally_block->refd_local_defs.size() > 0) {
-        str = mputprintf(str, "%s_finally(", tmp_id.c_str());
-        if (include_location_info) {
-          str = mputstr(str, "TTCN_Location& p_loc");
-        }
-        for (size_t i = 0; i < finally_block->refd_local_defs.size(); ++i) {
-          bool is_const = false;
-          bool is_template = false;
-          Common::Assignment* def = finally_block->refd_local_defs.get_nth_key(i);
-          switch (def->get_asstype()) {
-          case Common::Assignment::A_PAR_TEMPL_IN:
-          case Common::Assignment::A_TEMPLATE:
-            is_template = true;
-            // fall through
-          case Common::Assignment::A_PAR_VAL:
-          case Common::Assignment::A_PAR_VAL_IN:
-          case Common::Assignment::A_CONST:
-            is_const = true;
-            break;
-          case Common::Assignment::A_VAR_TEMPLATE:
-          case Common::Assignment::A_PAR_TEMPL_INOUT:
-          case Common::Assignment::A_PAR_TEMPL_OUT:
-            is_template = true;
-            break;
-          default:
-            break;
-          }
-          str = mputprintf(str, "%s%s%s& p_%s",
-            include_location_info || i > 0 ? ", " : "", is_const ? "const " : "",
-            is_template ? def->get_Type()->get_genname_template(my_sb).c_str() : def->get_Type()->get_genname_value(my_sb).c_str(),
-            def->get_id().get_name().c_str());
-        }
-        str = mputstr(str, "): ");
-        if (include_location_info) {
-          str = mputstr(str, "current_location(p_loc)");
-        }
-        for (size_t i = 0; i < finally_block->refd_local_defs.size(); ++i) {
-          Common::Assignment* def = finally_block->refd_local_defs.get_nth_key(i);
-          str = mputprintf(str, "%s%s(p_%s)", include_location_info || i > 0 ? ", " : "",
-            def->get_id().get_name().c_str(), def->get_id().get_name().c_str());
-        }
-        str = mputstr(str, " { }\n");
-      }
-      str = mputprintf(str,
-        "~%s_finally() {\n"
-        "try {\n", tmp_id.c_str());
-      str = finally_block->generate_code(str, def_glob_vars, src_glob_vars);
-      str = mputprintf(str,
-        "} catch (...) {\n"
-        "fprintf(stderr, \"Unhandled exception or dynamic test case error in a finally block. Terminating application.\\n\");\n"
-        "exit(EXIT_FAILURE);\n"
-        "}\n"
-        "}\n"
-        "};\n"
-        "%s_finally %s",
-        tmp_id.c_str(), tmp_id.c_str());
-      if (include_location_info || finally_block->refd_local_defs.size() > 0) {
-        str = mputc(str, '(');
-        if (include_location_info) {
-          str = mputstr(str, "current_location");
-        }
-        for (size_t i = 0; i < finally_block->refd_local_defs.size(); ++i) {
-          str = mputprintf(str, "%s%s", include_location_info || i > 0 ? ", " : "",
-            finally_block->refd_local_defs.get_nth_key(i)->get_id().get_name().c_str());
-        }
-        str = mputc(str, ')');
-      }
-      str = mputstr(str, ";\n");
+      str = finally_block->generate_code_finally(str, def_glob_vars, src_glob_vars);
 #else
       // C++11 version:
+      string tmp_id = get_scope_mod_gen()->get_temporary_id();
       str = mputprintf(str,
         "FINALLY %s([&] {\n", tmp_id.c_str());
       str = finally_block->generate_code(str, def_glob_vars, src_glob_vars);
@@ -747,6 +643,125 @@ namespace Ttcn {
     }
     return str;
   }
+  
+  char* StatementBlock::generate_code_finally(char* str, char*& def_glob_vars, char*& src_glob_vars)
+  {
+    // the finally block's code is generated into the destructor of a newly created class;
+    // all local definitions, parameters and the TTCN_Location object are added to the class as members,
+    // so the destructor can reach them
+    if (exception_handling != EH_OOP_FINALLY) {
+      FATAL_ERROR("StatementBlock::generate_code_finally");
+    }
+    string tmp_id = get_scope_mod_gen()->get_temporary_id();
+    str = mputprintf(str,
+      "class %s_finally {\n"
+      "public:\n", tmp_id.c_str());
+    
+    if (include_location_info) {
+      str = mputstr(str, "TTCN_Location& current_location;\n");
+    }
+    bool* is_const = NULL;
+    bool* is_template = NULL;
+    bool* is_shadowed = NULL;
+    if (refd_local_defs.size() > 0) {
+      is_const = new bool[refd_local_defs.size()];
+      is_template = new bool[refd_local_defs.size()];
+      is_shadowed = new bool[refd_local_defs.size()];
+    }
+    for (size_t i = 0; i < refd_local_defs.size(); ++i) {
+      is_const[i] = false;
+      is_template[i] = false;
+      is_shadowed[i] = false;
+      Common::Assignment* def = refd_local_defs.get_nth_key(i);
+      switch (def->get_asstype()) {
+      case Common::Assignment::A_PAR_TEMPL_IN:
+        is_template[i] = true;
+        // fall through
+      case Common::Assignment::A_PAR_VAL:
+      case Common::Assignment::A_PAR_VAL_IN: {
+        FormalPar* fpar = dynamic_cast<FormalPar*>(def);
+        if (fpar == NULL) {
+          FATAL_ERROR("StatementBlock::generate_code");
+        }
+        if (fpar->get_used_as_lvalue()) {
+          is_shadowed[i] = true;
+        }
+        else {
+          is_const[i] = true;
+        }
+        break; }
+      case Common::Assignment::A_CONST:
+        is_const[i] = true;
+        break;
+      case Common::Assignment::A_TEMPLATE:
+        is_const[i] = true;
+        // fall through
+      case Common::Assignment::A_VAR_TEMPLATE:
+      case Common::Assignment::A_PAR_TEMPL_INOUT:
+      case Common::Assignment::A_PAR_TEMPL_OUT:
+        is_template[i] = true;
+        break;
+      default:
+        break;
+      }
+      str = mputprintf(str, "%s%s& %s%s;\n",
+        is_const[i] ? "const " : "",
+        is_template[i] ? def->get_Type()->get_genname_template(my_sb).c_str() : def->get_Type()->get_genname_value(my_sb).c_str(),
+        def->get_id().get_name().c_str(), is_shadowed[i] ? "_shadow" : "");
+    }
+    if (include_location_info || refd_local_defs.size() > 0) {
+      str = mputprintf(str, "%s_finally(", tmp_id.c_str());
+      if (include_location_info) {
+        str = mputstr(str, "TTCN_Location& p_loc");
+      }
+      for (size_t i = 0; i < refd_local_defs.size(); ++i) {
+        Common::Assignment* def = refd_local_defs.get_nth_key(i);
+        str = mputprintf(str, "%s%s%s& p_%s",
+          include_location_info || i > 0 ? ", " : "", is_const[i] ? "const " : "",
+          is_template[i] ? def->get_Type()->get_genname_template(my_sb).c_str() : def->get_Type()->get_genname_value(my_sb).c_str(),
+          def->get_id().get_name().c_str());
+      }
+      str = mputstr(str, "): ");
+      if (include_location_info) {
+        str = mputstr(str, "current_location(p_loc)");
+      }
+      for (size_t i = 0; i < refd_local_defs.size(); ++i) {
+        Common::Assignment* def = refd_local_defs.get_nth_key(i);
+        str = mputprintf(str, "%s%s%s(p_%s)", include_location_info || i > 0 ? ", " : "",
+          def->get_id().get_name().c_str(), is_shadowed[i] ? "_shadow" : "", def->get_id().get_name().c_str());
+      }
+      str = mputstr(str, " { }\n");
+    }
+    str = mputprintf(str,
+      "~%s_finally() {\n"
+      "try {\n", tmp_id.c_str());
+    str = generate_code(str, def_glob_vars, src_glob_vars);
+    str = mputprintf(str,
+      "} catch (...) {\n"
+      "fprintf(stderr, \"Unhandled exception or dynamic test case error in a finally block. Terminating application.\\n\");\n"
+      "exit(EXIT_FAILURE);\n"
+      "}\n"
+      "}\n"
+      "};\n"
+      "%s_finally %s",
+      tmp_id.c_str(), tmp_id.c_str());
+    if (include_location_info || refd_local_defs.size() > 0) {
+      str = mputc(str, '(');
+      if (include_location_info) {
+        str = mputstr(str, "current_location");
+      }
+      for (size_t i = 0; i < refd_local_defs.size(); ++i) {
+        Common::Assignment* def = refd_local_defs.get_nth_key(i);
+        str = mputprintf(str, "%s%s%s", include_location_info || i > 0 ? ", " : "",
+          def->get_id().get_name().c_str(), is_shadowed[i] ? "_shadow" : "");
+      }
+      str = mputc(str, ')');
+    }
+    delete[] is_const;
+    delete[] is_template;
+    delete[] is_shadowed;
+    return mputstr(str, ";\n");
+  }
 
   void StatementBlock::ilt_generate_code(ILT *ilt)
   {
diff --git a/compiler2/ttcn3/Statement.hh b/compiler2/ttcn3/Statement.hh
index 5571323964e0e0ae19454a025bad6b5e5c210c36..ee8e16ccd59a433705acf173d1d52479377681dc 100644
--- a/compiler2/ttcn3/Statement.hh
+++ b/compiler2/ttcn3/Statement.hh
@@ -182,6 +182,7 @@ namespace Ttcn {
       * '@update' statement. 
       * Also generates code for the alt-guards if this is the statement block of an altstep. */
     char* generate_code(char *str, char*& def_glob_vars, char*& src_glob_vars, AltGuards* alt_guards = NULL);
+    char* generate_code_finally(char* str, char*& def_glob_vars, char*& src_glob_vars);
     void ilt_generate_code(ILT *ilt);
 
     virtual void set_parent_path(WithAttribPath* p_path);
diff --git a/regression_test/oop/exceptions.ttcn b/regression_test/oop/exceptions.ttcn
index a47b3ef6b7ce0b878f38f80138becc970b8d0a78..2a6319a4c3eb1cfbf03dcb33064aacc2e0444daf 100644
--- a/regression_test/oop/exceptions.ttcn
+++ b/regression_test/oop/exceptions.ttcn
@@ -45,8 +45,9 @@ type component CT {
 type integer SmallNumber (-100..100);
 
 
-function test_block(in integer p_exception_type) runs on CT {
+function test_block(in integer p_exception_type, in charstring p_dummy := "a") runs on CT {
   var integer v_finally_increment := 1;
+  p_dummy := "abc";
   {
     select (p_exception_type) {
       case (1) {
@@ -119,6 +120,7 @@ function test_block(in integer p_exception_type) runs on CT {
   }
   finally {
     cv_finally_counter := cv_finally_counter + v_finally_increment; // test referencing a local variable (in the main block)
+    log(p_dummy); // test referencing a shadowed function parameter (in the main block)
     if (cv_finally_counter > 0) {
       log(p_exception_type); // test referencing a function parameter (in a sub-block)
     }